@ledgerhq/live-common 34.54.0-nightly.20251203105022 → 34.54.0-nightly.20251204023901

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.
@@ -12,4 +12,6 @@ export type TransferInstructionParams = {
12
12
  };
13
13
  export type TransferInstructionType = "accept-transfer-instruction" | "reject-transfer-instruction" | "withdraw-transfer-instruction";
14
14
  export declare function useCantonAcceptOrRejectOffer({ currency, account, partyId, }: UseCantonAcceptOrRejectOfferOptions): ({ contractId, deviceId, reason }: TransferInstructionParams, type: TransferInstructionType) => Promise<void>;
15
+ export declare const getRemainingTime: (diff: number) => string;
16
+ export declare const useTimeRemaining: (expiresAtMicros?: number, isExpired?: boolean) => string;
15
17
  //# sourceMappingURL=react.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"react.d.ts","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC;AAE/C,MAAM,MAAM,mCAAmC,GAAG;IAChD,QAAQ,EAAE,cAAc,CAAC;IACzB,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,yBAAyB,GAAG;IACtC,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,uBAAuB,GAC/B,6BAA6B,GAC7B,6BAA6B,GAC7B,+BAA+B,CAAC;AAEpC,wBAAgB,4BAA4B,CAAC,EAC3C,QAAQ,EACR,OAAO,EACP,OAAO,GACR,EAAE,mCAAmC,sCAKE,yBAAyB,QACrD,uBAAuB,mBAgBlC"}
1
+ {"version":3,"file":"react.d.ts","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC;AAE/C,MAAM,MAAM,mCAAmC,GAAG;IAChD,QAAQ,EAAE,cAAc,CAAC;IACzB,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,yBAAyB,GAAG;IACtC,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,uBAAuB,GAC/B,6BAA6B,GAC7B,6BAA6B,GAC7B,+BAA+B,CAAC;AAEpC,wBAAgB,4BAA4B,CAAC,EAC3C,QAAQ,EACR,OAAO,EACP,OAAO,GACR,EAAE,mCAAmC,sCAKE,yBAAyB,QACrD,uBAAuB,mBAgBlC;AAED,eAAO,MAAM,gBAAgB,SAAU,MAAM,KAAG,MAgB/C,CAAC;AAEF,eAAO,MAAM,gBAAgB,qDAA6C,MAwBzE,CAAC"}
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.useCantonAcceptOrRejectOffer = void 0;
3
+ exports.useTimeRemaining = exports.getRemainingTime = exports.useCantonAcceptOrRejectOffer = void 0;
4
4
  const react_1 = require("react");
5
5
  const bridge_1 = require("../../bridge");
6
6
  function useCantonAcceptOrRejectOffer({ currency, account, partyId, }) {
@@ -11,4 +11,43 @@ function useCantonAcceptOrRejectOffer({ currency, account, partyId, }) {
11
11
  return transferInstruction;
12
12
  }
13
13
  exports.useCantonAcceptOrRejectOffer = useCantonAcceptOrRejectOffer;
14
+ const getRemainingTime = (diff) => {
15
+ if (diff <= 0) {
16
+ return "";
17
+ }
18
+ const days = Math.floor(diff / (1000 * 60 * 60 * 24));
19
+ const hours = Math.floor((diff % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
20
+ const minutes = Math.floor((diff % (1000 * 60 * 60)) / (1000 * 60));
21
+ const seconds = Math.floor((diff % (1000 * 60)) / 1000);
22
+ let format = "";
23
+ if (days > 0)
24
+ format += `${days}d `;
25
+ if (hours > 0)
26
+ format += `${hours}h `;
27
+ if (minutes > 0)
28
+ format += `${minutes}m `;
29
+ format += `${seconds}s`;
30
+ return format.trim();
31
+ };
32
+ exports.getRemainingTime = getRemainingTime;
33
+ const useTimeRemaining = (expiresAtMicros = 0, isExpired = false) => {
34
+ const [timeRemaining, setTimeRemaining] = (0, react_1.useState)("");
35
+ (0, react_1.useEffect)(() => {
36
+ if (expiresAtMicros <= 0 || isExpired) {
37
+ setTimeRemaining("");
38
+ return;
39
+ }
40
+ const updateTimeRemaining = () => {
41
+ const now = Date.now();
42
+ const expiresAt = expiresAtMicros / 1000;
43
+ const diff = expiresAt - now;
44
+ setTimeRemaining((0, exports.getRemainingTime)(diff));
45
+ };
46
+ updateTimeRemaining();
47
+ const interval = setInterval(updateTimeRemaining, 1000);
48
+ return () => clearInterval(interval);
49
+ }, [expiresAtMicros, isExpired]);
50
+ return timeRemaining;
51
+ };
52
+ exports.useTimeRemaining = useTimeRemaining;
14
53
  //# sourceMappingURL=react.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"react.js","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":";;;AAAA,iCAAoC;AACpC,yCAAiD;AAsBjD,SAAgB,4BAA4B,CAAC,EAC3C,QAAQ,EACR,OAAO,EACP,OAAO,GAC6B;IACpC,MAAM,YAAY,GAAG,IAAA,0BAAiB,EAAC,QAAQ,CAAyB,CAAC;IAEzE,MAAM,mBAAmB,GAAG,IAAA,mBAAW,EACrC,CACE,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAA6B,EAC3D,IAA6B,EAC7B,EAAE;QACF,OAAO,YAAY,CAAC,mBAAmB,CACrC,QAAQ,EACR,QAAQ,EACR,OAAO,EACP,OAAO,EACP,UAAU,EACV,IAAI,EACJ,MAAM,CACP,CAAC;IACJ,CAAC,EACD,CAAC,YAAY,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,CAAC,CAC3C,CAAC;IAEF,OAAO,mBAAmB,CAAC;AAC7B,CAAC;AA1BD,oEA0BC"}
1
+ {"version":3,"file":"react.js","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"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"}
@@ -3,12 +3,12 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  const live_env_1 = require("@ledgerhq/live-env");
4
4
  function getEnvironmentParams(environment) {
5
5
  return {
6
- cloudSyncApiBaseUrl: environment && environment === "PROD"
7
- ? (0, live_env_1.getEnv)("CLOUD_SYNC_API_PROD")
8
- : (0, live_env_1.getEnv)("CLOUD_SYNC_API_STAGING"),
9
- trustchainApiBaseUrl: environment && environment === "PROD"
10
- ? (0, live_env_1.getEnv)("TRUSTCHAIN_API_PROD")
11
- : (0, live_env_1.getEnv)("TRUSTCHAIN_API_STAGING"),
6
+ cloudSyncApiBaseUrl: environment && environment === "STAGING"
7
+ ? (0, live_env_1.getEnv)("CLOUD_SYNC_API_STAGING")
8
+ : (0, live_env_1.getEnv)("CLOUD_SYNC_API_PROD"),
9
+ trustchainApiBaseUrl: environment && environment === "STAGING"
10
+ ? (0, live_env_1.getEnv)("TRUSTCHAIN_API_STAGING")
11
+ : (0, live_env_1.getEnv)("TRUSTCHAIN_API_PROD"),
12
12
  };
13
13
  }
14
14
  exports.default = getEnvironmentParams;
@@ -1 +1 @@
1
- {"version":3,"file":"getEnvironmentParams.js","sourceRoot":"","sources":["../../src/walletSync/getEnvironmentParams.ts"],"names":[],"mappings":";;AAAA,iDAA4C;AAQ5C,SAAwB,oBAAoB,CAC1C,WAA8C;IAE9C,OAAO;QACL,mBAAmB,EACjB,WAAW,IAAI,WAAW,KAAK,MAAM;YACnC,CAAC,CAAC,IAAA,iBAAM,EAAC,qBAAqB,CAAC;YAC/B,CAAC,CAAC,IAAA,iBAAM,EAAC,wBAAwB,CAAC;QACtC,oBAAoB,EAClB,WAAW,IAAI,WAAW,KAAK,MAAM;YACnC,CAAC,CAAC,IAAA,iBAAM,EAAC,qBAAqB,CAAC;YAC/B,CAAC,CAAC,IAAA,iBAAM,EAAC,wBAAwB,CAAC;KACvC,CAAC;AACJ,CAAC;AAbD,uCAaC"}
1
+ {"version":3,"file":"getEnvironmentParams.js","sourceRoot":"","sources":["../../src/walletSync/getEnvironmentParams.ts"],"names":[],"mappings":";;AAAA,iDAA4C;AAQ5C,SAAwB,oBAAoB,CAC1C,WAA8C;IAE9C,OAAO;QACL,mBAAmB,EACjB,WAAW,IAAI,WAAW,KAAK,SAAS;YACtC,CAAC,CAAC,IAAA,iBAAM,EAAC,wBAAwB,CAAC;YAClC,CAAC,CAAC,IAAA,iBAAM,EAAC,qBAAqB,CAAC;QACnC,oBAAoB,EAClB,WAAW,IAAI,WAAW,KAAK,SAAS;YACtC,CAAC,CAAC,IAAA,iBAAM,EAAC,wBAAwB,CAAC;YAClC,CAAC,CAAC,IAAA,iBAAM,EAAC,qBAAqB,CAAC;KACpC,CAAC;AACJ,CAAC;AAbD,uCAaC"}
@@ -12,4 +12,6 @@ export type TransferInstructionParams = {
12
12
  };
13
13
  export type TransferInstructionType = "accept-transfer-instruction" | "reject-transfer-instruction" | "withdraw-transfer-instruction";
14
14
  export declare function useCantonAcceptOrRejectOffer({ currency, account, partyId, }: UseCantonAcceptOrRejectOfferOptions): ({ contractId, deviceId, reason }: TransferInstructionParams, type: TransferInstructionType) => Promise<void>;
15
+ export declare const getRemainingTime: (diff: number) => string;
16
+ export declare const useTimeRemaining: (expiresAtMicros?: number, isExpired?: boolean) => string;
15
17
  //# sourceMappingURL=react.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"react.d.ts","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC;AAE/C,MAAM,MAAM,mCAAmC,GAAG;IAChD,QAAQ,EAAE,cAAc,CAAC;IACzB,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,yBAAyB,GAAG;IACtC,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,uBAAuB,GAC/B,6BAA6B,GAC7B,6BAA6B,GAC7B,+BAA+B,CAAC;AAEpC,wBAAgB,4BAA4B,CAAC,EAC3C,QAAQ,EACR,OAAO,EACP,OAAO,GACR,EAAE,mCAAmC,sCAKE,yBAAyB,QACrD,uBAAuB,mBAgBlC"}
1
+ {"version":3,"file":"react.d.ts","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AAC9D,OAAO,EAAE,OAAO,EAAE,MAAM,sBAAsB,CAAC;AAE/C,MAAM,MAAM,mCAAmC,GAAG;IAChD,QAAQ,EAAE,cAAc,CAAC;IACzB,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,yBAAyB,GAAG;IACtC,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,uBAAuB,GAC/B,6BAA6B,GAC7B,6BAA6B,GAC7B,+BAA+B,CAAC;AAEpC,wBAAgB,4BAA4B,CAAC,EAC3C,QAAQ,EACR,OAAO,EACP,OAAO,GACR,EAAE,mCAAmC,sCAKE,yBAAyB,QACrD,uBAAuB,mBAgBlC;AAED,eAAO,MAAM,gBAAgB,SAAU,MAAM,KAAG,MAgB/C,CAAC;AAEF,eAAO,MAAM,gBAAgB,qDAA6C,MAwBzE,CAAC"}
@@ -1,4 +1,4 @@
1
- import { useCallback } from "react";
1
+ import { useCallback, useState, useEffect } from "react";
2
2
  import { getCurrencyBridge } from "../../bridge";
3
3
  export function useCantonAcceptOrRejectOffer({ currency, account, partyId, }) {
4
4
  const cantonBridge = getCurrencyBridge(currency);
@@ -7,4 +7,41 @@ export function useCantonAcceptOrRejectOffer({ currency, account, partyId, }) {
7
7
  }, [cantonBridge, currency, account, partyId]);
8
8
  return transferInstruction;
9
9
  }
10
+ export const getRemainingTime = (diff) => {
11
+ if (diff <= 0) {
12
+ return "";
13
+ }
14
+ const days = Math.floor(diff / (1000 * 60 * 60 * 24));
15
+ const hours = Math.floor((diff % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
16
+ const minutes = Math.floor((diff % (1000 * 60 * 60)) / (1000 * 60));
17
+ const seconds = Math.floor((diff % (1000 * 60)) / 1000);
18
+ let format = "";
19
+ if (days > 0)
20
+ format += `${days}d `;
21
+ if (hours > 0)
22
+ format += `${hours}h `;
23
+ if (minutes > 0)
24
+ format += `${minutes}m `;
25
+ format += `${seconds}s`;
26
+ return format.trim();
27
+ };
28
+ export const useTimeRemaining = (expiresAtMicros = 0, isExpired = false) => {
29
+ const [timeRemaining, setTimeRemaining] = useState("");
30
+ useEffect(() => {
31
+ if (expiresAtMicros <= 0 || isExpired) {
32
+ setTimeRemaining("");
33
+ return;
34
+ }
35
+ const updateTimeRemaining = () => {
36
+ const now = Date.now();
37
+ const expiresAt = expiresAtMicros / 1000;
38
+ const diff = expiresAt - now;
39
+ setTimeRemaining(getRemainingTime(diff));
40
+ };
41
+ updateTimeRemaining();
42
+ const interval = setInterval(updateTimeRemaining, 1000);
43
+ return () => clearInterval(interval);
44
+ }, [expiresAtMicros, isExpired]);
45
+ return timeRemaining;
46
+ };
10
47
  //# sourceMappingURL=react.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"react.js","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,OAAO,CAAC;AACpC,OAAO,EAAE,iBAAiB,EAAE,MAAM,cAAc,CAAC;AAsBjD,MAAM,UAAU,4BAA4B,CAAC,EAC3C,QAAQ,EACR,OAAO,EACP,OAAO,GAC6B;IACpC,MAAM,YAAY,GAAG,iBAAiB,CAAC,QAAQ,CAAyB,CAAC;IAEzE,MAAM,mBAAmB,GAAG,WAAW,CACrC,CACE,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAA6B,EAC3D,IAA6B,EAC7B,EAAE;QACF,OAAO,YAAY,CAAC,mBAAmB,CACrC,QAAQ,EACR,QAAQ,EACR,OAAO,EACP,OAAO,EACP,UAAU,EACV,IAAI,EACJ,MAAM,CACP,CAAC;IACJ,CAAC,EACD,CAAC,YAAY,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,CAAC,CAC3C,CAAC;IAEF,OAAO,mBAAmB,CAAC;AAC7B,CAAC"}
1
+ {"version":3,"file":"react.js","sourceRoot":"","sources":["../../../src/families/canton/react.ts"],"names":[],"mappings":"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"}
@@ -1,12 +1,12 @@
1
1
  import { getEnv } from "@ledgerhq/live-env";
2
2
  export default function getEnvironmentParams(environment) {
3
3
  return {
4
- cloudSyncApiBaseUrl: environment && environment === "PROD"
5
- ? getEnv("CLOUD_SYNC_API_PROD")
6
- : getEnv("CLOUD_SYNC_API_STAGING"),
7
- trustchainApiBaseUrl: environment && environment === "PROD"
8
- ? getEnv("TRUSTCHAIN_API_PROD")
9
- : getEnv("TRUSTCHAIN_API_STAGING"),
4
+ cloudSyncApiBaseUrl: environment && environment === "STAGING"
5
+ ? getEnv("CLOUD_SYNC_API_STAGING")
6
+ : getEnv("CLOUD_SYNC_API_PROD"),
7
+ trustchainApiBaseUrl: environment && environment === "STAGING"
8
+ ? getEnv("TRUSTCHAIN_API_STAGING")
9
+ : getEnv("TRUSTCHAIN_API_PROD"),
10
10
  };
11
11
  }
12
12
  //# sourceMappingURL=getEnvironmentParams.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"getEnvironmentParams.js","sourceRoot":"","sources":["../../src/walletSync/getEnvironmentParams.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAQ5C,MAAM,CAAC,OAAO,UAAU,oBAAoB,CAC1C,WAA8C;IAE9C,OAAO;QACL,mBAAmB,EACjB,WAAW,IAAI,WAAW,KAAK,MAAM;YACnC,CAAC,CAAC,MAAM,CAAC,qBAAqB,CAAC;YAC/B,CAAC,CAAC,MAAM,CAAC,wBAAwB,CAAC;QACtC,oBAAoB,EAClB,WAAW,IAAI,WAAW,KAAK,MAAM;YACnC,CAAC,CAAC,MAAM,CAAC,qBAAqB,CAAC;YAC/B,CAAC,CAAC,MAAM,CAAC,wBAAwB,CAAC;KACvC,CAAC;AACJ,CAAC"}
1
+ {"version":3,"file":"getEnvironmentParams.js","sourceRoot":"","sources":["../../src/walletSync/getEnvironmentParams.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAQ5C,MAAM,CAAC,OAAO,UAAU,oBAAoB,CAC1C,WAA8C;IAE9C,OAAO;QACL,mBAAmB,EACjB,WAAW,IAAI,WAAW,KAAK,SAAS;YACtC,CAAC,CAAC,MAAM,CAAC,wBAAwB,CAAC;YAClC,CAAC,CAAC,MAAM,CAAC,qBAAqB,CAAC;QACnC,oBAAoB,EAClB,WAAW,IAAI,WAAW,KAAK,SAAS;YACtC,CAAC,CAAC,MAAM,CAAC,wBAAwB,CAAC;YAClC,CAAC,CAAC,MAAM,CAAC,qBAAqB,CAAC;KACpC,CAAC;AACJ,CAAC"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@ledgerhq/live-common",
3
3
  "description": "Common ground for the Ledger Live apps",
4
- "version": "34.54.0-nightly.20251203105022",
4
+ "version": "34.54.0-nightly.20251204023901",
5
5
  "repository": {
6
6
  "type": "git",
7
7
  "url": "https://github.com/LedgerHQ/ledger-live.git"
@@ -176,78 +176,78 @@
176
176
  "xstate": "^5.19.2",
177
177
  "yargs": "^17.0.0",
178
178
  "zod": "^3.22.4",
179
- "@ledgerhq/coin-algorand": "^0.14.0-nightly.20251203105022",
180
- "@ledgerhq/coin-aptos": "^3.6.1-nightly.20251203105022",
181
- "@ledgerhq/coin-bitcoin": "^0.26.0-nightly.20251203105022",
182
- "@ledgerhq/coin-cardano": "^0.15.1-nightly.20251203105022",
183
- "@ledgerhq/coin-canton": "^0.11.0-nightly.20251203105022",
184
- "@ledgerhq/coin-cosmos": "^0.21.0-nightly.20251203105022",
185
- "@ledgerhq/coin-casper": "^2.4.1-nightly.20251203105022",
186
- "@ledgerhq/coin-evm": "^2.36.0-nightly.20251203105022",
187
- "@ledgerhq/coin-filecoin": "^1.15.0-nightly.20251203105022",
188
- "@ledgerhq/coin-celo": "^1.7.2-nightly.20251203105022",
189
- "@ledgerhq/coin-framework": "^6.10.0-nightly.20251203105022",
190
- "@ledgerhq/coin-hedera": "^1.15.0-nightly.20251203105022",
191
- "@ledgerhq/coin-icon": "^0.15.1-nightly.20251203105022",
192
- "@ledgerhq/coin-internet_computer": "^1.11.1-nightly.20251203105022",
193
- "@ledgerhq/coin-mina": "^1.4.2-nightly.20251203105022",
194
- "@ledgerhq/coin-kaspa": "^1.4.2-nightly.20251203105022",
195
- "@ledgerhq/coin-polkadot": "^6.14.0-nightly.20251203105022",
196
- "@ledgerhq/coin-near": "^0.16.1-nightly.20251203105022",
197
- "@ledgerhq/coin-solana": "^0.38.0-nightly.20251203105022",
198
- "@ledgerhq/coin-multiversx": "^0.8.1-nightly.20251203105022",
199
- "@ledgerhq/coin-stacks": "^0.12.1-nightly.20251203105022",
200
- "@ledgerhq/coin-stellar": "^6.8.0-nightly.20251203105022",
201
- "@ledgerhq/coin-ton": "^0.17.1-nightly.20251203105022",
202
- "@ledgerhq/coin-sui": "^0.18.1-nightly.20251203105022",
203
- "@ledgerhq/coin-tron": "^5.7.1-nightly.20251203105022",
204
- "@ledgerhq/coin-vechain": "^2.12.2-nightly.20251203105022",
205
- "@ledgerhq/coin-tezos": "^6.10.0-nightly.20251203105022",
206
- "@ledgerhq/coin-xrp": "^7.9.0-nightly.20251203105022",
179
+ "@ledgerhq/coin-algorand": "^0.14.0-nightly.20251204023901",
180
+ "@ledgerhq/coin-aptos": "^3.6.1-nightly.20251204023901",
181
+ "@ledgerhq/coin-bitcoin": "^0.26.0-nightly.20251204023901",
182
+ "@ledgerhq/coin-cardano": "^0.15.1-nightly.20251204023901",
183
+ "@ledgerhq/coin-canton": "^0.11.0-nightly.20251204023901",
184
+ "@ledgerhq/coin-casper": "^2.4.1-nightly.20251204023901",
185
+ "@ledgerhq/coin-celo": "^1.7.2-nightly.20251204023901",
186
+ "@ledgerhq/coin-cosmos": "^0.21.0-nightly.20251204023901",
187
+ "@ledgerhq/coin-evm": "^2.36.0-nightly.20251204023901",
188
+ "@ledgerhq/coin-filecoin": "^1.15.0-nightly.20251204023901",
189
+ "@ledgerhq/coin-framework": "^6.10.0-nightly.20251204023901",
190
+ "@ledgerhq/coin-hedera": "^1.15.0-nightly.20251204023901",
191
+ "@ledgerhq/coin-icon": "^0.15.1-nightly.20251204023901",
192
+ "@ledgerhq/coin-internet_computer": "^1.11.1-nightly.20251204023901",
193
+ "@ledgerhq/coin-kaspa": "^1.4.2-nightly.20251204023901",
194
+ "@ledgerhq/coin-mina": "^1.4.2-nightly.20251204023901",
195
+ "@ledgerhq/coin-multiversx": "^0.8.1-nightly.20251204023901",
196
+ "@ledgerhq/coin-near": "^0.16.1-nightly.20251204023901",
197
+ "@ledgerhq/coin-polkadot": "^6.14.0-nightly.20251204023901",
198
+ "@ledgerhq/coin-solana": "^0.38.0-nightly.20251204023901",
199
+ "@ledgerhq/coin-stacks": "^0.12.1-nightly.20251204023901",
200
+ "@ledgerhq/coin-stellar": "^6.8.0-nightly.20251204023901",
201
+ "@ledgerhq/coin-sui": "^0.18.1-nightly.20251204023901",
202
+ "@ledgerhq/coin-tezos": "^6.10.0-nightly.20251204023901",
203
+ "@ledgerhq/coin-ton": "^0.17.1-nightly.20251204023901",
204
+ "@ledgerhq/coin-vechain": "^2.12.2-nightly.20251204023901",
205
+ "@ledgerhq/coin-xrp": "^7.9.0-nightly.20251204023901",
206
+ "@ledgerhq/cryptoassets": "^13.34.0-nightly.20251204023901",
207
+ "@ledgerhq/device-core": "^0.6.9-nightly.20251204023901",
207
208
  "@ledgerhq/devices": "8.7.0",
208
- "@ledgerhq/device-core": "^0.6.9-nightly.20251203105022",
209
- "@ledgerhq/hw-app-algorand": "^6.31.9",
210
- "@ledgerhq/cryptoassets": "^13.34.0-nightly.20251203105022",
209
+ "@ledgerhq/coin-tron": "^5.7.1-nightly.20251204023901",
211
210
  "@ledgerhq/errors": "^6.27.0",
212
211
  "@ledgerhq/hw-app-aptos": "^6.34.9",
213
- "@ledgerhq/hw-app-btc": "^10.12.1-nightly.20251203105022",
214
- "@ledgerhq/hw-app-celo": "^6.35.4-nightly.20251203105022",
212
+ "@ledgerhq/hw-app-btc": "^10.12.1-nightly.20251204023901",
213
+ "@ledgerhq/hw-app-celo": "^6.35.4-nightly.20251204023901",
215
214
  "@ledgerhq/hw-app-cosmos": "^6.32.9",
216
- "@ledgerhq/hw-app-eth": "^7.0.0-nightly.20251203105022",
217
- "@ledgerhq/hw-app-exchange": "^0.18.0-nightly.20251203105022",
218
- "@ledgerhq/hw-app-kaspa": "^1.3.2",
215
+ "@ledgerhq/hw-app-eth": "^7.0.0-nightly.20251204023901",
216
+ "@ledgerhq/hw-app-exchange": "^0.18.0-nightly.20251204023901",
219
217
  "@ledgerhq/hw-app-hedera": "^1.2.9",
218
+ "@ledgerhq/hw-app-algorand": "^6.31.9",
219
+ "@ledgerhq/hw-app-kaspa": "^1.3.2",
220
220
  "@ledgerhq/hw-app-multiversx": "^6.26.0",
221
- "@ledgerhq/hw-app-polkadot": "^6.34.9",
222
221
  "@ledgerhq/hw-app-near": "^6.31.9",
222
+ "@ledgerhq/hw-app-polkadot": "^6.34.9",
223
223
  "@ledgerhq/hw-app-str": "^7.2.9",
224
224
  "@ledgerhq/hw-app-icon": "^1.3.9",
225
+ "@ledgerhq/hw-app-sui": "^1.4.0",
225
226
  "@ledgerhq/hw-app-tezos": "^6.31.9",
226
227
  "@ledgerhq/hw-app-trx": "^6.31.9",
227
- "@ledgerhq/hw-app-vet": "^0.8.0-nightly.20251203105022",
228
+ "@ledgerhq/hw-app-vet": "^0.8.0-nightly.20251204023901",
228
229
  "@ledgerhq/hw-app-xrp": "^6.32.7",
229
- "@ledgerhq/hw-transport": "6.31.13",
230
230
  "@ledgerhq/hw-bolos": "^6.32.9",
231
- "@ledgerhq/ledger-cal-service": "^1.9.1-nightly.20251203105022",
232
- "@ledgerhq/ledger-trust-service": "^0.4.3-nightly.20251203105022",
233
- "@ledgerhq/live-config": "^3.2.0",
234
- "@ledgerhq/hw-app-sui": "^1.4.0",
231
+ "@ledgerhq/hw-transport": "6.31.13",
235
232
  "@ledgerhq/hw-transport-mocker": "^6.29.13",
236
- "@ledgerhq/live-countervalues-react": "^0.7.3-nightly.20251203105022",
233
+ "@ledgerhq/ledger-cal-service": "^1.9.1-nightly.20251204023901",
234
+ "@ledgerhq/ledger-trust-service": "^0.4.3-nightly.20251204023901",
235
+ "@ledgerhq/live-config": "^3.2.0",
236
+ "@ledgerhq/live-countervalues": "^0.10.1-nightly.20251204023901",
237
+ "@ledgerhq/live-countervalues-react": "^0.7.3-nightly.20251204023901",
237
238
  "@ledgerhq/live-dmk-shared": "^0.15.0",
238
- "@ledgerhq/live-countervalues": "^0.10.1-nightly.20251203105022",
239
- "@ledgerhq/live-env": "^2.22.0-nightly.20251203105022",
240
- "@ledgerhq/live-hooks": "0.2.0",
241
- "@ledgerhq/live-network": "^2.1.2-nightly.20251203105022",
239
+ "@ledgerhq/live-env": "^2.22.0-nightly.20251204023901",
240
+ "@ledgerhq/live-network": "^2.1.2-nightly.20251204023901",
242
241
  "@ledgerhq/live-promise": "^0.1.1",
243
- "@ledgerhq/live-signer-canton": "^0.5.2-nightly.20251203105022",
244
- "@ledgerhq/live-signer-solana": "^0.6.2-nightly.20251203105022",
245
- "@ledgerhq/live-signer-evm": "^0.10.2-nightly.20251203105022",
242
+ "@ledgerhq/live-signer-canton": "^0.5.2-nightly.20251204023901",
243
+ "@ledgerhq/live-signer-evm": "^0.10.2-nightly.20251204023901",
244
+ "@ledgerhq/live-signer-solana": "^0.6.2-nightly.20251204023901",
245
+ "@ledgerhq/live-wallet": "^0.16.1-nightly.20251204023901",
246
246
  "@ledgerhq/logs": "^6.13.0",
247
- "@ledgerhq/wallet-api-exchange-module": "^0.19.0-nightly.20251203105022",
248
- "@ledgerhq/speculos-transport": "^0.3.0-nightly.20251203105022",
249
- "@ledgerhq/wallet-api-acre-module": "^0.10.0-nightly.20251203105022",
250
- "@ledgerhq/live-wallet": "^0.16.1-nightly.20251203105022"
247
+ "@ledgerhq/live-hooks": "0.2.0",
248
+ "@ledgerhq/wallet-api-acre-module": "^0.10.0-nightly.20251204023901",
249
+ "@ledgerhq/wallet-api-exchange-module": "^0.19.0-nightly.20251204023901",
250
+ "@ledgerhq/speculos-transport": "^0.3.0-nightly.20251204023901"
251
251
  },
252
252
  "devDependencies": {
253
253
  "@solana/web3.js": "1.95.4",
@@ -294,10 +294,10 @@
294
294
  "undici": "6.19.2",
295
295
  "uuid": "^8.3.2",
296
296
  "ws": "7",
297
- "@ledgerhq/device-react": "^0.3.3-nightly.20251203105022",
297
+ "@ledgerhq/device-react": "^0.3.3-nightly.20251204023901",
298
298
  "@ledgerhq/types-cryptoassets": "^7.30.0",
299
299
  "@ledgerhq/types-devices": "^6.27.0",
300
- "@ledgerhq/types-live": "^6.90.0-nightly.20251203105022"
300
+ "@ledgerhq/types-live": "^6.90.0-nightly.20251204023901"
301
301
  },
302
302
  "scripts": {
303
303
  "build": "zx ./scripts/build-ts.mjs",
@@ -0,0 +1,147 @@
1
+ /**
2
+ * @jest-environment jsdom
3
+ */
4
+
5
+ // Mock the bridge to avoid loading all bridge implementations with ESM dependencies
6
+ jest.mock("../../bridge", () => ({
7
+ getCurrencyBridge: jest.fn(),
8
+ }));
9
+
10
+ import { renderHook, act } from "@testing-library/react";
11
+ import { getRemainingTime, useTimeRemaining } from "./react";
12
+
13
+ const MILLISECOND = 1000;
14
+ const SECOND = 1 * MILLISECOND;
15
+ const MINUTE = 60 * SECOND;
16
+ const HOUR = 60 * MINUTE;
17
+ const DAY = 24 * HOUR;
18
+
19
+ describe("getRemainingTime", () => {
20
+ test("should return empty string for negative or zero diff", () => {
21
+ expect(getRemainingTime(0)).toBe("");
22
+ expect(getRemainingTime(-1000)).toBe("");
23
+ expect(getRemainingTime(-100000)).toBe("");
24
+ });
25
+
26
+ test("should format seconds only", () => {
27
+ expect(getRemainingTime(30 * SECOND)).toBe("30s");
28
+ expect(getRemainingTime(59 * SECOND)).toBe("59s");
29
+ expect(getRemainingTime(1 * SECOND)).toBe("1s");
30
+ });
31
+
32
+ test("should format minutes and seconds", () => {
33
+ expect(getRemainingTime(1 * MINUTE + 30 * SECOND)).toBe("1m 30s");
34
+ expect(getRemainingTime(5 * MINUTE + 15 * SECOND)).toBe("5m 15s");
35
+ expect(getRemainingTime(59 * MINUTE + 59 * SECOND)).toBe("59m 59s");
36
+ });
37
+
38
+ test("should format hours, minutes and seconds", () => {
39
+ expect(getRemainingTime(1 * HOUR + 30 * MINUTE + 15 * SECOND)).toBe("1h 30m 15s");
40
+ expect(getRemainingTime(2 * HOUR + 5 * MINUTE + 10 * SECOND)).toBe("2h 5m 10s");
41
+ expect(getRemainingTime(23 * HOUR + 59 * MINUTE + 59 * SECOND)).toBe("23h 59m 59s");
42
+ });
43
+
44
+ test("should format days, hours, minutes and seconds", () => {
45
+ expect(getRemainingTime(1 * DAY + 2 * HOUR + 30 * MINUTE + 15 * SECOND)).toBe("1d 2h 30m 15s");
46
+ expect(getRemainingTime(5 * DAY + 10 * HOUR + 5 * MINUTE + 10 * SECOND)).toBe("5d 10h 5m 10s");
47
+ expect(getRemainingTime(10 * DAY + 23 * HOUR + 59 * MINUTE + 59 * SECOND)).toBe(
48
+ "10d 23h 59m 59s",
49
+ );
50
+ });
51
+
52
+ test("should skip zero values", () => {
53
+ expect(getRemainingTime(1 * DAY)).toBe("1d 0s");
54
+ expect(getRemainingTime(1 * HOUR)).toBe("1h 0s");
55
+ expect(getRemainingTime(1 * MINUTE)).toBe("1m 0s");
56
+ expect(getRemainingTime(1 * DAY + 1 * MINUTE)).toBe("1d 1m 0s");
57
+ expect(getRemainingTime(1 * DAY + 1 * HOUR)).toBe("1d 1h 0s");
58
+ });
59
+ });
60
+
61
+ describe("useTimeRemaining", () => {
62
+ beforeAll(() => {
63
+ jest.useFakeTimers();
64
+ });
65
+
66
+ afterAll(() => {
67
+ jest.runOnlyPendingTimers();
68
+ jest.useRealTimers();
69
+ });
70
+
71
+ beforeEach(() => {
72
+ jest.setSystemTime(new Date("2024-01-01T12:00:00Z"));
73
+ });
74
+
75
+ test("should return empty string when proposal is null", () => {
76
+ const { result } = renderHook(() => useTimeRemaining());
77
+ expect(result.current).toBe("");
78
+ });
79
+
80
+ test("should return empty string when proposal is expired", () => {
81
+ const proposal = {
82
+ expiresAt: (Date.now() - 1000) * 1000, // 1 second ago
83
+ isExpired: true,
84
+ };
85
+ const { result } = renderHook(() => useTimeRemaining(proposal.expiresAt, proposal.isExpired));
86
+ expect(result.current).toBe("");
87
+ });
88
+
89
+ test("should return formatted time remaining for valid proposal", () => {
90
+ const expiresAt = Date.now() + 2 * HOUR + 30 * MINUTE + 15 * SECOND;
91
+ const { result } = renderHook(() => useTimeRemaining(expiresAt * 1000));
92
+ expect(result.current).toBe("2h 30m 15s");
93
+ });
94
+
95
+ test("should update time remaining every second", () => {
96
+ const expiresAt = Date.now() + 2 * MINUTE + 30 * SECOND;
97
+ const { result } = renderHook(() => useTimeRemaining(expiresAt * 1000));
98
+
99
+ expect(result.current).toBe("2m 30s");
100
+
101
+ act(() => {
102
+ jest.advanceTimersByTime(1 * SECOND);
103
+ });
104
+ expect(result.current).toBe("2m 29s");
105
+
106
+ act(() => {
107
+ jest.advanceTimersByTime(1 * SECOND);
108
+ });
109
+ expect(result.current).toBe("2m 28s");
110
+
111
+ act(() => {
112
+ jest.advanceTimersByTime(30 * SECOND);
113
+ });
114
+ expect(result.current).toBe("1m 58s");
115
+ });
116
+
117
+ test("should return empty string when time expires", () => {
118
+ const expiresAt = Date.now() + 2 * SECOND;
119
+ const { result } = renderHook(() => useTimeRemaining(expiresAt * 1000));
120
+
121
+ expect(result.current).toBe("2s");
122
+
123
+ act(() => {
124
+ jest.advanceTimersByTime(1 * SECOND);
125
+ });
126
+ expect(result.current).toBe("1s");
127
+
128
+ act(() => {
129
+ jest.advanceTimersByTime(1 * SECOND);
130
+ });
131
+ expect(result.current).toBe("");
132
+ });
133
+
134
+ test("should clear interval on unmount", () => {
135
+ const expiresAt = Date.now() + 1 * HOUR;
136
+ const { result, unmount } = renderHook(() => useTimeRemaining(expiresAt * 1000));
137
+
138
+ expect(result.current).toBe("1h 0s");
139
+
140
+ unmount();
141
+
142
+ // Advance time after unmount - should not cause errors
143
+ act(() => {
144
+ jest.advanceTimersByTime(1 * SECOND);
145
+ });
146
+ });
147
+ });
@@ -1,4 +1,4 @@
1
- import { useCallback } from "react";
1
+ import { useCallback, useState, useEffect } from "react";
2
2
  import { getCurrencyBridge } from "../../bridge";
3
3
  import { CantonCurrencyBridge } from "@ledgerhq/coin-canton/types";
4
4
  import { CryptoCurrency } from "@ledgerhq/types-cryptoassets";
@@ -48,3 +48,47 @@ export function useCantonAcceptOrRejectOffer({
48
48
 
49
49
  return transferInstruction;
50
50
  }
51
+
52
+ export const getRemainingTime = (diff: number): string => {
53
+ if (diff <= 0) {
54
+ return "";
55
+ }
56
+ const days = Math.floor(diff / (1000 * 60 * 60 * 24));
57
+ const hours = Math.floor((diff % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
58
+ const minutes = Math.floor((diff % (1000 * 60 * 60)) / (1000 * 60));
59
+ const seconds = Math.floor((diff % (1000 * 60)) / 1000);
60
+
61
+ let format = "";
62
+ if (days > 0) format += `${days}d `;
63
+ if (hours > 0) format += `${hours}h `;
64
+ if (minutes > 0) format += `${minutes}m `;
65
+ format += `${seconds}s`;
66
+
67
+ return format.trim();
68
+ };
69
+
70
+ export const useTimeRemaining = (expiresAtMicros = 0, isExpired = false): string => {
71
+ const [timeRemaining, setTimeRemaining] = useState<string>("");
72
+
73
+ useEffect(() => {
74
+ if (expiresAtMicros <= 0 || isExpired) {
75
+ setTimeRemaining("");
76
+ return;
77
+ }
78
+
79
+ const updateTimeRemaining = () => {
80
+ const now = Date.now();
81
+ const expiresAt = expiresAtMicros / 1000;
82
+ const diff = expiresAt - now;
83
+
84
+ setTimeRemaining(getRemainingTime(diff));
85
+ };
86
+
87
+ updateTimeRemaining();
88
+ const interval = setInterval(updateTimeRemaining, 1000);
89
+
90
+ return () => clearInterval(interval);
91
+ }, [expiresAtMicros, isExpired]);
92
+
93
+ return timeRemaining;
94
+ };
@@ -11,12 +11,12 @@ export default function getEnvironmentParams(
11
11
  ): EnvironmentParams {
12
12
  return {
13
13
  cloudSyncApiBaseUrl:
14
- environment && environment === "PROD"
15
- ? getEnv("CLOUD_SYNC_API_PROD")
16
- : getEnv("CLOUD_SYNC_API_STAGING"),
14
+ environment && environment === "STAGING"
15
+ ? getEnv("CLOUD_SYNC_API_STAGING")
16
+ : getEnv("CLOUD_SYNC_API_PROD"),
17
17
  trustchainApiBaseUrl:
18
- environment && environment === "PROD"
19
- ? getEnv("TRUSTCHAIN_API_PROD")
20
- : getEnv("TRUSTCHAIN_API_STAGING"),
18
+ environment && environment === "STAGING"
19
+ ? getEnv("TRUSTCHAIN_API_STAGING")
20
+ : getEnv("TRUSTCHAIN_API_PROD"),
21
21
  };
22
22
  }