@getpara/cosmos-wallet-connectors 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,3 @@
1
+ export { CosmosExternalWalletContext, CosmosExternalWalletProvider } from './providers/CosmosExternalWalletContext.js';
2
+ export { ParaCosmosProvider } from './providers/ParaCosmosContext.js';
3
+ export * from './wallets/connectors/index.js';
package/dist/index.js ADDED
@@ -0,0 +1,3 @@
1
+ export { CosmosExternalWalletContext, CosmosExternalWalletProvider } from './providers/CosmosExternalWalletContext.js';
2
+ export { ParaCosmosProvider } from './providers/ParaCosmosContext.js';
3
+ export * from './wallets/connectors/index.js';
@@ -0,0 +1,29 @@
1
+ import { ReactNode } from 'react';
2
+ import { CommonChain, CommonWallet } from '../types/CommonTypes.js';
3
+ import ParaWeb from '@getpara/react-sdk';
4
+ export declare const defaultCosmosExternalWallet: {
5
+ wallets: any[];
6
+ chains: any[];
7
+ chainId: any;
8
+ disconnect: () => Promise<void>;
9
+ switchChain: () => Promise<{}>;
10
+ };
11
+ export declare const CosmosExternalWalletContext: import("react").Context<{
12
+ wallets: CommonWallet[];
13
+ chains: CommonChain[];
14
+ chainId: string;
15
+ disconnect: () => Promise<void>;
16
+ switchChain: (chainId: string) => Promise<{
17
+ error?: string[];
18
+ }>;
19
+ }>;
20
+ interface CosmosExternalWalletProviderProps {
21
+ children: ReactNode;
22
+ para: ParaWeb;
23
+ onSwitchWallet: (args: {
24
+ address?: string;
25
+ error?: string;
26
+ }) => void;
27
+ }
28
+ export declare function CosmosExternalWalletProvider({ children, para, onSwitchWallet }: CosmosExternalWalletProviderProps): import("react/jsx-runtime").JSX.Element;
29
+ export {};
@@ -0,0 +1,179 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ import { jsx as _jsx } from "react/jsx-runtime";
11
+ import { createContext, useEffect, useMemo, useState } from 'react';
12
+ import { useParaCosmos } from './ParaCosmosContext.js';
13
+ import { WalletType } from '@getpara/react-sdk';
14
+ import { checkWallet, WalletType as GrazWalletType, useAccount, useActiveChainIds, useActiveWalletType, useConnect, useDisconnect, useSuggestChainAndConnect, getChainInfo, } from '@usecapsule/graz';
15
+ export const defaultCosmosExternalWallet = {
16
+ wallets: [],
17
+ chains: [],
18
+ chainId: undefined,
19
+ disconnect: () => Promise.resolve(),
20
+ switchChain: () => Promise.resolve({}),
21
+ };
22
+ export const CosmosExternalWalletContext = createContext(defaultCosmosExternalWallet);
23
+ export function CosmosExternalWalletProvider({ children, para, onSwitchWallet }) {
24
+ var _a, _b;
25
+ const { selectedChainId, wallets: incompleteWallets, chains, multiChain, shouldUseSuggestChainAndConnect, onSwitchChain, } = useParaCosmos();
26
+ const { suggestAndConnectAsync } = useSuggestChainAndConnect();
27
+ const { data: account, isConnecting, isReconnecting, } = useAccount({
28
+ chainId: multiChain ? chains.map(c => c.chainId) : selectedChainId,
29
+ multiChain,
30
+ });
31
+ const activeChainIds = useActiveChainIds();
32
+ const { connectAsync } = useConnect();
33
+ const { disconnectAsync } = useDisconnect();
34
+ const { walletType } = useActiveWalletType();
35
+ const [isLocalConnecting, setIsLocalConnecting] = useState(false);
36
+ const bufferAddress = multiChain ? (_a = account === null || account === void 0 ? void 0 : account[selectedChainId]) === null || _a === void 0 ? void 0 : _a.address.toString() : account === null || account === void 0 ? void 0 : account.address.toString();
37
+ const address = multiChain ? (_b = account === null || account === void 0 ? void 0 : account[selectedChainId]) === null || _b === void 0 ? void 0 : _b.bech32Address : account === null || account === void 0 ? void 0 : account.bech32Address;
38
+ const reset = () => __awaiter(this, void 0, void 0, function* () {
39
+ yield disconnectAsync();
40
+ yield para.logout();
41
+ });
42
+ const switchChain = (chainId) => __awaiter(this, void 0, void 0, function* () {
43
+ let error;
44
+ const hasActiveChain = activeChainIds.includes(chainId);
45
+ if (!hasActiveChain) {
46
+ setIsLocalConnecting(true);
47
+ let changeResp;
48
+ changeResp = yield connect(walletType, chainId);
49
+ // Calling onSwitchWallet here so the modal correctly processes any error from the reconnection.
50
+ onSwitchWallet(changeResp);
51
+ setIsLocalConnecting(false);
52
+ if (changeResp.error) {
53
+ error = [changeResp === null || changeResp === void 0 ? void 0 : changeResp.error];
54
+ }
55
+ }
56
+ if (!error) {
57
+ onSwitchChain(chainId);
58
+ }
59
+ return { error };
60
+ });
61
+ const login = (bufferAddress, address, providerName) => __awaiter(this, void 0, void 0, function* () {
62
+ try {
63
+ yield para.externalWalletLogin({
64
+ address: bufferAddress,
65
+ type: WalletType.COSMOS,
66
+ provider: providerName,
67
+ addressBech32: address,
68
+ });
69
+ }
70
+ catch (err) {
71
+ yield reset();
72
+ throw 'Error logging you in. Please try again.';
73
+ }
74
+ });
75
+ useEffect(() => {
76
+ const storedExternalWallet = para.externalWallets[bufferAddress !== null && bufferAddress !== void 0 ? bufferAddress : ''];
77
+ if (!isConnecting &&
78
+ !isReconnecting &&
79
+ !isLocalConnecting &&
80
+ address &&
81
+ storedExternalWallet &&
82
+ storedExternalWallet.address !== address &&
83
+ walletType !== GrazWalletType.CAPSULE_EMBEDDED) {
84
+ para.setExternalWallet({
85
+ address: bufferAddress,
86
+ type: WalletType.COSMOS,
87
+ provider: getProviderName(walletType),
88
+ addressBech32: address,
89
+ });
90
+ }
91
+ }, [isConnecting, isReconnecting, address]);
92
+ useEffect(() => {
93
+ const storedExternalWallet = para.externalWallets[bufferAddress !== null && bufferAddress !== void 0 ? bufferAddress : ''];
94
+ if (!isConnecting &&
95
+ !isReconnecting &&
96
+ !isLocalConnecting &&
97
+ !!bufferAddress &&
98
+ !storedExternalWallet &&
99
+ walletType !== GrazWalletType.CAPSULE_EMBEDDED) {
100
+ reset();
101
+ }
102
+ }, [isConnecting, isReconnecting]);
103
+ const connect = (walletType, chainId) => __awaiter(this, void 0, void 0, function* () {
104
+ setIsLocalConnecting(true);
105
+ // chainID is passed in when switching chains, in that case we can skip disconnecting
106
+ if (!chainId) {
107
+ yield disconnectAsync();
108
+ }
109
+ const _chainId = chainId !== null && chainId !== void 0 ? chainId : (multiChain ? chains.map(c => c.chainId) : selectedChainId);
110
+ if (!_chainId) {
111
+ console.error('Chain id not provided.');
112
+ return;
113
+ }
114
+ let address;
115
+ let bufferAddress;
116
+ let error;
117
+ // The logic in the modal should prevent this from happening, logging for edge cases.
118
+ if (!walletType) {
119
+ console.error('Graz wallet type not provided.');
120
+ return;
121
+ }
122
+ else {
123
+ try {
124
+ let chainInfo;
125
+ if (shouldUseSuggestChainAndConnect) {
126
+ if (typeof _chainId !== 'string') {
127
+ console.error('multiChain is not compatible with shouldUseSuggestChainAndConnect.');
128
+ return;
129
+ }
130
+ chainInfo = getChainInfo({ chainId: _chainId });
131
+ if (!chainInfo) {
132
+ console.error('Chain not found.');
133
+ return;
134
+ }
135
+ }
136
+ const connectedWallet = yield (shouldUseSuggestChainAndConnect
137
+ ? suggestAndConnectAsync({ walletType, chainInfo })
138
+ : connectAsync({ walletType, chainId: _chainId }));
139
+ const firstChain = typeof _chainId === 'string' ? _chainId : _chainId[0];
140
+ address = connectedWallet.accounts[firstChain].bech32Address;
141
+ bufferAddress = connectedWallet.accounts[firstChain].address.toString();
142
+ if (connectedWallet.accounts[firstChain]) {
143
+ try {
144
+ yield login(bufferAddress, address, getProviderName(walletType));
145
+ }
146
+ catch (err) {
147
+ bufferAddress = undefined;
148
+ address = undefined;
149
+ error = err;
150
+ }
151
+ }
152
+ }
153
+ catch (err) {
154
+ if (err.message === 'No wallet exists') {
155
+ error = err.message;
156
+ }
157
+ else {
158
+ console.error('Graz connection error:', err);
159
+ error = 'An unknown error occurred.';
160
+ }
161
+ }
162
+ }
163
+ setIsLocalConnecting(false);
164
+ return { address, bufferAddress, error };
165
+ });
166
+ const getProviderName = (walletType) => { var _a; return (_a = incompleteWallets.find(w => w.grazType === walletType || w.grazMobileType === walletType)) === null || _a === void 0 ? void 0 : _a.name; };
167
+ const wallets = incompleteWallets
168
+ .map(wallet => {
169
+ return Object.assign(Object.assign({ connect: () => connect(wallet.grazType), connectMobile: () => connect(wallet.grazMobileType), getQrUri: () => '', type: WalletType.COSMOS }, wallet), { installed: checkWallet(wallet.grazType) });
170
+ })
171
+ .filter(w => !!w);
172
+ const formattedChains = chains.map(c => {
173
+ return {
174
+ id: c.chainId,
175
+ name: c.chainName,
176
+ };
177
+ });
178
+ return (_jsx(CosmosExternalWalletContext.Provider, { value: useMemo(() => ({ wallets, chains: formattedChains, chainId: selectedChainId, disconnect: disconnectAsync, switchChain }), [wallets, formattedChains, selectedChainId, disconnectAsync, switchChain]), children: children }));
179
+ }
@@ -0,0 +1,46 @@
1
+ import { ReactNode } from 'react';
2
+ import { WalletList, WalletWithType } from '../types/Wallet.js';
3
+ import { ChainInfo } from '@keplr-wallet/types';
4
+ import { ConfigureGrazArgs } from '@usecapsule/graz';
5
+ export declare const ParaCosmosContext: import("react").Context<{
6
+ selectedChainId?: string;
7
+ wallets: WalletWithType[];
8
+ chains: ChainInfo[];
9
+ multiChain?: boolean;
10
+ shouldUseSuggestChainAndConnect?: boolean;
11
+ onSwitchChain: (chainId: string) => void;
12
+ }>;
13
+ interface ParaCosmosProviderProps extends Omit<ConfigureGrazArgs, 'chains'> {
14
+ children: ReactNode;
15
+ wallets: WalletList;
16
+ chains: ChainInfo[];
17
+ /**
18
+ * If true, the initial connection request will request to connect to all passed in chains.
19
+ * If false, the initial connection request will only request for the selectedChainId.
20
+ * Note: this will not affect the chain that's shown in the Para modal. Even if multiple chains are connected the modal will only display what's passed in via `selectedChainId`
21
+ */
22
+ multiChain?: boolean;
23
+ /**
24
+ * Selected chain to display in the Para modal.
25
+ */
26
+ selectedChainId?: string;
27
+ /**
28
+ * Called with the newly selected chainId when a the chain value is changed in the Para modal.
29
+ */
30
+ /**
31
+ * If true, the initial connection request will use the Graz useSuggestChainAndConnect hook to connect to the selected wallet
32
+ * Ref: https://graz.sh/docs/hooks/useSuggestChainAndConnect
33
+ */
34
+ shouldUseSuggestChainAndConnect?: boolean;
35
+ onSwitchChain: (chainId: string) => void;
36
+ }
37
+ export declare function ParaCosmosProvider({ children, wallets, chains, selectedChainId, multiChain, shouldUseSuggestChainAndConnect, onSwitchChain, ...grazOpts }: ParaCosmosProviderProps): import("react/jsx-runtime").JSX.Element;
38
+ export declare const useParaCosmos: () => {
39
+ selectedChainId?: string;
40
+ wallets: WalletWithType[];
41
+ chains: ChainInfo[];
42
+ multiChain?: boolean;
43
+ shouldUseSuggestChainAndConnect?: boolean;
44
+ onSwitchChain: (chainId: string) => void;
45
+ };
46
+ export {};
@@ -0,0 +1,73 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ var __rest = (this && this.__rest) || function (s, e) {
11
+ var t = {};
12
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
13
+ t[p] = s[p];
14
+ if (s != null && typeof Object.getOwnPropertySymbols === "function")
15
+ for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
16
+ if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
17
+ t[p[i]] = s[p[i]];
18
+ }
19
+ return t;
20
+ };
21
+ import { jsx as _jsx } from "react/jsx-runtime";
22
+ import { createContext, useCallback, useContext, useEffect, useMemo } from 'react';
23
+ import { useExternalWalletProviderStore } from '@getpara/react-sdk';
24
+ import { CosmosExternalWalletContext, CosmosExternalWalletProvider } from './CosmosExternalWalletContext.js';
25
+ import { GrazProvider, WalletType, connect } from '@usecapsule/graz';
26
+ export const ParaCosmosContext = createContext({ wallets: [], chains: [], onSwitchChain: () => { } });
27
+ export function ParaCosmosProvider(_a) {
28
+ var { children, wallets, chains, selectedChainId, multiChain, shouldUseSuggestChainAndConnect, onSwitchChain } = _a, grazOpts = __rest(_a, ["children", "wallets", "chains", "selectedChainId", "multiChain", "shouldUseSuggestChainAndConnect", "onSwitchChain"]);
29
+ const updateExternalWalletProviderState = useExternalWalletProviderStore(state => state.updateState);
30
+ const CosmosProvider = useExternalWalletProviderStore(state => state.CosmosProvider);
31
+ const cosmosContext = useExternalWalletProviderStore(state => state.cosmosContext);
32
+ const connectParaCosmosWallet = useCallback(() => __awaiter(this, void 0, void 0, function* () {
33
+ if (!grazOpts.capsule) {
34
+ return { error: 'No capsule instance passed to Graz' };
35
+ }
36
+ try {
37
+ const chainId = multiChain ? chains.map(c => c.chainId) : selectedChainId;
38
+ const result = yield connect({ walletType: WalletType.CAPSULE_EMBEDDED, chainId });
39
+ return { result };
40
+ }
41
+ catch (err) {
42
+ const error = err instanceof Error ? err.message : 'Unknown error';
43
+ return { error };
44
+ }
45
+ }), [connect]);
46
+ useEffect(() => {
47
+ if (!cosmosContext || !CosmosProvider) {
48
+ updateExternalWalletProviderState({
49
+ CosmosProvider: CosmosExternalWalletProvider,
50
+ cosmosContext: CosmosExternalWalletContext,
51
+ connectParaCosmosWallet,
52
+ });
53
+ }
54
+ }, []);
55
+ const walletsWithType = [];
56
+ wallets.forEach(w => {
57
+ const wallet = w();
58
+ walletsWithType.push(wallet);
59
+ });
60
+ const value = useMemo(() => ({
61
+ selectedChainId,
62
+ wallets: walletsWithType,
63
+ chains,
64
+ multiChain,
65
+ shouldUseSuggestChainAndConnect,
66
+ onSwitchChain,
67
+ }), [selectedChainId, walletsWithType, chains, multiChain, shouldUseSuggestChainAndConnect, onSwitchChain]);
68
+ if (!cosmosContext || !CosmosProvider) {
69
+ return null;
70
+ }
71
+ return (_jsx(GrazProvider, { grazOptions: Object.assign({ chains, autoReconnect: true }, grazOpts), children: _jsx(ParaCosmosContext.Provider, { value: value, children: children }) }));
72
+ }
73
+ export const useParaCosmos = () => useContext(ParaCosmosContext);
@@ -0,0 +1,26 @@
1
+ export type WalletMetadata = {
2
+ id: string;
3
+ name: string;
4
+ iconUrl: string;
5
+ installed?: boolean;
6
+ isExtension?: boolean;
7
+ isMobile?: boolean;
8
+ isWeb?: boolean;
9
+ downloadUrl?: string;
10
+ getQrUri?: () => Promise<string>;
11
+ };
12
+ export type CommonWallet = {
13
+ connect: () => Promise<{
14
+ address?: string;
15
+ error?: string;
16
+ }>;
17
+ connectMobile: (isManualWalletConnect?: boolean) => Promise<{
18
+ address?: string;
19
+ error?: string;
20
+ }>;
21
+ type: 'EVM' | 'SOLANA' | 'COSMOS';
22
+ } & WalletMetadata;
23
+ export type CommonChain = {
24
+ id: string | number;
25
+ name: string;
26
+ };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,7 @@
1
+ import { WalletType } from '@usecapsule/graz';
2
+ import { WalletMetadata } from './CommonTypes.js';
3
+ export type WalletList = (() => WalletWithType)[];
4
+ export type WalletWithType = {
5
+ grazType: WalletType;
6
+ grazMobileType?: WalletType;
7
+ } & WalletMetadata;
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1 @@
1
+ export declare const hasInstalledExtension: (id: string) => boolean;
@@ -0,0 +1 @@
1
+ export const hasInstalledExtension = (id) => !!window[id];
@@ -0,0 +1,3 @@
1
+ import { keplrWallet } from './keplr/keplr.js';
2
+ import { leapWallet } from './leap/leap.js';
3
+ export { keplrWallet, leapWallet };
@@ -0,0 +1,3 @@
1
+ import { keplrWallet } from './keplr/keplr.js';
2
+ import { leapWallet } from './leap/leap.js';
3
+ export { keplrWallet, leapWallet };
@@ -0,0 +1,2 @@
1
+ import { WalletWithType } from '../../../types/Wallet.js';
2
+ export declare const keplrWallet: () => WalletWithType;
@@ -0,0 +1,15 @@
1
+ import { icon } from './keplrIcon.js';
2
+ import { WalletType } from '@usecapsule/graz';
3
+ import { isMobile } from '@getpara/react-sdk';
4
+ export const keplrWallet = () => {
5
+ return {
6
+ id: 'keplr',
7
+ name: 'Keplr',
8
+ iconUrl: icon,
9
+ isExtension: true,
10
+ isMobile: isMobile() && true,
11
+ downloadUrl: 'https://www.keplr.app/get',
12
+ grazType: WalletType.KEPLR,
13
+ grazMobileType: WalletType.WC_KEPLR_MOBILE,
14
+ };
15
+ };
@@ -0,0 +1 @@
1
+ export declare const icon = "";
@@ -0,0 +1 @@
1
+ export const icon = '';
@@ -0,0 +1,2 @@
1
+ import { WalletWithType } from '../../../types/Wallet.js';
2
+ export declare const leapWallet: () => WalletWithType;
@@ -0,0 +1,15 @@
1
+ import { icon } from './leapIcon.js';
2
+ import { WalletType } from '@usecapsule/graz';
3
+ import { isMobile } from '@getpara/react-sdk';
4
+ export const leapWallet = () => {
5
+ return {
6
+ id: 'leap',
7
+ name: 'Leap',
8
+ iconUrl: icon,
9
+ isExtension: true,
10
+ isMobile: isMobile() && true,
11
+ downloadUrl: 'https://www.leapwallet.io/download',
12
+ grazType: WalletType.LEAP,
13
+ grazMobileType: WalletType.WC_LEAP_MOBILE,
14
+ };
15
+ };
@@ -0,0 +1 @@
1
+ export declare const icon = "";
@@ -0,0 +1 @@
1
+ export const icon = '';
package/package.json ADDED
@@ -0,0 +1,40 @@
1
+ {
2
+ "name": "@getpara/cosmos-wallet-connectors",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "exports": {
8
+ ".": "./dist/index.js",
9
+ "./connectors": "./dist/wallets/connectors/index.js"
10
+ },
11
+ "dependencies": {
12
+ "@getpara/react-sdk": "0.1.0"
13
+ },
14
+ "scripts": {
15
+ "build": "rm -rf dist && tsc",
16
+ "test": "vitest run --coverage"
17
+ },
18
+ "devDependencies": {
19
+ "@leapwallet/cosmos-social-login-capsule-provider": "^0.0.41",
20
+ "@types/react": "^18.0.31",
21
+ "@types/react-dom": "^18.2.7",
22
+ "@usecapsule/graz": "^0.3.0",
23
+ "typescript": "^5.4.3"
24
+ },
25
+ "peerDependencies": {
26
+ "@cosmjs/cosmwasm-stargate": "<=0.31.3",
27
+ "@cosmjs/launchpad": "*",
28
+ "@cosmjs/proto-signing": "<=0.31.3",
29
+ "@cosmjs/stargate": "<=0.31.3",
30
+ "@cosmjs/tendermint-rpc": "<=0.31.3",
31
+ "@usecapsule/graz": "^0.3.0",
32
+ "react": ">=18",
33
+ "react-dom": ">=18"
34
+ },
35
+ "files": [
36
+ "dist",
37
+ "package.json"
38
+ ],
39
+ "gitHead": "625aaa94001a5461dcde8d6775c3b73f3862c76c"
40
+ }