Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feature/browser #1029

Merged
merged 18 commits into from
Jan 28, 2025
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
18 commits
Select commit Hold shift + click to select a range
29ae9f7
feat: add BrowserScreen and integrate MetaMask webview
ArturHoncharuk Jan 20, 2025
20190d5
feat: enhance BrowserScreen with wallet integration and message handling
ArturHoncharuk Jan 20, 2025
81392fa
refactor: remove deprecated injected JavaScript and enhance createAMB…
ArturHoncharuk Jan 22, 2025
1ba39bb
refactor: update createAMBProvider to deprecate networkUrl parameter
ArturHoncharuk Jan 22, 2025
c5285e6
feat: enhance BrowserScreen with wallet permissions and request handling
ArturHoncharuk Jan 22, 2025
9e8cdf2
feat: enhance BrowserScreen with console logging and message signing …
ArturHoncharuk Jan 22, 2025
1b37470
feat: enhance BrowserScreen with Ethereum state management and permis…
ArturHoncharuk Jan 22, 2025
aceeb36
feat: enhance injected provider with improved event handling and stat…
ArturHoncharuk Jan 22, 2025
e41a58a
fix: update AMB Wallet icon in injectable provider
ArturHoncharuk Jan 22, 2025
cfcafc6
feat: enhance injectable provider and BrowserScreen with improved sta…
ArturHoncharuk Jan 23, 2025
406bef6
feat: enhance injectable provider with dynamic UUID and EIP-6963 inte…
ArturHoncharuk Jan 23, 2025
fef3387
refactor: refactor browser logic
EvgeniyJB Jan 23, 2025
4d4f33f
feat: enhance browser feature structure and refactor middleware
ArturHoncharuk Jan 23, 2025
4133c51
fix: update import paths
ArturHoncharuk Jan 23, 2025
d710675
fix: correct import paths in middleware.helpers.ts
ArturHoncharuk Jan 24, 2025
58cacf7
hotfix: change browser types file
EvgeniyJB Jan 28, 2025
e11e80a
Merge branch 'dev' into feature/browser
EvgeniyJB Jan 28, 2025
ca9b9ab
hotfix: hotfix app crush
EvgeniyJB Jan 28, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
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
Loading