mirror of
https://github.com/cake-tech/cake_wallet.git
synced 2025-01-24 11:36:21 +00:00
Merge branch 'main' into wownero
This commit is contained in:
commit
3c4f7431fc
12 changed files with 157 additions and 141 deletions
|
@ -2,9 +2,9 @@ import 'dart:convert';
|
||||||
import 'dart:math';
|
import 'dart:math';
|
||||||
import 'dart:typed_data';
|
import 'dart:typed_data';
|
||||||
import 'package:crypto/crypto.dart';
|
import 'package:crypto/crypto.dart';
|
||||||
import 'package:unorm_dart/unorm_dart.dart' as unorm;
|
|
||||||
import 'package:cryptography/cryptography.dart' as cryptography;
|
import 'package:cryptography/cryptography.dart' as cryptography;
|
||||||
import 'package:cw_core/sec_random_native.dart';
|
import 'package:cw_core/sec_random_native.dart';
|
||||||
|
import 'package:cw_core/utils/text_normalizer.dart';
|
||||||
|
|
||||||
const segwit = '100';
|
const segwit = '100';
|
||||||
final wordlist = englishWordlist;
|
final wordlist = englishWordlist;
|
||||||
|
@ -137,121 +137,6 @@ bool validateMnemonic(String mnemonic, {String prefix = segwit}) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
final COMBININGCODEPOINTS = combiningcodepoints();
|
|
||||||
|
|
||||||
List<int> combiningcodepoints() {
|
|
||||||
final source = '300:34e|350:36f|483:487|591:5bd|5bf|5c1|5c2|5c4|5c5|5c7|610:61a|64b:65f|670|' +
|
|
||||||
'6d6:6dc|6df:6e4|6e7|6e8|6ea:6ed|711|730:74a|7eb:7f3|816:819|81b:823|825:827|' +
|
|
||||||
'829:82d|859:85b|8d4:8e1|8e3:8ff|93c|94d|951:954|9bc|9cd|a3c|a4d|abc|acd|b3c|' +
|
|
||||||
'b4d|bcd|c4d|c55|c56|cbc|ccd|d4d|dca|e38:e3a|e48:e4b|eb8|eb9|ec8:ecb|f18|f19|' +
|
|
||||||
'f35|f37|f39|f71|f72|f74|f7a:f7d|f80|f82:f84|f86|f87|fc6|1037|1039|103a|108d|' +
|
|
||||||
'135d:135f|1714|1734|17d2|17dd|18a9|1939:193b|1a17|1a18|1a60|1a75:1a7c|1a7f|' +
|
|
||||||
'1ab0:1abd|1b34|1b44|1b6b:1b73|1baa|1bab|1be6|1bf2|1bf3|1c37|1cd0:1cd2|' +
|
|
||||||
'1cd4:1ce0|1ce2:1ce8|1ced|1cf4|1cf8|1cf9|1dc0:1df5|1dfb:1dff|20d0:20dc|20e1|' +
|
|
||||||
'20e5:20f0|2cef:2cf1|2d7f|2de0:2dff|302a:302f|3099|309a|a66f|a674:a67d|a69e|' +
|
|
||||||
'a69f|a6f0|a6f1|a806|a8c4|a8e0:a8f1|a92b:a92d|a953|a9b3|a9c0|aab0|aab2:aab4|' +
|
|
||||||
'aab7|aab8|aabe|aabf|aac1|aaf6|abed|fb1e|fe20:fe2f|101fd|102e0|10376:1037a|' +
|
|
||||||
'10a0d|10a0f|10a38:10a3a|10a3f|10ae5|10ae6|11046|1107f|110b9|110ba|11100:11102|' +
|
|
||||||
'11133|11134|11173|111c0|111ca|11235|11236|112e9|112ea|1133c|1134d|11366:1136c|' +
|
|
||||||
'11370:11374|11442|11446|114c2|114c3|115bf|115c0|1163f|116b6|116b7|1172b|11c3f|' +
|
|
||||||
'16af0:16af4|16b30:16b36|1bc9e|1d165:1d169|1d16d:1d172|1d17b:1d182|1d185:1d18b|' +
|
|
||||||
'1d1aa:1d1ad|1d242:1d244|1e000:1e006|1e008:1e018|1e01b:1e021|1e023|1e024|' +
|
|
||||||
'1e026:1e02a|1e8d0:1e8d6|1e944:1e94a';
|
|
||||||
|
|
||||||
return source.split('|').map((e) {
|
|
||||||
if (e.contains(':')) {
|
|
||||||
return e.split(':').map((hex) => int.parse(hex, radix: 16));
|
|
||||||
}
|
|
||||||
|
|
||||||
return int.parse(e, radix: 16);
|
|
||||||
}).fold(<int>[], (List<int> acc, element) {
|
|
||||||
if (element is List) {
|
|
||||||
for (var i = element[0] as int; i <= (element[1] as int); i++) {}
|
|
||||||
} else if (element is int) {
|
|
||||||
acc.add(element);
|
|
||||||
}
|
|
||||||
|
|
||||||
return acc;
|
|
||||||
}).toList();
|
|
||||||
}
|
|
||||||
|
|
||||||
String removeCombiningCharacters(String source) {
|
|
||||||
return source
|
|
||||||
.split('')
|
|
||||||
.where((char) => !COMBININGCODEPOINTS.contains(char.codeUnits.first))
|
|
||||||
.join('');
|
|
||||||
}
|
|
||||||
|
|
||||||
bool isCJK(String char) {
|
|
||||||
final n = char.codeUnitAt(0);
|
|
||||||
|
|
||||||
for (var x in CJKINTERVALS) {
|
|
||||||
final imin = x[0] as num;
|
|
||||||
final imax = x[1] as num;
|
|
||||||
|
|
||||||
if (n >= imin && n <= imax) return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
String removeCJKSpaces(String source) {
|
|
||||||
final splitted = source.split('');
|
|
||||||
final filtered = <String>[];
|
|
||||||
|
|
||||||
for (var i = 0; i < splitted.length; i++) {
|
|
||||||
final char = splitted[i];
|
|
||||||
final isSpace = char.trim() == '';
|
|
||||||
final prevIsCJK = i != 0 && isCJK(splitted[i - 1]);
|
|
||||||
final nextIsCJK = i != splitted.length - 1 && isCJK(splitted[i + 1]);
|
|
||||||
|
|
||||||
if (!(isSpace && prevIsCJK && nextIsCJK)) {
|
|
||||||
filtered.add(char);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return filtered.join('');
|
|
||||||
}
|
|
||||||
|
|
||||||
String normalizeText(String source) {
|
|
||||||
final res =
|
|
||||||
removeCombiningCharacters(unorm.nfkd(source).toLowerCase()).trim().split('/\s+/').join(' ');
|
|
||||||
|
|
||||||
return removeCJKSpaces(res);
|
|
||||||
}
|
|
||||||
|
|
||||||
const CJKINTERVALS = [
|
|
||||||
[0x4e00, 0x9fff, 'CJK Unified Ideographs'],
|
|
||||||
[0x3400, 0x4dbf, 'CJK Unified Ideographs Extension A'],
|
|
||||||
[0x20000, 0x2a6df, 'CJK Unified Ideographs Extension B'],
|
|
||||||
[0x2a700, 0x2b73f, 'CJK Unified Ideographs Extension C'],
|
|
||||||
[0x2b740, 0x2b81f, 'CJK Unified Ideographs Extension D'],
|
|
||||||
[0xf900, 0xfaff, 'CJK Compatibility Ideographs'],
|
|
||||||
[0x2f800, 0x2fa1d, 'CJK Compatibility Ideographs Supplement'],
|
|
||||||
[0x3190, 0x319f, 'Kanbun'],
|
|
||||||
[0x2e80, 0x2eff, 'CJK Radicals Supplement'],
|
|
||||||
[0x2f00, 0x2fdf, 'CJK Radicals'],
|
|
||||||
[0x31c0, 0x31ef, 'CJK Strokes'],
|
|
||||||
[0x2ff0, 0x2fff, 'Ideographic Description Characters'],
|
|
||||||
[0xe0100, 0xe01ef, 'Variation Selectors Supplement'],
|
|
||||||
[0x3100, 0x312f, 'Bopomofo'],
|
|
||||||
[0x31a0, 0x31bf, 'Bopomofo Extended'],
|
|
||||||
[0xff00, 0xffef, 'Halfwidth and Fullwidth Forms'],
|
|
||||||
[0x3040, 0x309f, 'Hiragana'],
|
|
||||||
[0x30a0, 0x30ff, 'Katakana'],
|
|
||||||
[0x31f0, 0x31ff, 'Katakana Phonetic Extensions'],
|
|
||||||
[0x1b000, 0x1b0ff, 'Kana Supplement'],
|
|
||||||
[0xac00, 0xd7af, 'Hangul Syllables'],
|
|
||||||
[0x1100, 0x11ff, 'Hangul Jamo'],
|
|
||||||
[0xa960, 0xa97f, 'Hangul Jamo Extended A'],
|
|
||||||
[0xd7b0, 0xd7ff, 'Hangul Jamo Extended B'],
|
|
||||||
[0x3130, 0x318f, 'Hangul Compatibility Jamo'],
|
|
||||||
[0xa4d0, 0xa4ff, 'Lisu'],
|
|
||||||
[0x16f00, 0x16f9f, 'Miao'],
|
|
||||||
[0xa000, 0xa48f, 'Yi Syllables'],
|
|
||||||
[0xa490, 0xa4cf, 'Yi Radicals'],
|
|
||||||
];
|
|
||||||
|
|
||||||
final englishWordlist = <String>[
|
final englishWordlist = <String>[
|
||||||
'abandon',
|
'abandon',
|
||||||
'ability',
|
'ability',
|
||||||
|
|
|
@ -863,7 +863,7 @@ packages:
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.3.2"
|
version: "1.3.2"
|
||||||
unorm_dart:
|
unorm_dart:
|
||||||
dependency: "direct main"
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
name: unorm_dart
|
name: unorm_dart
|
||||||
sha256: "5b35bff83fce4d76467641438f9e867dc9bcfdb8c1694854f230579d68cd8f4b"
|
sha256: "5b35bff83fce4d76467641438f9e867dc9bcfdb8c1694854f230579d68cd8f4b"
|
||||||
|
|
|
@ -28,7 +28,6 @@ dependencies:
|
||||||
url: https://github.com/cake-tech/bitbox-flutter.git
|
url: https://github.com/cake-tech/bitbox-flutter.git
|
||||||
ref: Add-Support-For-OP-Return-data
|
ref: Add-Support-For-OP-Return-data
|
||||||
rxdart: ^0.27.5
|
rxdart: ^0.27.5
|
||||||
unorm_dart: ^0.2.0
|
|
||||||
cryptography: ^2.0.5
|
cryptography: ^2.0.5
|
||||||
bitcoin_base:
|
bitcoin_base:
|
||||||
git:
|
git:
|
||||||
|
|
117
cw_core/lib/utils/text_normalizer.dart
Normal file
117
cw_core/lib/utils/text_normalizer.dart
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
import 'package:unorm_dart/unorm_dart.dart' as unorm;
|
||||||
|
|
||||||
|
const CJKINTERVALS = [
|
||||||
|
[0x4e00, 0x9fff, 'CJK Unified Ideographs'],
|
||||||
|
[0x3400, 0x4dbf, 'CJK Unified Ideographs Extension A'],
|
||||||
|
[0x20000, 0x2a6df, 'CJK Unified Ideographs Extension B'],
|
||||||
|
[0x2a700, 0x2b73f, 'CJK Unified Ideographs Extension C'],
|
||||||
|
[0x2b740, 0x2b81f, 'CJK Unified Ideographs Extension D'],
|
||||||
|
[0xf900, 0xfaff, 'CJK Compatibility Ideographs'],
|
||||||
|
[0x2f800, 0x2fa1d, 'CJK Compatibility Ideographs Supplement'],
|
||||||
|
[0x3190, 0x319f, 'Kanbun'],
|
||||||
|
[0x2e80, 0x2eff, 'CJK Radicals Supplement'],
|
||||||
|
[0x2f00, 0x2fdf, 'CJK Radicals'],
|
||||||
|
[0x31c0, 0x31ef, 'CJK Strokes'],
|
||||||
|
[0x2ff0, 0x2fff, 'Ideographic Description Characters'],
|
||||||
|
[0xe0100, 0xe01ef, 'Variation Selectors Supplement'],
|
||||||
|
[0x3100, 0x312f, 'Bopomofo'],
|
||||||
|
[0x31a0, 0x31bf, 'Bopomofo Extended'],
|
||||||
|
[0xff00, 0xffef, 'Halfwidth and Fullwidth Forms'],
|
||||||
|
[0x3040, 0x309f, 'Hiragana'],
|
||||||
|
[0x30a0, 0x30ff, 'Katakana'],
|
||||||
|
[0x31f0, 0x31ff, 'Katakana Phonetic Extensions'],
|
||||||
|
[0x1b000, 0x1b0ff, 'Kana Supplement'],
|
||||||
|
[0xac00, 0xd7af, 'Hangul Syllables'],
|
||||||
|
[0x1100, 0x11ff, 'Hangul Jamo'],
|
||||||
|
[0xa960, 0xa97f, 'Hangul Jamo Extended A'],
|
||||||
|
[0xd7b0, 0xd7ff, 'Hangul Jamo Extended B'],
|
||||||
|
[0x3130, 0x318f, 'Hangul Compatibility Jamo'],
|
||||||
|
[0xa4d0, 0xa4ff, 'Lisu'],
|
||||||
|
[0x16f00, 0x16f9f, 'Miao'],
|
||||||
|
[0xa000, 0xa48f, 'Yi Syllables'],
|
||||||
|
[0xa490, 0xa4cf, 'Yi Radicals'],
|
||||||
|
];
|
||||||
|
|
||||||
|
final COMBININGCODEPOINTS = combiningcodepoints();
|
||||||
|
|
||||||
|
List<int> combiningcodepoints() {
|
||||||
|
final source = '300:34e|350:36f|483:487|591:5bd|5bf|5c1|5c2|5c4|5c5|5c7|610:61a|64b:65f|670|' +
|
||||||
|
'6d6:6dc|6df:6e4|6e7|6e8|6ea:6ed|711|730:74a|7eb:7f3|816:819|81b:823|825:827|' +
|
||||||
|
'829:82d|859:85b|8d4:8e1|8e3:8ff|93c|94d|951:954|9bc|9cd|a3c|a4d|abc|acd|b3c|' +
|
||||||
|
'b4d|bcd|c4d|c55|c56|cbc|ccd|d4d|dca|e38:e3a|e48:e4b|eb8|eb9|ec8:ecb|f18|f19|' +
|
||||||
|
'f35|f37|f39|f71|f72|f74|f7a:f7d|f80|f82:f84|f86|f87|fc6|1037|1039|103a|108d|' +
|
||||||
|
'135d:135f|1714|1734|17d2|17dd|18a9|1939:193b|1a17|1a18|1a60|1a75:1a7c|1a7f|' +
|
||||||
|
'1ab0:1abd|1b34|1b44|1b6b:1b73|1baa|1bab|1be6|1bf2|1bf3|1c37|1cd0:1cd2|' +
|
||||||
|
'1cd4:1ce0|1ce2:1ce8|1ced|1cf4|1cf8|1cf9|1dc0:1df5|1dfb:1dff|20d0:20dc|20e1|' +
|
||||||
|
'20e5:20f0|2cef:2cf1|2d7f|2de0:2dff|302a:302f|3099|309a|a66f|a674:a67d|a69e|' +
|
||||||
|
'a69f|a6f0|a6f1|a806|a8c4|a8e0:a8f1|a92b:a92d|a953|a9b3|a9c0|aab0|aab2:aab4|' +
|
||||||
|
'aab7|aab8|aabe|aabf|aac1|aaf6|abed|fb1e|fe20:fe2f|101fd|102e0|10376:1037a|' +
|
||||||
|
'10a0d|10a0f|10a38:10a3a|10a3f|10ae5|10ae6|11046|1107f|110b9|110ba|11100:11102|' +
|
||||||
|
'11133|11134|11173|111c0|111ca|11235|11236|112e9|112ea|1133c|1134d|11366:1136c|' +
|
||||||
|
'11370:11374|11442|11446|114c2|114c3|115bf|115c0|1163f|116b6|116b7|1172b|11c3f|' +
|
||||||
|
'16af0:16af4|16b30:16b36|1bc9e|1d165:1d169|1d16d:1d172|1d17b:1d182|1d185:1d18b|' +
|
||||||
|
'1d1aa:1d1ad|1d242:1d244|1e000:1e006|1e008:1e018|1e01b:1e021|1e023|1e024|' +
|
||||||
|
'1e026:1e02a|1e8d0:1e8d6|1e944:1e94a';
|
||||||
|
|
||||||
|
return source.split('|').map((e) {
|
||||||
|
if (e.contains(':')) {
|
||||||
|
return e.split(':').map((hex) => int.parse(hex, radix: 16));
|
||||||
|
}
|
||||||
|
|
||||||
|
return int.parse(e, radix: 16);
|
||||||
|
}).fold(<int>[], (List<int> acc, element) {
|
||||||
|
if (element is List) {
|
||||||
|
for (var i = element[0] as int; i <= (element[1] as int); i++) {}
|
||||||
|
} else if (element is int) {
|
||||||
|
acc.add(element);
|
||||||
|
}
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
}).toList();
|
||||||
|
}
|
||||||
|
|
||||||
|
String _removeCombiningCharacters(String source) {
|
||||||
|
return source
|
||||||
|
.split('')
|
||||||
|
.where((char) => !COMBININGCODEPOINTS.contains(char.codeUnits.first))
|
||||||
|
.join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
String _removeCJKSpaces(String source) {
|
||||||
|
final splitted = source.split('');
|
||||||
|
final filtered = <String>[];
|
||||||
|
|
||||||
|
for (var i = 0; i < splitted.length; i++) {
|
||||||
|
final char = splitted[i];
|
||||||
|
final isSpace = char.trim() == '';
|
||||||
|
final prevIsCJK = i != 0 && _isCJK(splitted[i - 1]);
|
||||||
|
final nextIsCJK = i != splitted.length - 1 && _isCJK(splitted[i + 1]);
|
||||||
|
|
||||||
|
if (!(isSpace && prevIsCJK && nextIsCJK)) {
|
||||||
|
filtered.add(char);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return filtered.join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
bool _isCJK(String char) {
|
||||||
|
final n = char.codeUnitAt(0);
|
||||||
|
|
||||||
|
for (var x in CJKINTERVALS) {
|
||||||
|
final imin = x[0] as num;
|
||||||
|
final imax = x[1] as num;
|
||||||
|
|
||||||
|
if (n >= imin && n <= imax) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This method normalize text which transforms Unicode text into an equivalent decomposed form, allowing for easier sorting and searching of text.
|
||||||
|
String normalizeText(String source) {
|
||||||
|
final res =
|
||||||
|
_removeCombiningCharacters(unorm.nfkd(source).toLowerCase()).trim().split('/\s+/').join(' ');
|
||||||
|
|
||||||
|
return _removeCJKSpaces(res);
|
||||||
|
}
|
|
@ -656,6 +656,14 @@ packages:
|
||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.3.2"
|
version: "1.3.2"
|
||||||
|
unorm_dart:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: unorm_dart
|
||||||
|
sha256: "5b35bff83fce4d76467641438f9e867dc9bcfdb8c1694854f230579d68cd8f4b"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.2.0"
|
||||||
vector_math:
|
vector_math:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|
|
@ -20,6 +20,7 @@ dependencies:
|
||||||
intl: ^0.18.0
|
intl: ^0.18.0
|
||||||
encrypt: ^5.0.1
|
encrypt: ^5.0.1
|
||||||
socks5_proxy: ^1.0.4
|
socks5_proxy: ^1.0.4
|
||||||
|
unorm_dart: ^0.3.0
|
||||||
# tor:
|
# tor:
|
||||||
# git:
|
# git:
|
||||||
# url: https://github.com/cake-tech/tor.git
|
# url: https://github.com/cake-tech/tor.git
|
||||||
|
|
|
@ -19,7 +19,7 @@ dependencies:
|
||||||
flutter_mobx: ^2.0.6+1
|
flutter_mobx: ^2.0.6+1
|
||||||
intl: ^0.18.0
|
intl: ^0.18.0
|
||||||
encrypt: ^5.0.1
|
encrypt: ^5.0.1
|
||||||
polyseed: ^0.0.4
|
polyseed: ^0.0.5
|
||||||
cw_core:
|
cw_core:
|
||||||
path: ../cw_core
|
path: ../cw_core
|
||||||
monero:
|
monero:
|
||||||
|
|
|
@ -1,16 +1,16 @@
|
||||||
import 'package:cake_wallet/bitcoin/bitcoin.dart';
|
import 'package:cake_wallet/bitcoin/bitcoin.dart';
|
||||||
import 'package:cake_wallet/ethereum/ethereum.dart';
|
|
||||||
import 'package:cake_wallet/haven/haven.dart';
|
|
||||||
import 'package:cake_wallet/core/validator.dart';
|
import 'package:cake_wallet/core/validator.dart';
|
||||||
import 'package:cake_wallet/entities/mnemonic_item.dart';
|
import 'package:cake_wallet/entities/mnemonic_item.dart';
|
||||||
|
import 'package:cake_wallet/ethereum/ethereum.dart';
|
||||||
|
import 'package:cake_wallet/haven/haven.dart';
|
||||||
|
import 'package:cake_wallet/monero/monero.dart';
|
||||||
|
import 'package:cake_wallet/nano/nano.dart';
|
||||||
import 'package:cake_wallet/polygon/polygon.dart';
|
import 'package:cake_wallet/polygon/polygon.dart';
|
||||||
import 'package:cake_wallet/solana/solana.dart';
|
import 'package:cake_wallet/solana/solana.dart';
|
||||||
import 'package:cake_wallet/tron/tron.dart';
|
import 'package:cake_wallet/tron/tron.dart';
|
||||||
import 'package:cake_wallet/wownero/wownero.dart';
|
import 'package:cake_wallet/wownero/wownero.dart';
|
||||||
import 'package:cw_core/wallet_type.dart';
|
|
||||||
import 'package:cake_wallet/monero/monero.dart';
|
|
||||||
import 'package:cake_wallet/nano/nano.dart';
|
|
||||||
import 'package:cake_wallet/utils/language_list.dart';
|
import 'package:cake_wallet/utils/language_list.dart';
|
||||||
|
import 'package:cw_core/wallet_type.dart';
|
||||||
|
|
||||||
class SeedValidator extends Validator<MnemonicItem> {
|
class SeedValidator extends Validator<MnemonicItem> {
|
||||||
SeedValidator({required this.type, required this.language})
|
SeedValidator({required this.type, required this.language})
|
||||||
|
@ -51,6 +51,9 @@ class SeedValidator extends Validator<MnemonicItem> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool needsNormalization(String language) =>
|
||||||
|
["POLYSEED_French", "POLYSEED_Spanish"].contains(language);
|
||||||
|
|
||||||
static List<String> getBitcoinWordList(String language) {
|
static List<String> getBitcoinWordList(String language) {
|
||||||
assert(language.toLowerCase() == LanguageList.english.toLowerCase());
|
assert(language.toLowerCase() == LanguageList.english.toLowerCase());
|
||||||
return bitcoin!.getWordList();
|
return bitcoin!.getWordList();
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
|
import 'package:cake_wallet/core/seed_validator.dart';
|
||||||
import 'package:cake_wallet/generated/i18n.dart';
|
import 'package:cake_wallet/generated/i18n.dart';
|
||||||
import 'package:cake_wallet/themes/extensions/cake_text_theme.dart';
|
|
||||||
import 'package:cw_core/wallet_type.dart';
|
|
||||||
import 'package:cake_wallet/src/widgets/validable_annotated_editable_text.dart';
|
import 'package:cake_wallet/src/widgets/validable_annotated_editable_text.dart';
|
||||||
|
import 'package:cake_wallet/themes/extensions/cake_text_theme.dart';
|
||||||
|
import 'package:cake_wallet/themes/extensions/send_page_theme.dart';
|
||||||
|
import 'package:cw_core/wallet_type.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:cake_wallet/core/seed_validator.dart';
|
|
||||||
import 'package:cake_wallet/themes/extensions/send_page_theme.dart';
|
|
||||||
|
|
||||||
class SeedWidget extends StatefulWidget {
|
class SeedWidget extends StatefulWidget {
|
||||||
SeedWidget({
|
SeedWidget({
|
||||||
|
@ -23,7 +23,6 @@ class SeedWidget extends StatefulWidget {
|
||||||
}
|
}
|
||||||
|
|
||||||
class SeedWidgetState extends State<SeedWidget> {
|
class SeedWidgetState extends State<SeedWidget> {
|
||||||
|
|
||||||
SeedWidgetState(String language, this.type)
|
SeedWidgetState(String language, this.type)
|
||||||
: controller = TextEditingController(),
|
: controller = TextEditingController(),
|
||||||
focusNode = FocusNode(),
|
focusNode = FocusNode(),
|
||||||
|
@ -46,6 +45,7 @@ class SeedWidgetState extends State<SeedWidget> {
|
||||||
final FocusNode focusNode;
|
final FocusNode focusNode;
|
||||||
final WalletType type;
|
final WalletType type;
|
||||||
List<String> words;
|
List<String> words;
|
||||||
|
bool normalizeSeed = false;
|
||||||
bool _showPlaceholder;
|
bool _showPlaceholder;
|
||||||
|
|
||||||
String get text => controller.text;
|
String get text => controller.text;
|
||||||
|
@ -60,6 +60,7 @@ class SeedWidgetState extends State<SeedWidget> {
|
||||||
void changeSeedLanguage(String language) {
|
void changeSeedLanguage(String language) {
|
||||||
setState(() {
|
setState(() {
|
||||||
words = SeedValidator.getWordList(type: type, language: language);
|
words = SeedValidator.getWordList(type: type, language: language);
|
||||||
|
normalizeSeed = SeedValidator.needsNormalization(language);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -97,6 +98,7 @@ class SeedWidgetState extends State<SeedWidget> {
|
||||||
focusNode: focusNode,
|
focusNode: focusNode,
|
||||||
controller: controller,
|
controller: controller,
|
||||||
words: words,
|
words: words,
|
||||||
|
normalizeSeed: normalizeSeed,
|
||||||
textStyle: TextStyle(
|
textStyle: TextStyle(
|
||||||
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor,
|
color: Theme.of(context).extension<CakeTextTheme>()!.titleColor,
|
||||||
backgroundColor: Colors.transparent,
|
backgroundColor: Colors.transparent,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
|
import 'package:cw_core/utils/text_normalizer.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
|
||||||
extension Compare<T> on Comparable<T> {
|
extension Compare<T> on Comparable<T> {
|
||||||
bool operator <=(T other) => compareTo(other) <= 0;
|
bool operator <=(T other) => compareTo(other) <= 0;
|
||||||
bool operator >=(T other) => compareTo(other) >= 0;
|
bool operator >=(T other) => compareTo(other) >= 0;
|
||||||
|
@ -39,6 +39,7 @@ class ValidatableAnnotatedEditableText extends EditableText {
|
||||||
required this.validStyle,
|
required this.validStyle,
|
||||||
required this.invalidStyle,
|
required this.invalidStyle,
|
||||||
required this.words,
|
required this.words,
|
||||||
|
this.normalizeSeed = false,
|
||||||
TextStyle textStyle = const TextStyle(
|
TextStyle textStyle = const TextStyle(
|
||||||
color: Colors.black,
|
color: Colors.black,
|
||||||
backgroundColor: Colors.transparent,
|
backgroundColor: Colors.transparent,
|
||||||
|
@ -74,6 +75,7 @@ class ValidatableAnnotatedEditableText extends EditableText {
|
||||||
showSelectionHandles: true,
|
showSelectionHandles: true,
|
||||||
showCursor: true);
|
showCursor: true);
|
||||||
|
|
||||||
|
final bool normalizeSeed;
|
||||||
final List<String> words;
|
final List<String> words;
|
||||||
final TextStyle validStyle;
|
final TextStyle validStyle;
|
||||||
final TextStyle invalidStyle;
|
final TextStyle invalidStyle;
|
||||||
|
@ -137,7 +139,8 @@ class ValidatableAnnotatedEditableTextState extends EditableTextState {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool validate(String source) => widget.words.indexOf(source) >= 0;
|
bool validate(String source) =>
|
||||||
|
widget.words.indexOf(widget.normalizeSeed ? normalizeText(source) : source) >= 0;
|
||||||
|
|
||||||
List<TextRange> range(String pattern, String source) {
|
List<TextRange> range(String pattern, String source) {
|
||||||
final result = <TextRange>[];
|
final result = <TextRange>[];
|
||||||
|
|
|
@ -60,8 +60,6 @@ dependencies:
|
||||||
git:
|
git:
|
||||||
url: https://github.com/cake-tech/flutter_file_picker.git
|
url: https://github.com/cake-tech/flutter_file_picker.git
|
||||||
ref: master
|
ref: master
|
||||||
unorm_dart: ^0.2.0
|
|
||||||
# check unorm_dart for usage and for replace
|
|
||||||
permission_handler: ^10.0.0
|
permission_handler: ^10.0.0
|
||||||
device_display_brightness:
|
device_display_brightness:
|
||||||
git:
|
git:
|
||||||
|
@ -100,7 +98,7 @@ dependencies:
|
||||||
# ref: main
|
# ref: main
|
||||||
socks5_proxy: ^1.0.4
|
socks5_proxy: ^1.0.4
|
||||||
flutter_svg: ^2.0.9
|
flutter_svg: ^2.0.9
|
||||||
polyseed: ^0.0.4
|
polyseed: ^0.0.5
|
||||||
nostr_tools: ^1.0.9
|
nostr_tools: ^1.0.9
|
||||||
solana: ^0.30.1
|
solana: ^0.30.1
|
||||||
bitcoin_base:
|
bitcoin_base:
|
||||||
|
|
|
@ -266,7 +266,7 @@
|
||||||
"errorSigningTransaction": "Une erreur s'est produite lors de la signature de la transaction",
|
"errorSigningTransaction": "Une erreur s'est produite lors de la signature de la transaction",
|
||||||
"estimated": "Estimé",
|
"estimated": "Estimé",
|
||||||
"estimated_new_fee": "De nouveaux frais estimés",
|
"estimated_new_fee": "De nouveaux frais estimés",
|
||||||
"etherscan_history": "Historique d'Etherscan",
|
"etherscan_history": "Historique Etherscan",
|
||||||
"event": "Événement",
|
"event": "Événement",
|
||||||
"events": "Événements",
|
"events": "Événements",
|
||||||
"exchange": "Échanger",
|
"exchange": "Échanger",
|
||||||
|
@ -404,7 +404,7 @@
|
||||||
"node_test": "Tester",
|
"node_test": "Tester",
|
||||||
"nodes": "Nœuds",
|
"nodes": "Nœuds",
|
||||||
"nodes_list_reset_to_default_message": "Êtes vous certain de vouloir revenir aux réglages par défaut ?",
|
"nodes_list_reset_to_default_message": "Êtes vous certain de vouloir revenir aux réglages par défaut ?",
|
||||||
"none_of_selected_providers_can_exchange": "Aucun des prestataires sélectionnés ne peut effectuer cet échange",
|
"none_of_selected_providers_can_exchange": "Aucun des fournisseurs sélectionnés ne peut effectuer cet échange",
|
||||||
"noNFTYet": "Pas encore de NFT",
|
"noNFTYet": "Pas encore de NFT",
|
||||||
"normal": "Normal",
|
"normal": "Normal",
|
||||||
"note_optional": "Note (optionnelle)",
|
"note_optional": "Note (optionnelle)",
|
||||||
|
@ -794,7 +794,7 @@
|
||||||
"unconfirmed": "Solde non confirmé",
|
"unconfirmed": "Solde non confirmé",
|
||||||
"understand": "J'ai compris",
|
"understand": "J'ai compris",
|
||||||
"unmatched_currencies": "La devise de votre portefeuille (wallet) actuel ne correspond pas à celle du QR code scanné",
|
"unmatched_currencies": "La devise de votre portefeuille (wallet) actuel ne correspond pas à celle du QR code scanné",
|
||||||
"unspent_change": "Changement",
|
"unspent_change": "Monnaie",
|
||||||
"unspent_coins_details_title": "Détails des pièces (coins) non dépensées",
|
"unspent_coins_details_title": "Détails des pièces (coins) non dépensées",
|
||||||
"unspent_coins_title": "Pièces (coins) non dépensées",
|
"unspent_coins_title": "Pièces (coins) non dépensées",
|
||||||
"unsupported_asset": "Nous ne prenons pas en charge cette action pour cet élément. Veuillez créer ou passer à un portefeuille d'un type d'actif pris en charge.",
|
"unsupported_asset": "Nous ne prenons pas en charge cette action pour cet élément. Veuillez créer ou passer à un portefeuille d'un type d'actif pris en charge.",
|
||||||
|
|
Loading…
Reference in a new issue