diff --git a/src/i18n/i18n.ts b/src/i18n/i18n.ts index 244bc102..75bb7fc2 100644 --- a/src/i18n/i18n.ts +++ b/src/i18n/i18n.ts @@ -225,6 +225,7 @@ const dict = { will_receive: "Will receive", refund_available_in: "Refund will be available in {{ blocks }} blocks", no_wallet_connected: "No wallet connected", + qr_scan_supported: "This browser doesn't support refunds via QR", }, de: { language: "Deutsch", @@ -462,6 +463,8 @@ const dict = { will_receive: "Sie erhalten", refund_available_in: "Rückerstattung möglich in {{ blocks }} Blöcken", no_wallet_connected: "Kein Wallet verbunden", + qr_scan_supported: + "Dieser Browser unterstützt keine Erstattungen über QR", }, es: { language: "Español", @@ -695,6 +698,8 @@ const dict = { will_receive: "Recibirá", refund_available_in: "Reembolso disponible en {{ blocks }} bloques", no_wallet_connected: "No hay monedero conectado", + qr_scan_supported: + "Este navegador no admite devoluciones a través de QR", }, zh: { language: "中文", @@ -902,6 +907,7 @@ const dict = { will_receive: "将收到", refund_available_in: "退款将分 {{ blocks }} 区块提供", no_wallet_connected: "未连接钱包", + qr_scan_supported: "此浏览器不支持通过 QR 退款", }, ja: { language: "日本語", @@ -1134,6 +1140,7 @@ const dict = { will_receive: "受信予定", refund_available_in: "返金は {{ blocks }} つのブロックに分かれる", no_wallet_connected: "財布はつながっていない!", + qr_scan_supported: "このブラウザはQRによる払い戻しに対応していません。", }, }; diff --git a/src/pages/Refund.tsx b/src/pages/Refund.tsx index 6f7edaa7..46ce02be 100644 --- a/src/pages/Refund.tsx +++ b/src/pages/Refund.tsx @@ -20,10 +20,16 @@ import { LogRefundData, scanLogsForPossibleRefunds, } from "../utils/contractLogs"; +import { qrScanProbe } from "../utils/qrScanProbe"; import { validateRefundFile } from "../utils/refundFile"; import { SomeSwap } from "../utils/swapCreator"; import ErrorWasm from "./ErrorWasm"; +enum RefundError { + InvalidData, + QrScanNotSupported, +} + const Refund = () => { const navigate = useNavigate(); const { getSwap, getSwaps, updateSwapStatus, wasmSupported, t } = @@ -31,7 +37,9 @@ const Refund = () => { const { signer, providers, getEtherSwap } = useWeb3Signer(); const [swapFound, setSwapFound] = createSignal(null); - const [refundInvalid, setRefundInvalid] = createSignal(false); + const [refundInvalid, setRefundInvalid] = createSignal< + RefundError | undefined + >(undefined); const [refundJson, setRefundJson] = createSignal(null); const [refundTxId, setRefundTxId] = createSignal(""); @@ -52,47 +60,47 @@ const Refund = () => { } setRefundJson(data); - setRefundInvalid(false); + setRefundInvalid(undefined); } catch (e) { log.warn("Refund json validation failed", e); - setRefundInvalid(true); + setRefundInvalid(RefundError.InvalidData); input.setCustomValidity(t("invalid_refund_file")); } }; - const uploadChange = (e: Event) => { + const uploadChange = async (e: Event) => { const input = e.currentTarget as HTMLInputElement; const inputFile = input.files[0]; input.setCustomValidity(""); setRefundJson(null); setSwapFound(null); - setRefundInvalid(false); + setRefundInvalid(undefined); if (["image/png", "image/jpg", "image/jpeg"].includes(inputFile.type)) { - QrScanner.scanImage(inputFile, { returnDetailedScanResult: true }) - .then( - async (result) => - await checkRefundJsonKeys( - input, - JSON.parse(result.data), - ), - ) - .catch((e) => { - log.error("invalid QR code upload", e); - setRefundInvalid(true); - input.setCustomValidity(t("invalid_refund_file")); + if (!(await qrScanProbe())) { + setRefundInvalid(RefundError.QrScanNotSupported); + return; + } + + try { + const res = await QrScanner.scanImage(inputFile, { + returnDetailedScanResult: true, }); + await checkRefundJsonKeys(input, JSON.parse(res.data)); + } catch (e) { + log.error("invalid QR code upload", e); + setRefundInvalid(RefundError.InvalidData); + input.setCustomValidity(t("invalid_refund_file")); + } } else { - inputFile - .text() - .then(async (result) => { - await checkRefundJsonKeys(input, JSON.parse(result)); - }) - .catch((e) => { - log.error("invalid file upload", e); - setRefundInvalid(true); - input.setCustomValidity(t("invalid_refund_file")); - }); + try { + const data = await inputFile.text(); + await checkRefundJsonKeys(input, JSON.parse(data)); + } catch (e) { + log.error("invalid file upload", e); + setRefundInvalid(RefundError.InvalidData); + input.setCustomValidity(t("invalid_refund_file")); + } } }; @@ -249,10 +257,18 @@ const Refund = () => { {t("open_swap")} - +
diff --git a/src/utils/qrScanProbe.ts b/src/utils/qrScanProbe.ts new file mode 100644 index 00000000..4bcf64da --- /dev/null +++ b/src/utils/qrScanProbe.ts @@ -0,0 +1,19 @@ +import log from "loglevel"; +import QrScanner from "qr-scanner"; + +import { formatError } from "./errors"; + +const image = + ""; + +export const qrScanProbe = async () => { + try { + const res = await QrScanner.scanImage(new URL(image), { + returnDetailedScanResult: true, + }); + return res.data === "21"; + } catch (e) { + log.warn(`QR code scanning probe failed: ${formatError(e)}`); + return false; + } +};