Skip to content

Commit

Permalink
multi: Add ledger ui.
Browse files Browse the repository at this point in the history
Copy trezor elements and apply to ledger where possible.
  • Loading branch information
JoeGruffins committed Jul 3, 2023
1 parent 632d455 commit 95d0210
Show file tree
Hide file tree
Showing 48 changed files with 713 additions and 88 deletions.
7 changes: 5 additions & 2 deletions app/actions/DaemonActions.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import { stopNotifcations } from "./NotificationActions";
import { saveSettings, updateStateSettingsChanged } from "./SettingsActions";
import { rescanCancel, showCantCloseModal } from "./ControlActions";
import { enableTrezor } from "./TrezorActions";
import { enableLedger } from "./LedgerActions";
import {
DEX_LOGOUT_ATTEMPT,
DEX_LOGOUT_SUCCESS,
Expand Down Expand Up @@ -309,7 +310,7 @@ export const removeWallet = (selectedWallet) => (dispatch) => {
// selectedWallet = {
// label: newWalletName,
// value: {
// wallet: newWalletName, isWatchingOnly, isTrezor, isNew,
// wallet: newWalletName, isWatchingOnly, isTrezor, isLedger, isNew
// network: isTestNet ? "testnet" : "mainnet"
// }
// }
Expand All @@ -336,7 +337,8 @@ export const createWallet = (selectedWallet) => (dispatch, getState) =>
dispatch({
isWatchingOnly: selectedWallet.value.isWatchingOnly,
createNewWallet: selectedWallet.value.isNew,
isTrezor: selectedWallet.value.istrezor,
isTrezor: selectedWallet.value.isTrezor,
isLedger: selectedWallet.value.isLedger,
type: WALLETCREATED
});
dispatch(setSelectedWallet(selectedWallet));
Expand Down Expand Up @@ -509,6 +511,7 @@ export const startWallet =
confirmDexSeed
});
selectedWallet.value.isTrezor && dispatch(enableTrezor());
selectedWallet.value.isLedger && dispatch(enableLedger());
await dispatch(getVersionServiceAttempt());
await dispatch(openWalletAttempt("", false, selectedWallet));
return discoverAccountsComplete;
Expand Down
12 changes: 0 additions & 12 deletions app/actions/LedgerActions.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,6 @@ import {
const coin = "decred";

import * as selectors from "selectors";
import * as cfgConstants from "constants/config";

export const LDG_LEDGER_ENABLED = "LDG_LEDGER_ENABLED";
export const LDG_WALLET_CLOSED = "LDG_WALLET_CLOSED";
Expand All @@ -26,18 +25,7 @@ export const LDG_WALLET_CLOSED = "LDG_WALLET_CLOSED";
// enableLedger only sets a value in the config. Ledger connections are made
// per action then dropped.
export const enableLedger = () => (dispatch, getState) => {
const walletName = selectors.getWalletName(getState());

if (walletName) {
const config = wallet.getWalletCfg(
selectors.isTestNet(getState()),
walletName
);
config.set(cfgConstants.LEDGER, true);
}

dispatch({ type: LDG_LEDGER_ENABLED });

connect()(dispatch, getState);
};

Expand Down
11 changes: 0 additions & 11 deletions app/actions/TrezorActions.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import {
addressPath
} from "helpers/trezor";
import { publishTransactionAttempt } from "./ControlActions";
import * as cfgConstants from "constants/config";
import {
MODEL1_DECRED_HOMESCREEN,
MODELT_DECRED_HOMESCREEN
Expand Down Expand Up @@ -49,16 +48,6 @@ export const TRZ_TREZOR_ENABLED = "TRZ_TREZOR_ENABLED";
// enableTrezor attepts to start a connection with connect if none exist and
// connect to a trezor device.
export const enableTrezor = () => (dispatch, getState) => {
const walletName = selectors.getWalletName(getState());

if (walletName) {
const config = wallet.getWalletCfg(
selectors.isTestNet(getState()),
walletName
);
config.set(cfgConstants.TREZOR, true);
}

dispatch({ type: TRZ_TREZOR_ENABLED });

if (!setListeners) {
Expand Down
12 changes: 10 additions & 2 deletions app/actions/WalletLoaderActions.js
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ import {
getBestBlockHeightAttempt
} from "./ClientActions";
import { WALLETREMOVED_FAILED } from "./DaemonActions";
import { isTestNet, trezorDevice } from "selectors";
import { isTestNet, trezorDevice, ledgerDevice } from "selectors";
import { walletrpc as api } from "middleware/walletrpc/api_pb";
import { push as pushHistory } from "connected-react-router";
import { stopNotifcations } from "./NotificationActions";
Expand All @@ -28,6 +28,7 @@ import * as cfgConstants from "constants/config";
import { RESCAN_PROGRESS } from "./ControlActions";
import { stopAccountMixer } from "./AccountMixerActions";
import { TRZ_WALLET_CLOSED } from "actions/TrezorActions";
import { LDG_WALLET_CLOSED } from "actions/LedgerActions";
import { saveSettings, updateStateSettingsChanged } from "./SettingsActions";

const { SyncNotificationType } = api;
Expand Down Expand Up @@ -156,7 +157,7 @@ export const CREATEWATCHONLYWALLET_FAILED = "CREATEWATCHONLYWALLET_FAILED";
export const CREATEWATCHONLYWALLET_SUCCESS = "CREATEWATCHONLYWALLET_SUCCESS";

export const createWatchOnlyWalletRequest =
(extendedPubKey, pubPass = "") =>
(extendedPubKey, isLedger, isTrezor, pubPass = "") =>
(dispatch, getState) =>
new Promise((resolve, reject) => {
dispatch({ type: CREATEWATCHONLYWALLET_ATTEMPT });
Expand All @@ -172,6 +173,12 @@ export const createWatchOnlyWalletRequest =
} = getState();
const config = wallet.getWalletCfg(isTestNet(getState()), walletName);
config.set(cfgConstants.IS_WATCH_ONLY, true);
if (isTrezor) {
config.set(cfgConstants.TREZOR, true);
}
if (isLedger) {
config.set(cfgConstants.LEDGER, true);
}
config.delete(cfgConstants.DISCOVER_ACCOUNTS);
wallet.setIsWatchingOnly(true);
dispatch({ response: {}, type: CREATEWATCHONLYWALLET_SUCCESS });
Expand Down Expand Up @@ -269,6 +276,7 @@ const finalCloseWallet = () => async (dispatch, getState) => {
await wallet.stopWallet();
dispatch({ type: CLOSEWALLET_SUCCESS });
if (trezorDevice(getState())) dispatch({ type: TRZ_WALLET_CLOSED });
if (ledgerDevice(getState())) dispatch({ type: LDG_WALLET_CLOSED });
dispatch(pushHistory("/getstarted/initial"));
} catch (error) {
dispatch({ error, type: CLOSEWALLET_FAILED });
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,10 @@ const SendTransactionButton = ({
unsignedTransaction,
isSendingTransaction,
isTrezor,
isLedger,
onAttemptSignTransaction,
onAttemptSignTransactionTrezor
onAttemptSignTransactionTrezor,
onAttemptSignTransactionLedger
} = useSendTransactionButton();

const signTransaction = (privpass) => {
Expand All @@ -31,6 +33,12 @@ const SendTransactionButton = ({
onSubmit?.();
};

const signTransactionLedger = () => {
if (disabled) return;
onAttemptSignTransactionLedger?.(unsignedTransaction);
onSubmit?.();
};

if (isTrezor) {
return (
<KeyBlueButton
Expand All @@ -40,6 +48,15 @@ const SendTransactionButton = ({
{buttonLabel ? buttonLabel : <T id="send.sendBtn" m="Send" />}
</KeyBlueButton>
);
} else if (isLedger) {
return (
<KeyBlueButton
onClick={signTransactionLedger}
disabled={disabled || isSendingTransaction}
loading={isSendingTransaction}>
{buttonLabel ? buttonLabel : <T id="send.sendBtn" m="Send" />}
</KeyBlueButton>
);
} else {
return (
<PassphraseModalButton
Expand Down
8 changes: 7 additions & 1 deletion app/components/buttons/SendTransactionButton/hooks.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,26 +2,32 @@ import { useDispatch, useSelector } from "react-redux";
import * as sel from "selectors";
import * as ca from "actions/ControlActions";
import * as tza from "actions/TrezorActions";
import * as ldgr from "actions/LedgerActions";

export function useSendTransactionButton() {
const unsignedTransaction = useSelector(sel.unsignedTransaction);
const constructTxResponse = useSelector(sel.constructTxResponse);
const isSendingTransaction = useSelector(sel.isSendingTransaction);
const isTrezor = useSelector(sel.isTrezor);
const isLedger = useSelector(sel.isLedger);

const dispatch = useDispatch();
const onAttemptSignTransaction = (passphrase, rawTx, acct) => {
dispatch(ca.signTransactionAttempt(passphrase, rawTx, acct));
};
const onAttemptSignTransactionTrezor = (rawUnsigTx, constructTxResponse) =>
dispatch(tza.signTransactionAttemptTrezor(rawUnsigTx, constructTxResponse));
const onAttemptSignTransactionLedger = (rawUnsigTx) =>
dispatch(ldgr.signTransactionAttemptLedger(rawUnsigTx));

return {
unsignedTransaction,
constructTxResponse,
isSendingTransaction,
isTrezor,
isLedger,
onAttemptSignTransaction,
onAttemptSignTransactionTrezor
onAttemptSignTransactionTrezor,
onAttemptSignTransactionLedger
};
}
7 changes: 5 additions & 2 deletions app/components/shared/SendTransaction/Form.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ const Form = ({
unsignedRawTx,
isWatchingOnly,
isTrezor,
isLedger,
insuficientFunds,
styles,
hideDetails,
Expand Down Expand Up @@ -71,7 +72,9 @@ const Form = ({
<T id="send.insuficient.funds" m="Insufficient funds" />
</div>
)}
{((isTrezor && isWatchingOnly) || !isWatchingOnly) &&
{((isTrezor && isWatchingOnly) ||
(isLedger && isWatchingOnly) ||
!isWatchingOnly) &&
(getRunningIndicator ? (
<Tooltip
contentClassName={styles.disabledTooltip}
Expand Down Expand Up @@ -144,7 +147,7 @@ const Form = ({
</div>
</div>
</div>
{unsignedRawTx && isWatchingOnly && !isTrezor && (
{unsignedRawTx && isWatchingOnly && !isTrezor && !isLedger && (
<UnsignedTx
title={<T id="send.unsignedRawTxTite" m="Unsigned Raw Transaction:" />}
tx={unsignedRawTx}
Expand Down
2 changes: 2 additions & 0 deletions app/components/shared/SendTransaction/SendTransaction.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ const SendTransaction = ({
totalSpent,
notMixedAccounts,
isTrezor,
isLedger,
isWatchingOnly,
isConstructingTransaction,
attemptConstructTransaction,
Expand Down Expand Up @@ -394,6 +395,7 @@ const SendTransaction = ({
willEnter,
isWatchingOnly,
isTrezor,
isLedger,
insuficientFunds,
styles,
hideDetails,
Expand Down
2 changes: 2 additions & 0 deletions app/components/shared/SendTransaction/hooks.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ export function useSendTransaction() {
shallowEqual
);
const isTrezor = useSelector(sel.isTrezor);
const isLedger = useSelector(sel.isLedger);
const isWatchingOnly = useSelector(sel.isWatchingOnly);
const isConstructingTransaction = useSelector(sel.isConstructingTransaction);
const constructTxRequestAttempt = useSelector(sel.constructTxRequestAttempt);
Expand Down Expand Up @@ -61,6 +62,7 @@ export function useSendTransaction() {
totalSpent,
notMixedAccounts,
isTrezor,
isLedger,
isWatchingOnly,
isConstructingTransaction,
attemptConstructTransaction,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -103,18 +103,16 @@ const CreateWalletPage = ({ createWalletRef, onSendBack }) => {
cancelCreateWallet
]);

const onCreateWatchOnly = useCallback(() => {
createWatchOnlyWalletRequest(walletMasterPubKey)
.then(() => sendEvent({ type: "WALLET_CREATED" }))
.catch((error) => sendEvent({ type: "ERROR", error }));
// we send a continue so we go to loading state
sendContinue();
}, [
createWatchOnlyWalletRequest,
sendEvent,
sendContinue,
walletMasterPubKey
]);
const onCreateWatchOnly = useCallback(
(isLedger, isTrezor) => {
createWatchOnlyWalletRequest(walletMasterPubKey, isLedger, isTrezor)
.then(() => sendEvent({ type: "WALLET_CREATED" }))
.catch((error) => sendEvent({ type: "ERROR", error }));
// we send a continue so we go to loading state
sendContinue();
},
[createWatchOnlyWalletRequest, sendEvent, sendContinue, walletMasterPubKey]
);

const getStateComponent = useCallback(() => {
const { mnemonic, error } = current.context;
Expand Down Expand Up @@ -188,7 +186,8 @@ const CreateWalletPage = ({ createWalletRef, onSendBack }) => {
]);

useEffect(() => {
const { isNew, walletMasterPubKey, mnemonic } = current.context;
const { isNew, walletMasterPubKey, mnemonic, isLedger, isTrezor } =
current.context;
switch (current.value) {
case "createWalletInit":
setIsNew(isNew);
Expand Down Expand Up @@ -222,7 +221,7 @@ const CreateWalletPage = ({ createWalletRef, onSendBack }) => {
checkIsValid();
break;
case "restoreWatchingOnly":
onCreateWatchOnly();
onCreateWatchOnly(isLedger, isTrezor);
break;
case "finished":
break;
Expand Down
11 changes: 9 additions & 2 deletions app/components/views/GetStartedPage/CreateWalletPage/hooks.js
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,15 @@ export const useCreateWallet = () => {
);
// TODO implement pubpass
const createWatchOnlyWalletRequest = useCallback(
(extendedPubKey, pubPass = "") =>
dispatch(wla.createWatchOnlyWalletRequest(extendedPubKey, pubPass)),
(extendedPubKey, isLedger, isTrezor, pubPass = "") =>
dispatch(
wla.createWatchOnlyWalletRequest(
extendedPubKey,
isLedger,
isTrezor,
pubPass
)
),
[dispatch]
);
const createWalletRequest = useCallback(
Expand Down
Loading

0 comments on commit 95d0210

Please sign in to comment.