Skip to content

Commit

Permalink
Merge pull request #1029 from ambrosus/feature/browser
Browse files Browse the repository at this point in the history
Feature/browser
  • Loading branch information
EvgeniyJB authored Jan 28, 2025
2 parents 6bb935e + ca9b9ab commit 87abb61
Show file tree
Hide file tree
Showing 22 changed files with 1,026 additions and 6 deletions.
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
"@babel/preset-env": "^7.21.5",
"@craftzdog/react-native-buffer": "^6.0.5",
"@ethersproject/shims": "^5.7.0",
"@metamask/react-native-webview": "^14.0.4",
"@metamask/safe-event-emitter": "^3.0.0",
"@morrowdigital/watermelondb-expo-plugin": "^2.3.1",
"@mymonero/mymonero-paymentid-utils": "^3.0.0",
Expand Down
1 change: 1 addition & 0 deletions src/appTypes/navigation/wallets.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ export type HomeParamsList = {
};
Wallets: { screen: string };
Harbor: undefined;
BrowserScreen: undefined;
CreateWalletStep2: undefined;
} & CommonStackParamsList;

Expand Down
9 changes: 9 additions & 0 deletions src/entities/browser/model/browser-store.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import { create } from 'zustand';
import { BrowserStoreModel } from '@entities/harbor/model/types';

export const useBrowserStore = create<BrowserStoreModel>((set) => ({
connectedAddress: '',
setConnectedAddress: async (address: string) => {
set({ connectedAddress: address });
}
}));
1 change: 1 addition & 0 deletions src/entities/browser/model/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export * from './browser-store';
11 changes: 11 additions & 0 deletions src/entities/browser/model/types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
export interface BrowserStoreModel {
connectedAddress: string;
setConnectedAddress: (address: string) => void;
}

export interface BrowserConfig {
id: string;
name: string;
url: string;
isAirDaoApp: string;
}
1 change: 1 addition & 0 deletions src/features/browser/constants/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export * from './rpc';
2 changes: 2 additions & 0 deletions src/features/browser/constants/rpc.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export const AMB_CHAIN_ID_HEX = '0x414e';
export const AMB_CHAIN_ID_DEC = 16718;
9 changes: 9 additions & 0 deletions src/features/browser/lib/eip6963.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import Constants from 'expo-constants';

const appSlug = Constants.expoConfig?.name;

export const EIP6963_PROVIDER_INFO = {
name: `${appSlug} Wallet`,
icon: '',
rdns: Constants.expoConfig?.ios?.bundleIdentifier
} as const;
5 changes: 5 additions & 0 deletions src/features/browser/lib/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
export * from './injectable.provider';
export * from './rpc-middleware';
export * from './rpc-methods';
export * from './eip6963';
export * from './middleware.helpers';
227 changes: 227 additions & 0 deletions src/features/browser/lib/injectable.provider.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,227 @@
import { randomUUID } from 'expo-crypto';
import { AMB_CHAIN_ID_DEC, AMB_CHAIN_ID_HEX } from '../constants';
import { EIP6963_PROVIDER_INFO } from './eip6963';

const uuid = randomUUID;

export const INJECTED_PROVIDER_JS = `
(function() {
if (window.ethereum) return true;
let isInitialized = false;
let requestCounter = 0;
const pendingRequests = new Map();
let lastEmittedState = {
address: null,
chainId: null,
connected: false
};
// EIP-6963
const eip6963ProviderInfo = {
uuid: '${uuid()}',
name: '${EIP6963_PROVIDER_INFO.name}',
icon: '${EIP6963_PROVIDER_INFO.icon}',
rdns: '${EIP6963_PROVIDER_INFO.rdns}'
};
// Create the provider object
const provider = {
isMetaMask: true,
selectedAddress: null,
chainId: '${AMB_CHAIN_ID_HEX}',
networkVersion: ${AMB_CHAIN_ID_DEC},
_events: new Map(),
isConnected: () => true,
_metamask: {
isUnlocked: () => true,
},
request: function(args) {
return new Promise((resolve, reject) => {
const { method, params } = args;
const id = requestCounter++;
// Special handling for eth_requestAccounts on page refresh
if (method === 'eth_requestAccounts' && this.selectedAddress) {
return resolve([this.selectedAddress]);
}
pendingRequests.set(id, { resolve, reject });
window.ReactNativeWebView.postMessage(JSON.stringify({
id,
jsonrpc: '2.0',
method,
params: params || []
}));
});
},
on: function(eventName, callback) {
if (!this._events.has(eventName)) {
this._events.set(eventName, new Set());
}
this._events.get(eventName).add(callback);
// Only emit initial state if it's different from last emitted
if (this.selectedAddress && this.selectedAddress !== lastEmittedState.address) {
switch(eventName) {
case 'connect':
if (!lastEmittedState.connected) {
callback({ chainId: this.chainId });
lastEmittedState.connected = true;
}
break;
case 'accountsChanged':
if (this.selectedAddress !== lastEmittedState.address) {
callback([this.selectedAddress]);
lastEmittedState.address = this.selectedAddress;
}
break;
case 'chainChanged':
if (this.chainId !== lastEmittedState.chainId) {
callback(this.chainId);
lastEmittedState.chainId = this.chainId;
}
break;
}
}
return () => this._events.get(eventName).delete(callback);
},
emit: function(eventName, data) {
// Prevent duplicate emissions
switch(eventName) {
case 'connect':
if (lastEmittedState.connected) return;
lastEmittedState.connected = true;
break;
case 'accountsChanged':
const newAddress = Array.isArray(data) ? data[0] : null;
if (newAddress === lastEmittedState.address) return;
lastEmittedState.address = newAddress;
break;
case 'chainChanged':
if (data === lastEmittedState.chainId) return;
lastEmittedState.chainId = data;
break;
}
const listeners = this._events.get(eventName);
if (listeners) {
listeners.forEach(listener => {
try {
listener(data);
} catch(e) {
console.error('Event listener error:', e);
}
});
}
},
removeListener: function(eventName, callback) {
if (this._events.has(eventName)) {
this._events.get(eventName).delete(callback);
}
},
enable: function() {
return this.request({ method: 'eth_requestAccounts' });
},
info: eip6963ProviderInfo,
announceProvider: function() {
window.dispatchEvent(
new CustomEvent('eip6963:announceProvider', {
detail: {
info: this.info,
provider: this
}
})
);
}
};
window.addEventListener('message', function(event) {
try {
const response = JSON.parse(event.data);
const { id, result, error } = response;
const pendingRequest = pendingRequests.get(id);
if (pendingRequest) {
pendingRequests.delete(id);
if (error) {
pendingRequest.reject(new Error(error.message));
} else {
if (response.method === 'eth_requestAccounts' || response.method === 'eth_accounts') {
provider.selectedAddress = result[0];
const listeners = provider._events.get('accountsChanged');
if (listeners) {
listeners.forEach(listener => listener(result));
}
}
pendingRequest.resolve(result);
}
}
} catch (error) {
console.error('Failed to process message:', error);
}
});
window.ethereum = provider;
if (!isInitialized) {
isInitialized = true;
provider.announceProvider();
window.addEventListener('eip6963:requestProvider', function() {
provider.announceProvider();
});
window.dispatchEvent(new Event('ethereum#initialized'));
}
return true;
})();
`;

export const REVOKE_PERMISSIONS_JS = `
(function() {
try {
if (window.ethereum) {
window.ethereum.selectedAddress = null;
window.ethereum.emit('accountsChanged', []);
console.log('Permissions revoked');
}
} catch(e) {
console.error('Revoke permissions error:', e);
}
return true;
})();
`;

export const UPDATE_ETHEREUM_STATE_JS = (address: string, chainId: string) => `
(function() {
try {
if (window.ethereum) {
window.ethereum.selectedAddress = '${address}';
window.ethereum.chainId = '${chainId}';
window.ethereum.emit('connect', { chainId: '${chainId}' });
window.ethereum.emit('chainChanged', '${chainId}');
window.ethereum.emit('accountsChanged', ['${address}']);
console.log('Ethereum state updated:', {
address: '${address}',
chainId: '${chainId}'
});
}
} catch(e) {
console.error('State update error:', e);
}
return true;
})();
`;
Loading

0 comments on commit 87abb61

Please sign in to comment.