mirror of
https://github.com/balena-io/etcher.git
synced 2025-11-09 10:28:32 +00:00
Compare commits
981 Commits
v1.5.63
...
aethernet/
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ec7e0b745e | ||
|
|
d5ba1ea5e1 | ||
|
|
54d3636a22 | ||
|
|
45f6ee667d | ||
|
|
d25eda9a7d | ||
|
|
7420283249 | ||
|
|
453952440f | ||
|
|
2475d576c7 | ||
|
|
8cd6da1260 | ||
|
|
82dd4fc1d1 | ||
|
|
33fe4b2c1a | ||
|
|
b1c1188107 | ||
|
|
63b45aefae | ||
|
|
f79cb0fac5 | ||
|
|
ec42892c7c | ||
|
|
371716fe6a | ||
|
|
d5fb6bec15 | ||
|
|
c5e7bf26d7 | ||
|
|
e3072ac416 | ||
|
|
dfaf06e4cf | ||
|
|
6e24d25576 | ||
|
|
b59b171e43 | ||
|
|
28726584c2 | ||
|
|
00b151311a | ||
|
|
36c813714b | ||
|
|
2ae6764dd9 | ||
|
|
debefc9652 | ||
|
|
b068b847c7 | ||
|
|
6c410c07ce | ||
|
|
c01206c1f3 | ||
|
|
2e85fb45de | ||
|
|
67513e384d | ||
|
|
828dafa493 | ||
|
|
5c5a761222 | ||
|
|
fab10e5fc5 | ||
|
|
797345fc1c | ||
|
|
a0388a43c3 | ||
|
|
f5b0a3023b | ||
|
|
dc1d7bd1fd | ||
|
|
9d674321b6 | ||
|
|
f9c8378d6a | ||
|
|
65da751a52 | ||
|
|
72142be0de | ||
|
|
11cea7c926 | ||
|
|
8d46ee4c22 | ||
|
|
d63c09e2c2 | ||
|
|
c9e9d7d109 | ||
|
|
2412d20eb4 | ||
|
|
7f90d23a12 | ||
|
|
b9a82be29b | ||
|
|
638673ba5e | ||
|
|
898fe4f216 | ||
|
|
7e805662d1 | ||
|
|
baf59c73ac | ||
|
|
38ad9c97c6 | ||
|
|
8fc574f059 | ||
|
|
78b0f00e88 | ||
|
|
0f10f2d483 | ||
|
|
eb5f5bbb9e | ||
|
|
67d26ff790 | ||
|
|
17f2008d88 | ||
|
|
db1bf7e488 | ||
|
|
4b786b8a9f | ||
|
|
fdfa0d3258 | ||
|
|
757aa77d89 | ||
|
|
d70ea06565 | ||
|
|
f2ebd10053 | ||
|
|
cd67b442c9 | ||
|
|
852c83c4fb | ||
|
|
e3b2ee3b83 | ||
|
|
927a026b86 | ||
|
|
c851e1d54f | ||
|
|
e6fdca171f | ||
|
|
c9cfb87733 | ||
|
|
b0b7c53294 | ||
|
|
e8dc6579fe | ||
|
|
f0747abe3f | ||
|
|
32fab87340 | ||
|
|
adcd8e0325 | ||
|
|
7b5808eb2b | ||
|
|
a8f7422cf5 | ||
|
|
5ae9a26361 | ||
|
|
cf1fdb8c5f | ||
|
|
bf7ebde100 | ||
|
|
88c5fa5035 | ||
|
|
887b0dd538 | ||
|
|
364d1db56a | ||
|
|
c431222909 | ||
|
|
55a0f68b97 | ||
|
|
af2563dfc2 | ||
|
|
33f8851083 | ||
|
|
fe1f19b9fa | ||
|
|
871cf3ec0a | ||
|
|
686a5837b6 | ||
|
|
23f2dd5ce5 | ||
|
|
d5d39b395b | ||
|
|
2acad790d3 | ||
|
|
30133306d6 | ||
|
|
04a62f2ad8 | ||
|
|
17858a7d72 | ||
|
|
620307568f | ||
|
|
a349c5d9ac | ||
|
|
0d740ad12d | ||
|
|
85a3f28869 | ||
|
|
dbd5397405 | ||
|
|
85c183b9ef | ||
|
|
0d0af1d1dd | ||
|
|
ad423fc187 | ||
|
|
d8b2a7a236 | ||
|
|
13ec8cbe98 | ||
|
|
a7cae23612 | ||
|
|
86bb093f3d | ||
|
|
997e1eb2f2 | ||
|
|
34cc8b8933 | ||
|
|
f26b074811 | ||
|
|
adaa07b4b0 | ||
|
|
96f4569342 | ||
|
|
be190c6c80 | ||
|
|
809617a82d | ||
|
|
df02732002 | ||
|
|
d35f3c3049 | ||
|
|
8b047e3b14 | ||
|
|
fa41d21e27 | ||
|
|
54e6c5e2c1 | ||
|
|
43fc3dd7eb | ||
|
|
12a1340c8e | ||
|
|
cf8b5790a1 | ||
|
|
659d85a833 | ||
|
|
96c44d31c9 | ||
|
|
ba812b4f64 | ||
|
|
4087258fbd | ||
|
|
955be13129 | ||
|
|
32011c0dea | ||
|
|
b68325c71c | ||
|
|
84bce86fce | ||
|
|
d68eab1dda | ||
|
|
09cf014d14 | ||
|
|
d5bab5805f | ||
|
|
b5ab500a14 | ||
|
|
49253d37c9 | ||
|
|
97cf3b25ad | ||
|
|
99862b95a5 | ||
|
|
8b765d58e5 | ||
|
|
8f566e45b8 | ||
|
|
b8af86e30c | ||
|
|
784f193b6d | ||
|
|
3967adb1b5 | ||
|
|
0667d1110f | ||
|
|
61dd22bdf3 | ||
|
|
24eb8b05b0 | ||
|
|
6991a4950b | ||
|
|
bb169cf674 | ||
|
|
e5d0d2e262 | ||
|
|
72b4d4f4fa | ||
|
|
9b2f2eb4c3 | ||
|
|
ce52ef95a9 | ||
|
|
aa3756ad17 | ||
|
|
73081e726d | ||
|
|
d53dc4149b | ||
|
|
0d5bb4935f | ||
|
|
14aeb0060b | ||
|
|
239726f3ce | ||
|
|
4ed3002716 | ||
|
|
7286fba240 | ||
|
|
895c306fb7 | ||
|
|
f3844d56e2 | ||
|
|
540dc3150a | ||
|
|
035c8dfec3 | ||
|
|
03d6a011db | ||
|
|
27f64650f9 | ||
|
|
ccca009972 | ||
|
|
57a6ceff0e | ||
|
|
30c4baa58b | ||
|
|
a930d77064 | ||
|
|
0d1cfffa5c | ||
|
|
3c7422764c | ||
|
|
55176b9f8f | ||
|
|
156b9314b5 | ||
|
|
76d22280dc | ||
|
|
e4251a3862 | ||
|
|
831339bd2c | ||
|
|
952ea80e15 | ||
|
|
813c497e4b | ||
|
|
1b5b647135 | ||
|
|
7de99003ca | ||
|
|
e09bdd734b | ||
|
|
306e087ec6 | ||
|
|
c6b0178a87 | ||
|
|
4e581ea1ce | ||
|
|
26dc2d19e5 | ||
|
|
b99282acfb | ||
|
|
4e48724d0c | ||
|
|
448ce141d5 | ||
|
|
695f287190 | ||
|
|
4de3271e15 | ||
|
|
77b33b127d | ||
|
|
9cd13ba381 | ||
|
|
9df23c8a3f | ||
|
|
e3618b939e | ||
|
|
6a39f5869a | ||
|
|
fd472efadc | ||
|
|
7e2c2eae63 | ||
|
|
5266571ca4 | ||
|
|
797868c474 | ||
|
|
2c2a5c7c2b | ||
|
|
9e536d5337 | ||
|
|
860e680dd9 | ||
|
|
7bb52aa170 | ||
|
|
1c370f9100 | ||
|
|
ec7c772d0b | ||
|
|
cc0285a77d | ||
|
|
256d3550d1 | ||
|
|
db3a5f3b0a | ||
|
|
0e58edf113 | ||
|
|
db136926a9 | ||
|
|
d84e7211be | ||
|
|
8357cc19d2 | ||
|
|
2752b9fa95 | ||
|
|
0214be4953 | ||
|
|
a4f944e795 | ||
|
|
cd2ebf15fc | ||
|
|
7a7ea374e9 | ||
|
|
330df325f9 | ||
|
|
2fc0882b2e | ||
|
|
4dd779e010 | ||
|
|
3dc54405fe | ||
|
|
3f1aa5bac3 | ||
|
|
8f52fdb900 | ||
|
|
1b93891ed8 | ||
|
|
33adc8ecf8 | ||
|
|
0455f7ea58 | ||
|
|
ea5a167f4f | ||
|
|
8a1c4a4cc8 | ||
|
|
bd8bc81713 | ||
|
|
98a5ddf58a | ||
|
|
6223dbc541 | ||
|
|
7c56621c57 | ||
|
|
a61aa8e2be | ||
|
|
7df4f9615b | ||
|
|
5742452fdf | ||
|
|
fe09f9f862 | ||
|
|
3a4687ea0f | ||
|
|
db6490fb1b | ||
|
|
1642297101 | ||
|
|
5ecd223cfc | ||
|
|
306e40fd7b | ||
|
|
b58249b9c8 | ||
|
|
b23b4b34d0 | ||
|
|
73bc921713 | ||
|
|
f356e4c303 | ||
|
|
9888167f2e | ||
|
|
4561690478 | ||
|
|
576113febf | ||
|
|
cc139bf750 | ||
|
|
ae91958c06 | ||
|
|
33dea6267f | ||
|
|
c9a8bca96f | ||
|
|
8af376e608 | ||
|
|
9ab307df4f | ||
|
|
e8a716f8bb | ||
|
|
a40e64f6cd | ||
|
|
2e53feb38c | ||
|
|
5945ab1f50 | ||
|
|
59d67220d4 | ||
|
|
61610ded84 | ||
|
|
c87a132f40 | ||
|
|
350d4de32b | ||
|
|
f5f9025d6d | ||
|
|
549d744d04 | ||
|
|
6194460dc2 | ||
|
|
8370f638b4 | ||
|
|
ac34c51125 | ||
|
|
b241470fe1 | ||
|
|
179697040c | ||
|
|
335766ed12 | ||
|
|
4c5d052a71 | ||
|
|
86423342a8 | ||
|
|
d8b41552e3 | ||
|
|
11c65fb392 | ||
|
|
bed126506f | ||
|
|
f6aeb52b16 | ||
|
|
a5201942b8 | ||
|
|
c1f7164273 | ||
|
|
6774bf784c | ||
|
|
56ec8b4eac | ||
|
|
35868509af | ||
|
|
3ab6749f49 | ||
|
|
7a012a92bc | ||
|
|
aba01825a0 | ||
|
|
907a3308de | ||
|
|
4366bb372f | ||
|
|
a6f6cd4a19 | ||
|
|
03ee428039 | ||
|
|
8d652d064d | ||
|
|
28adc34239 | ||
|
|
120e9bf42f | ||
|
|
59f54e194b | ||
|
|
c4834e61a7 | ||
|
|
e4d02bc561 | ||
|
|
b9e54e39f7 | ||
|
|
f3c32eac65 | ||
|
|
9a303ab344 | ||
|
|
9c1b55bebc | ||
|
|
30ae4bbd86 | ||
|
|
c6126a980a | ||
|
|
ef90d048ca | ||
|
|
b938132038 | ||
|
|
3cb2e78fe7 | ||
|
|
ea9875ddf0 | ||
|
|
65dacd2ff2 | ||
|
|
a190818827 | ||
|
|
98e33b619b | ||
|
|
685ed715ac | ||
|
|
3cf3c4b398 | ||
|
|
1c2ef4b1d4 | ||
|
|
d22fc91585 | ||
|
|
0a28af5c35 | ||
|
|
0c1e5b88ef | ||
|
|
790201be90 | ||
|
|
d8d379f05e | ||
|
|
b5e9701048 | ||
|
|
292f86d6f5 | ||
|
|
76ca9934c8 | ||
|
|
37b826ee4e | ||
|
|
1e1bd3c508 | ||
|
|
00e8f11913 | ||
|
|
a3c24a26a0 | ||
|
|
4232928ad8 | ||
|
|
b165fb78da | ||
|
|
e9f6c5ead9 | ||
|
|
b2d0c1c9dd | ||
|
|
14d91400a4 | ||
|
|
d0114aece7 | ||
|
|
dff2df4aab | ||
|
|
13159f93ee | ||
|
|
3ece1fd841 | ||
|
|
f46963b6b3 | ||
|
|
b97f4e0031 | ||
|
|
e2d233d74b | ||
|
|
a7ca2e527b | ||
|
|
396a053c0a | ||
|
|
d1a3f1cb88 | ||
|
|
9f96558cdd | ||
|
|
b3bc589d70 | ||
|
|
18d2c28110 | ||
|
|
b272ef296d | ||
|
|
32ca28a3a9 | ||
|
|
4d5e5a3b0b | ||
|
|
8b3f37102d | ||
|
|
4b74253631 | ||
|
|
a81b552b95 | ||
|
|
53f53c0f75 | ||
|
|
fdaf5c69d6 | ||
|
|
061afca5d3 | ||
|
|
ccb08a48f1 | ||
|
|
a8f3d45b12 | ||
|
|
7e333caaf9 | ||
|
|
70229e8684 | ||
|
|
261700389b | ||
|
|
250aed2eb1 | ||
|
|
ed1f008fe2 | ||
|
|
e9ce270dab | ||
|
|
1ee110bc95 | ||
|
|
33dd07c675 | ||
|
|
39ccbbeeda | ||
|
|
55d2400ac7 | ||
|
|
0bdea5c54c | ||
|
|
3be372d49f | ||
|
|
d0c66b2c48 | ||
|
|
65082c4790 | ||
|
|
e87ed9beed | ||
|
|
bc5563d9c2 | ||
|
|
ad83ab5dcc | ||
|
|
0dc1cf9701 | ||
|
|
11489c6538 | ||
|
|
2619d4bc86 | ||
|
|
3730efd350 | ||
|
|
6ece32c546 | ||
|
|
fd9996a3cc | ||
|
|
f06cc89152 | ||
|
|
c1d7ab3fa9 | ||
|
|
b206483c7c | ||
|
|
c3eb8c7b56 | ||
|
|
0849d4f435 | ||
|
|
1dba3ae19b | ||
|
|
f33f2e3771 | ||
|
|
e56aaed973 | ||
|
|
a4659f038e | ||
|
|
cd462818da | ||
|
|
37769efbed | ||
|
|
0f70c4bbce | ||
|
|
48b5e8b9d9 | ||
|
|
1f138f0ecc | ||
|
|
73f67e99ca | ||
|
|
9114da2445 | ||
|
|
554bbcc780 | ||
|
|
4db2289cfd | ||
|
|
c15b56bc23 | ||
|
|
9f52dda6ae | ||
|
|
fadcefb11a | ||
|
|
361c32913c | ||
|
|
5c2042198e | ||
|
|
99df53098c | ||
|
|
aa563c87bd | ||
|
|
1188888956 | ||
|
|
f9d7991dc8 | ||
|
|
53954e81fd | ||
|
|
f82996bfd1 | ||
|
|
b74069eb41 | ||
|
|
e8c7591751 | ||
|
|
3521b61a81 | ||
|
|
93db90c725 | ||
|
|
1dc56aed14 | ||
|
|
d814202424 | ||
|
|
c54856a616 | ||
|
|
fc45df270a | ||
|
|
3cde2faed0 | ||
|
|
b4b8c89aad | ||
|
|
36d05724c0 | ||
|
|
b1e4e681d1 | ||
|
|
3987078c11 | ||
|
|
de0010eb72 | ||
|
|
1f94f44b18 | ||
|
|
fe0b45cae6 | ||
|
|
c32e485f27 | ||
|
|
409b78fc21 | ||
|
|
2f08142f5a | ||
|
|
8c4edaabba | ||
|
|
05497ce85c | ||
|
|
d3df2fe57e | ||
|
|
a0f07082f2 | ||
|
|
b7efa8e1f0 | ||
|
|
3647457bb5 | ||
|
|
2e5a39dcd8 | ||
|
|
edabacfb3a | ||
|
|
f46176fd10 | ||
|
|
2158e20380 | ||
|
|
fa593e33d1 | ||
|
|
50730bd3df | ||
|
|
4e68955981 | ||
|
|
3c0084d012 | ||
|
|
8bd11a01ae | ||
|
|
da3a22d0f6 | ||
|
|
e708212d41 | ||
|
|
a5ceba8435 | ||
|
|
446e8e1253 | ||
|
|
c69b2fa053 | ||
|
|
0597c0e908 | ||
|
|
af2b6bc8ca | ||
|
|
a2c7a542df | ||
|
|
e37ae2743f | ||
|
|
644d955f08 | ||
|
|
e7b4f09021 | ||
|
|
1e0a6a3129 | ||
|
|
ef3b8915d8 | ||
|
|
e58cfd89c5 | ||
|
|
1c52379ee3 | ||
|
|
e2c2b40690 | ||
|
|
bddb89e4a1 | ||
|
|
560ed91e2e | ||
|
|
1f8f7ad7f8 | ||
|
|
a2a0f2ef41 | ||
|
|
40e5fb2287 | ||
|
|
6c49c71b3f | ||
|
|
deb3db0fff | ||
|
|
4872fa3d6e | ||
|
|
640a7409ee | ||
|
|
a7637ad8d4 | ||
|
|
31409c61ca | ||
|
|
e74dc9eb60 | ||
|
|
06997fdf29 | ||
|
|
611e659626 | ||
|
|
e484ae9837 | ||
|
|
7e7ca9524e | ||
|
|
db09b7440d | ||
|
|
e9603505d2 | ||
|
|
0f45f6aca1 | ||
|
|
0a28a7794d | ||
|
|
7c2644ec51 | ||
|
|
ae62812c61 | ||
|
|
68e24df52b | ||
|
|
b9076d01af | ||
|
|
78a5339e3e | ||
|
|
b099770cb1 | ||
|
|
b76366a514 | ||
|
|
eeab351636 | ||
|
|
3e45691d0b | ||
|
|
f9d79521a1 | ||
|
|
14a89b3b8a | ||
|
|
8fa6e618c4 | ||
|
|
093008dee7 | ||
|
|
42838eba09 | ||
|
|
aa72c5d3bb | ||
|
|
bb04098062 | ||
|
|
dda022df37 | ||
|
|
377dfb8e22 | ||
|
|
07befd0bd1 | ||
|
|
2635a410df | ||
|
|
5e5f82c4b5 | ||
|
|
991cbf6b7f | ||
|
|
688d697a99 | ||
|
|
7894a67719 | ||
|
|
7a7ea74984 | ||
|
|
12cd8a39c1 | ||
|
|
2c07538f8f | ||
|
|
c9bfd350ed | ||
|
|
a485d2b4df | ||
|
|
8ed5ff25a5 | ||
|
|
a17a919c37 | ||
|
|
55cafb9268 | ||
|
|
92dfdc6edd | ||
|
|
fff9452509 | ||
|
|
27e560c961 | ||
|
|
34489f0d66 | ||
|
|
b7f8c8368c | ||
|
|
f383f0be6c | ||
|
|
ff08cb44f9 | ||
|
|
6cb914e969 | ||
|
|
a24be20e95 | ||
|
|
08716efbd5 | ||
|
|
24c8ede746 | ||
|
|
548475996c | ||
|
|
7f9add3f1e | ||
|
|
6eab47259e | ||
|
|
46663e3a6f | ||
|
|
9797a2152d | ||
|
|
a7c3431556 | ||
|
|
fef9cd7bec | ||
|
|
b2c4f7a250 | ||
|
|
88ae9fcbd1 | ||
|
|
bc092114c1 | ||
|
|
9f29dc8b76 | ||
|
|
5fbaa3a3db | ||
|
|
0c59168ceb | ||
|
|
540fe90609 | ||
|
|
1f44f3944f | ||
|
|
fbacb8187d | ||
|
|
ac2d4ae8f3 | ||
|
|
a3322e9fd7 | ||
|
|
281f119456 | ||
|
|
140f3452ed | ||
|
|
481be42eb5 | ||
|
|
f2a37079eb | ||
|
|
76fa698995 | ||
|
|
f8e21e2338 | ||
|
|
482c29bc2a | ||
|
|
0bf1ec4958 | ||
|
|
3b105d5a6a | ||
|
|
6d9c81da43 | ||
|
|
c2e23855b3 | ||
|
|
3f59d35fb6 | ||
|
|
44c74f33d9 | ||
|
|
512785e0a9 | ||
|
|
963fc574c3 | ||
|
|
3218fc2c83 | ||
|
|
dc9351713c | ||
|
|
e72049d6e8 | ||
|
|
170126a490 | ||
|
|
7d53d0aadc | ||
|
|
5eac622b8c | ||
|
|
175e41de8d | ||
|
|
61f4762341 | ||
|
|
7c24d1486f | ||
|
|
630f6c691c | ||
|
|
5c5273bd6c | ||
|
|
9bde38df5a | ||
|
|
391e4444d4 | ||
|
|
e5ee0f1961 | ||
|
|
c8737806c0 | ||
|
|
953f572b53 | ||
|
|
05d0f7142d | ||
|
|
ba29d76a00 | ||
|
|
692274691e | ||
|
|
394d3e0bf2 | ||
|
|
784dd03ba7 | ||
|
|
8560189a1e | ||
|
|
098ca9a9a1 | ||
|
|
3ca50a1e2d | ||
|
|
00f193541d | ||
|
|
8ce9eac704 | ||
|
|
76086a8f91 | ||
|
|
9b71772e35 | ||
|
|
72e5631167 | ||
|
|
339c7d56bd | ||
|
|
ba16995070 | ||
|
|
b32c4ee728 | ||
|
|
14e4cbf749 | ||
|
|
406955ca3e | ||
|
|
5a45f8b122 | ||
|
|
129e7e20e8 | ||
|
|
7165a8190b | ||
|
|
07fde0d73f | ||
|
|
a360370c4e | ||
|
|
92cd3d688d | ||
|
|
6554ccf0f8 | ||
|
|
9444f0e1b1 | ||
|
|
d63f5eca0d | ||
|
|
e39fed1f25 | ||
|
|
2dc359b19c | ||
|
|
7aec8a4ae2 | ||
|
|
af9d3ba9f1 | ||
|
|
b0c71b21b3 | ||
|
|
71c7fbd3a2 | ||
|
|
f8cc7c36b4 | ||
|
|
5d95fcb81f | ||
|
|
d481536a3f | ||
|
|
62b42e9254 | ||
|
|
03e3354d50 | ||
|
|
f01f1ddd7a | ||
|
|
2cb58bbbf0 | ||
|
|
2aedea3139 | ||
|
|
59e37182be | ||
|
|
52bdd02a4b | ||
|
|
b1376dfa73 | ||
|
|
37ed18c38b | ||
|
|
b7ad7bd729 | ||
|
|
b43ec4414e | ||
|
|
f05f9d33f9 | ||
|
|
fcc9c5e577 | ||
|
|
3259a8206f | ||
|
|
3fa9611971 | ||
|
|
b749c2d45a | ||
|
|
29e2e9c657 | ||
|
|
f983d88e52 | ||
|
|
1449478c5b | ||
|
|
7e7a669116 | ||
|
|
28f9954661 | ||
|
|
b7e82f7694 | ||
|
|
f0bbd1a1cd | ||
|
|
5f5c66e3f2 | ||
|
|
2fc8b07e29 | ||
|
|
bdb1690a49 | ||
|
|
10b028355f | ||
|
|
a4366556c0 | ||
|
|
9c25cc663a | ||
|
|
ba21da4f0b | ||
|
|
34349f64d5 | ||
|
|
f5c7dc932a | ||
|
|
4880275e7b | ||
|
|
6db0172a50 | ||
|
|
95ff5c98a8 | ||
|
|
e9f9f90137 | ||
|
|
0ebfecc60c | ||
|
|
afa29a0ed1 | ||
|
|
8d707dc815 | ||
|
|
5b509d147f | ||
|
|
bb6d909949 | ||
|
|
8513d63a3e | ||
|
|
d2f3345c7a | ||
|
|
aee3a0a281 | ||
|
|
4752fa6dd2 | ||
|
|
4e08cf3879 | ||
|
|
11bda8e76a | ||
|
|
e33172060f | ||
|
|
0dee6a9888 | ||
|
|
3d855dcbfc | ||
|
|
ed3b7f7971 | ||
|
|
c0a4fb16e2 | ||
|
|
688e7fff9c | ||
|
|
880e56e563 | ||
|
|
bf26d4ec95 | ||
|
|
d5df3de1d7 | ||
|
|
5d005211d4 | ||
|
|
cc08ac9236 | ||
|
|
09a6a340c9 | ||
|
|
2692104ccd | ||
|
|
b1fd539d25 | ||
|
|
33d48fe4f7 | ||
|
|
1ebc8e9362 | ||
|
|
8b5a5241f2 | ||
|
|
959b9ffbac | ||
|
|
c9cbe41f9e | ||
|
|
d62cbdc391 | ||
|
|
31bd8ce7ae | ||
|
|
c25db503e0 | ||
|
|
ac51e6aae3 | ||
|
|
72c9d616fd | ||
|
|
52f80293a2 | ||
|
|
a3a9edd41a | ||
|
|
f9cbff1eec | ||
|
|
b71482284f | ||
|
|
d90e3a816e | ||
|
|
869d875b5f | ||
|
|
fb1a360360 | ||
|
|
943765bd4d | ||
|
|
9280113350 | ||
|
|
627adb1755 | ||
|
|
ad421eae11 | ||
|
|
b0af9d535a | ||
|
|
5ab69dfb7f | ||
|
|
f1214e6ffd | ||
|
|
a09e029216 | ||
|
|
8782c70640 | ||
|
|
7099a36bdb | ||
|
|
7bd8b0c152 | ||
|
|
b1cbf54711 | ||
|
|
84f003d907 | ||
|
|
4257e696da | ||
|
|
c5c0d46ab8 | ||
|
|
b397240664 | ||
|
|
a31e27ee06 | ||
|
|
483d7b6e58 | ||
|
|
bfb6133871 | ||
|
|
917ff89d9d | ||
|
|
ef5762864f | ||
|
|
50586cdb42 | ||
|
|
82a0b8de0c | ||
|
|
6db800d6d2 | ||
|
|
b23bfc2f6e | ||
|
|
929279b35a | ||
|
|
795e4bad5f | ||
|
|
6e20b6034e | ||
|
|
6f34a27bd3 | ||
|
|
240a605977 | ||
|
|
4a6a471345 | ||
|
|
f1be4f50a3 | ||
|
|
8ef32e8081 | ||
|
|
71e02ef833 | ||
|
|
c70d7e475d | ||
|
|
0f31f05e61 | ||
|
|
7971a003cc | ||
|
|
49491b9b8c | ||
|
|
ea11f17954 | ||
|
|
ebd37b9e2f | ||
|
|
5de4fe3d23 | ||
|
|
eb47f1227a | ||
|
|
f84cde7d04 | ||
|
|
4d3eb2887c | ||
|
|
bc631612df | ||
|
|
5d8a211961 | ||
|
|
e62add6893 | ||
|
|
44fc429f64 | ||
|
|
ffe281f25d | ||
|
|
ba39ff433d | ||
|
|
795b8614ad | ||
|
|
745a2f1886 | ||
|
|
9bf58c89d4 | ||
|
|
ee62b9a4c7 | ||
|
|
e6125b893d | ||
|
|
83ed333fa5 | ||
|
|
39ed67d667 | ||
|
|
ac2e973cb0 | ||
|
|
94a0be3b05 | ||
|
|
124e8af649 | ||
|
|
f07ed68d82 | ||
|
|
8f39dbf6b1 | ||
|
|
6dde9ee6c4 | ||
|
|
dbe6fe442d | ||
|
|
1e2ac86ac6 | ||
|
|
83c5ba04cd | ||
|
|
b3f25c176b | ||
|
|
52cf6375eb | ||
|
|
82a3c37c16 | ||
|
|
d63df5a156 | ||
|
|
63ad3739fd | ||
|
|
7eddb16f2f | ||
|
|
7c4f4cacc9 | ||
|
|
dc6ad72b2d | ||
|
|
be729c87af | ||
|
|
4ee83d9da4 | ||
|
|
8b2f06442a | ||
|
|
21181f011f | ||
|
|
b4b099ecb1 | ||
|
|
166b30bb0a | ||
|
|
8eeb81f58e | ||
|
|
0b20a1eeaa | ||
|
|
d8cb8f7815 | ||
|
|
36f79593cf | ||
|
|
1014b25bf5 | ||
|
|
55dcfc1a85 | ||
|
|
9b6a628d51 | ||
|
|
8b5a42073d | ||
|
|
7991d40760 | ||
|
|
4203296414 | ||
|
|
93d319275f | ||
|
|
94d262263c | ||
|
|
ed90f21188 | ||
|
|
80e0231727 | ||
|
|
981197583a | ||
|
|
6f58344e7b | ||
|
|
07be844985 | ||
|
|
45262583e6 | ||
|
|
c113e38531 | ||
|
|
8771f311d7 | ||
|
|
fdec65e9bd | ||
|
|
f8b46dc647 | ||
|
|
847e47b5db | ||
|
|
227bad9e99 | ||
|
|
cb8168de41 | ||
|
|
c200a0c7ac | ||
|
|
81e80572d8 | ||
|
|
2aa6c83714 | ||
|
|
a22ea0b82b | ||
|
|
af64579eb2 | ||
|
|
f2705a611d | ||
|
|
990dcc9d5a | ||
|
|
c09237f0c3 | ||
|
|
571a3533fb | ||
|
|
6fcd9e1595 | ||
|
|
9caa42d257 | ||
|
|
18fdbbaabb | ||
|
|
7381c1c0cb | ||
|
|
2bdcae7209 | ||
|
|
fc694b90b6 | ||
|
|
945cd7ff8e | ||
|
|
3b32ca1e60 | ||
|
|
98611267d5 | ||
|
|
4d53002e5c | ||
|
|
f6b7b0d3d2 | ||
|
|
fbbd7ccf49 | ||
|
|
d41ce65a78 | ||
|
|
c477fd2071 | ||
|
|
7fab8395c8 | ||
|
|
7d72e0c046 | ||
|
|
9ce97be6a4 | ||
|
|
121b69b0c3 | ||
|
|
cb7cc2f276 | ||
|
|
d01849306e | ||
|
|
a4e87982a6 | ||
|
|
e1c3c80c0f | ||
|
|
fd6346ed59 | ||
|
|
2e4f7b5a8c | ||
|
|
d812d4e12e | ||
|
|
10b3f09e7e | ||
|
|
2d3776844c | ||
|
|
914a4574de | ||
|
|
2b3c84f21a | ||
|
|
f4eb1af8d0 | ||
|
|
c01fc332d2 | ||
|
|
b8fdbc3e94 | ||
|
|
3c7c55364b | ||
|
|
bff4355a1a | ||
|
|
9ea57a7df1 | ||
|
|
4c4171e7fb | ||
|
|
77ece044ad | ||
|
|
d633b36b23 | ||
|
|
2eda6601c0 | ||
|
|
6202393637 | ||
|
|
1b76044242 | ||
|
|
28648e27cf | ||
|
|
90921a74ea | ||
|
|
950b764ff1 | ||
|
|
15ba30bf8f | ||
|
|
c96654d50f | ||
|
|
b5f175d220 | ||
|
|
c535543922 | ||
|
|
9913030e6f | ||
|
|
e7f58fc7fa | ||
|
|
746ee50027 | ||
|
|
683c2da224 | ||
|
|
2671c83337 | ||
|
|
bd35c89c04 | ||
|
|
616baecafb | ||
|
|
bfe895c690 | ||
|
|
97aff2eb4c | ||
|
|
1c46ee2988 | ||
|
|
d0d4ee843d | ||
|
|
fd127da342 | ||
|
|
a8728336ca | ||
|
|
c0eb9bd1e9 | ||
|
|
c85896845f | ||
|
|
efe953d8cd | ||
|
|
b5593ef5b2 | ||
|
|
d08d2e00ee | ||
|
|
bc8908cca1 | ||
|
|
9109f0ccd5 | ||
|
|
30c2ef58cd | ||
|
|
23b295c7c1 | ||
|
|
db24ee4d37 | ||
|
|
e737a1edbd | ||
|
|
109d84302c | ||
|
|
e50974a86a | ||
|
|
ef491e1e96 | ||
|
|
f366a68159 | ||
|
|
0377faadd6 | ||
|
|
a5825373e1 | ||
|
|
fadfadd9e9 | ||
|
|
596b316d65 | ||
|
|
c1e24406d9 | ||
|
|
13dfb090b5 | ||
|
|
ddd1ff0101 | ||
|
|
b266a72726 | ||
|
|
255fae3a90 | ||
|
|
b4a60cfee2 | ||
|
|
233a2e6400 | ||
|
|
f31cb49e2a | ||
|
|
47fd12e7a4 | ||
|
|
d5eb679cf0 | ||
|
|
26d0e46367 | ||
|
|
146bfaa9de | ||
|
|
315051c14c | ||
|
|
3a7d770f6d | ||
|
|
2cd60af841 | ||
|
|
e2f5775b07 | ||
|
|
c27be733a9 | ||
|
|
54fda697ce | ||
|
|
04e0b56dd5 | ||
|
|
b71824c5e8 | ||
|
|
65293ea5e4 | ||
|
|
05c2f5bebd | ||
|
|
e8b2255be0 | ||
|
|
2c227d3475 | ||
|
|
958f7b535a | ||
|
|
9e34096139 | ||
|
|
12b5536e22 | ||
|
|
171a5b1793 | ||
|
|
b4fb82066b | ||
|
|
57145436ab | ||
|
|
cba69ca467 | ||
|
|
375fcab788 | ||
|
|
de65c02222 | ||
|
|
444b0beaca | ||
|
|
4c931278b8 | ||
|
|
3bdac794b3 | ||
|
|
67eb593164 | ||
|
|
fe230e7d30 | ||
|
|
2f0ce3ee37 | ||
|
|
992b8a6fb6 | ||
|
|
84e45caa6c | ||
|
|
68d9542816 | ||
|
|
c9c9c50d6c | ||
|
|
9f4e0ce920 | ||
|
|
388852d6b7 | ||
|
|
4e1f071951 | ||
|
|
8e47829905 | ||
|
|
84fe5004a9 | ||
|
|
28b51a9b46 | ||
|
|
07fc7af911 | ||
|
|
330405ae42 | ||
|
|
ffb26ba67f | ||
|
|
fc597abbc9 | ||
|
|
177f10f76d | ||
|
|
a7a7f83e3e | ||
|
|
b6fb44d6a5 | ||
|
|
996c2b55a4 | ||
|
|
21d9d31a27 | ||
|
|
00536cba3a | ||
|
|
641dde81e5 | ||
|
|
8177e98014 | ||
|
|
abfc6be84d | ||
|
|
1d15d582d9 | ||
|
|
5cd3c5fcc0 | ||
|
|
5e568d7dd8 | ||
|
|
c251bce44d | ||
|
|
1408dd48a1 | ||
|
|
a77734797a | ||
|
|
a119ae7efa | ||
|
|
7d284a7e18 | ||
|
|
4d65bd9f1b | ||
|
|
517511e5be | ||
|
|
2ef38fe06d | ||
|
|
b128d36121 | ||
|
|
082025f0b6 | ||
|
|
220b7f6d53 | ||
|
|
062723bf15 | ||
|
|
bcbbb64042 | ||
|
|
59230a0f9e | ||
|
|
18fb9c9de3 | ||
|
|
26e827e4dc | ||
|
|
2f828b1d39 | ||
|
|
5b22fcc2f5 | ||
|
|
4f36b00ec3 | ||
|
|
707c20513e | ||
|
|
cddd068887 | ||
|
|
cf6863b2c6 | ||
|
|
994d311ed3 | ||
|
|
1098f8cb1e | ||
|
|
1be1a2b8f7 | ||
|
|
07a6e40917 | ||
|
|
2c2057b5cb | ||
|
|
caf09e7498 | ||
|
|
9488468b67 | ||
|
|
d071bf8ade | ||
|
|
1626c01ff4 | ||
|
|
3dd6895662 | ||
|
|
0ab967b7a4 | ||
|
|
3b07946065 | ||
|
|
4c0a079d1e | ||
|
|
1878b39e21 | ||
|
|
7050111bf4 | ||
|
|
572f7d826a |
@@ -7,7 +7,6 @@ indent_size = 2
|
|||||||
end_of_line = lf
|
end_of_line = lf
|
||||||
charset = utf-8
|
charset = utf-8
|
||||||
trim_trailing_whitespace = true
|
trim_trailing_whitespace = true
|
||||||
insert_final_newline = true
|
|
||||||
|
|
||||||
[*.md]
|
[*.md]
|
||||||
trim_trailing_whitespace = false
|
trim_trailing_whitespace = false
|
||||||
|
|||||||
10
.gitattributes
vendored
10
.gitattributes
vendored
@@ -1,6 +1,11 @@
|
|||||||
|
# default
|
||||||
|
* text
|
||||||
|
|
||||||
# Javascript files must retain LF line-endings (to keep eslint happy)
|
# Javascript files must retain LF line-endings (to keep eslint happy)
|
||||||
*.js text eol=lf
|
*.js text eol=lf
|
||||||
*.jsx text eol=lf
|
*.jsx text eol=lf
|
||||||
|
*.ts text eol=lf
|
||||||
|
*.tsx text eol=lf
|
||||||
# CSS and SCSS files must retain LF line-endings (to keep ensure-staged-sass.sh happy)
|
# CSS and SCSS files must retain LF line-endings (to keep ensure-staged-sass.sh happy)
|
||||||
*.css text eol=lf
|
*.css text eol=lf
|
||||||
*.scss text eol=lf
|
*.scss text eol=lf
|
||||||
@@ -25,6 +30,7 @@ Makefile text
|
|||||||
*.yml text
|
*.yml text
|
||||||
*.patch text
|
*.patch text
|
||||||
*.txt text
|
*.txt text
|
||||||
|
*.tpl text
|
||||||
CODEOWNERS text
|
CODEOWNERS text
|
||||||
*.plist text
|
*.plist text
|
||||||
|
|
||||||
@@ -56,3 +62,7 @@ CODEOWNERS text
|
|||||||
*.ttf binary diff=hex
|
*.ttf binary diff=hex
|
||||||
xz-without-extension binary diff=hex
|
xz-without-extension binary diff=hex
|
||||||
wmic-output.txt binary diff=hex
|
wmic-output.txt binary diff=hex
|
||||||
|
|
||||||
|
# gitsecret
|
||||||
|
*.secret binary
|
||||||
|
.gitsecret/** binary
|
||||||
|
|||||||
7
.github/ISSUE_TEMPLATE.md
vendored
7
.github/ISSUE_TEMPLATE.md
vendored
@@ -1,6 +1,11 @@
|
|||||||
- **Etcher version:**
|
- **Etcher version:**
|
||||||
- **Operating system and architecture:**
|
- **Operating system and architecture:**
|
||||||
- **Image flashed:**
|
- **Image flashed:**
|
||||||
|
- **What do you think should have happened:** <!-- or a step by step reproduction process -->
|
||||||
|
- **What happened:**
|
||||||
- **Do you see any meaningful error information in the DevTools?**
|
- **Do you see any meaningful error information in the DevTools?**
|
||||||
|
<!-- You can open DevTools by pressing `Ctrl+Shift+I` (`Ctrl+Alt+I` for Etcher before v1.3.x), or `Cmd+Opt+I` if you're on macOS. -->
|
||||||
|
|
||||||
<!-- You can open DevTools by pressing `Ctrl+Shift+I` (`Ctrl+Alt+I` for Etcher before v1.3.x), or `Cmd+Alt+I` if you're on Mac OS. -->
|
<!-- issues with missing information will be labeled as not-enough-info and closed shortly -->
|
||||||
|
<!-- please try to include as many influencing elements as possible are you root, does any other process block the device, etc. -->
|
||||||
|
<!-- if you find a solution in the meantime thank you for sharing the fix and not just closing / abandoning your issue -->
|
||||||
|
|||||||
221
.github/actions/publish/action.yml
vendored
Normal file
221
.github/actions/publish/action.yml
vendored
Normal file
@@ -0,0 +1,221 @@
|
|||||||
|
---
|
||||||
|
name: package and publish GitHub (draft) release
|
||||||
|
# https://github.com/product-os/flowzone/tree/master/.github/actions
|
||||||
|
inputs:
|
||||||
|
json:
|
||||||
|
description: "JSON stringified object containing all the inputs from the calling workflow"
|
||||||
|
required: true
|
||||||
|
secrets:
|
||||||
|
description: "JSON stringified object containing all the secrets from the calling workflow"
|
||||||
|
required: true
|
||||||
|
|
||||||
|
# --- custom environment
|
||||||
|
XCODE_APP_LOADER_EMAIL:
|
||||||
|
type: string
|
||||||
|
default: "accounts+apple@balena.io"
|
||||||
|
NODE_VERSION:
|
||||||
|
type: string
|
||||||
|
default: "14.x"
|
||||||
|
VERBOSE:
|
||||||
|
type: string
|
||||||
|
default: "true"
|
||||||
|
|
||||||
|
runs:
|
||||||
|
# https://docs.github.com/en/actions/creating-actions/creating-a-composite-action
|
||||||
|
using: "composite"
|
||||||
|
steps:
|
||||||
|
- name: Download custom source artifact
|
||||||
|
uses: actions/download-artifact@v3
|
||||||
|
with:
|
||||||
|
name: custom-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ runner.os }}
|
||||||
|
path: ${{ runner.temp }}
|
||||||
|
|
||||||
|
- name: Extract custom source artifact
|
||||||
|
shell: pwsh
|
||||||
|
working-directory: .
|
||||||
|
run: tar -xf ${{ runner.temp }}/custom.tgz
|
||||||
|
|
||||||
|
- name: Setup Node.js
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: ${{ inputs.NODE_VERSION }}
|
||||||
|
cache: npm
|
||||||
|
|
||||||
|
- name: Install yq
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: choco install yq
|
||||||
|
if: runner.os == 'Windows'
|
||||||
|
|
||||||
|
# FIXME: resinci-deploy is not actively maintained
|
||||||
|
# https://github.com/product-os/resinci-deploy
|
||||||
|
- name: Checkout resinci-deploy
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
repository: product-os/resinci-deploy
|
||||||
|
token: ${{ fromJSON(inputs.secrets).FLOWZONE_TOKEN }}
|
||||||
|
path: resinci-deploy
|
||||||
|
|
||||||
|
- name: Build and install resinci-deploy
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: |
|
||||||
|
set -ea
|
||||||
|
|
||||||
|
[[ '${{ inputs.VERBOSE }}' =~ on|On|Yes|yes|true|True ]] && set -x
|
||||||
|
|
||||||
|
runner_os="$(echo "${RUNNER_OS}" | tr '[:upper:]' '[:lower:]')"
|
||||||
|
|
||||||
|
rm -rf ../resinci-deploy && mv resinci-deploy ..
|
||||||
|
|
||||||
|
pushd ../resinci-deploy && npm ci && npm link && popd
|
||||||
|
|
||||||
|
if [[ $runner_os =~ linux|macos ]]; then
|
||||||
|
chmod +x "$(dirname "$(which node)")/resinci-deploy" && which resinci-deploy
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Upload sourcemaps to sentry
|
||||||
|
- name: Generate Sentry DSN
|
||||||
|
id: sentry
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: |
|
||||||
|
set -ea
|
||||||
|
|
||||||
|
[[ '${{ inputs.VERBOSE }}' =~ on|On|Yes|yes|true|True ]] && set -x
|
||||||
|
|
||||||
|
branch="$(echo '${{ github.event.pull_request.head.ref }}' | sed 's/[^[:alnum:]]/-/g')"
|
||||||
|
|
||||||
|
stdout="$(resinci-deploy store sentry \
|
||||||
|
--branch="${branch}" \
|
||||||
|
--name="$(jq -r '.name' package.json)" \
|
||||||
|
--team="$(yq e '.sentry.team' repo.yml)" \
|
||||||
|
--org="$(yq e '.sentry.org' repo.yml)" \
|
||||||
|
--type="$(yq e '.sentry.type' repo.yml)")"
|
||||||
|
|
||||||
|
echo "dsn=$(echo "${stdout}" | tail -n 1)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
env:
|
||||||
|
SENTRY_TOKEN: ${{ fromJSON(inputs.secrets).SENTRY_AUTH_TOKEN }}
|
||||||
|
|
||||||
|
# https://www.electron.build/code-signing.html
|
||||||
|
# https://github.com/Apple-Actions/import-codesign-certs
|
||||||
|
- name: Import Apple code signing certificate
|
||||||
|
if: runner.os == 'macOS'
|
||||||
|
uses: apple-actions/import-codesign-certs@v1
|
||||||
|
with:
|
||||||
|
p12-file-base64: ${{ fromJSON(inputs.secrets).APPLE_SIGNING }}
|
||||||
|
p12-password: ${{ fromJSON(inputs.secrets).APPLE_SIGNING_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Import Windows code signing certificate
|
||||||
|
if: runner.os == 'Windows'
|
||||||
|
shell: powershell
|
||||||
|
run: |
|
||||||
|
Set-Content -Path ${{ runner.temp }}/certificate.base64 -Value $env:WINDOWS_CERTIFICATE
|
||||||
|
certutil -decode ${{ runner.temp }}/certificate.base64 ${{ runner.temp }}/certificate.pfx
|
||||||
|
Remove-Item -path ${{ runner.temp }} -include certificate.base64
|
||||||
|
|
||||||
|
Import-PfxCertificate `
|
||||||
|
-FilePath ${{ runner.temp }}/certificate.pfx `
|
||||||
|
-CertStoreLocation Cert:\CurrentUser\My `
|
||||||
|
-Password (ConvertTo-SecureString -String $env:WINDOWS_CERTIFICATE_PASSWORD -Force -AsPlainText)
|
||||||
|
|
||||||
|
Remove-Item -path ${{ runner.temp }} -include certificate.pfx
|
||||||
|
|
||||||
|
env:
|
||||||
|
WINDOWS_CERTIFICATE: ${{ fromJSON(inputs.secrets).WINDOWS_SIGNING }}
|
||||||
|
WINDOWS_CERTIFICATE_PASSWORD: ${{ fromJSON(inputs.secrets).WINDOWS_SIGNING_PASSWORD }}
|
||||||
|
|
||||||
|
# ... or refactor (e.g.) https://github.com/samuelmeuli/action-electron-builder
|
||||||
|
# https://github.com/product-os/scripts/tree/master/electron
|
||||||
|
# https://github.com/product-os/scripts/tree/master/shared
|
||||||
|
# https://github.com/product-os/balena-concourse/blob/master/pipelines/github-events/template.yml
|
||||||
|
- name: Package release
|
||||||
|
id: package_release
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: |
|
||||||
|
set -ea
|
||||||
|
|
||||||
|
[[ '${{ inputs.VERBOSE }}' =~ on|On|Yes|yes|true|True ]] && set -x
|
||||||
|
|
||||||
|
runner_os="$(echo "${RUNNER_OS}" | tr '[:upper:]' '[:lower:]')"
|
||||||
|
runner_arch="$(echo "${RUNNER_ARCH}" | tr '[:upper:]' '[:lower:]')"
|
||||||
|
|
||||||
|
ELECTRON_BUILDER_ARCHITECTURE="${runner_arch}"
|
||||||
|
APPLICATION_VERSION="$(jq -r '.version' package.json)"
|
||||||
|
ARCHITECTURE_FLAGS="--${ELECTRON_BUILDER_ARCHITECTURE}"
|
||||||
|
|
||||||
|
if [[ $runner_os =~ linux ]]; then
|
||||||
|
ELECTRON_BUILDER_OS='--linux'
|
||||||
|
TARGETS="$(yq e .linux.target[] electron-builder.yml)"
|
||||||
|
|
||||||
|
elif [[ $runner_os =~ darwin|macos|osx ]]; then
|
||||||
|
CSC_KEY_PASSWORD=${{ fromJSON(inputs.secrets).APPLE_SIGNING_PASSWORD }}
|
||||||
|
CSC_KEYCHAIN=signing_temp
|
||||||
|
CSC_LINK=${{ fromJSON(inputs.secrets).APPLE_SIGNING }}
|
||||||
|
ELECTRON_BUILDER_OS='--mac'
|
||||||
|
TARGETS="$(yq e .mac.target[] electron-builder.yml)"
|
||||||
|
|
||||||
|
elif [[ $runner_os =~ windows|win ]]; then
|
||||||
|
ARCHITECTURE_FLAGS="--ia32 ${ARCHITECTURE_FLAGS}"
|
||||||
|
CSC_KEY_PASSWORD=${{ fromJSON(inputs.secrets).WINDOWS_SIGNING_PASSWORD }}
|
||||||
|
CSC_LINK=${{ fromJSON(inputs.secrets).WINDOWS_SIGNING }}
|
||||||
|
ELECTRON_BUILDER_OS='--win'
|
||||||
|
TARGETS="$(yq e .win.target[] electron-builder.yml)"
|
||||||
|
|
||||||
|
else
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
npm link electron-builder
|
||||||
|
|
||||||
|
for target in ${TARGETS}; do
|
||||||
|
electron-builder ${ELECTRON_BUILDER_OS} ${target} ${ARCHITECTURE_FLAGS} \
|
||||||
|
--c.extraMetadata.analytics.sentry.token='${{ steps.sentry.outputs.dsn }}' \
|
||||||
|
--c.extraMetadata.analytics.mixpanel.token='balena-etcher' \
|
||||||
|
--c.extraMetadata.packageType="${target}"
|
||||||
|
|
||||||
|
find dist -type f -maxdepth 1
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "version=${APPLICATION_VERSION}" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
env:
|
||||||
|
# Apple notarization (afterSignHook.js)
|
||||||
|
XCODE_APP_LOADER_EMAIL: ${{ inputs.XCODE_APP_LOADER_EMAIL }}
|
||||||
|
XCODE_APP_LOADER_PASSWORD: ${{ fromJSON(inputs.secrets).XCODE_APP_LOADER_PASSWORD }}
|
||||||
|
# https://github.blog/2020-08-03-github-actions-improvements-for-fork-and-pull-request-workflows/#improvements-for-public-repository-forks
|
||||||
|
# https://docs.github.com/en/actions/managing-workflow-runs/approving-workflow-runs-from-public-forks#about-workflow-runs-from-public-forks
|
||||||
|
CSC_FOR_PULL_REQUEST: true
|
||||||
|
|
||||||
|
# https://www.electron.build/auto-update.html#staged-rollouts
|
||||||
|
- name: Configure staged rollout(s)
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: |
|
||||||
|
set -ea
|
||||||
|
|
||||||
|
[[ '${{ inputs.VERBOSE }}' =~ on|On|Yes|yes|true|True ]] && set -x
|
||||||
|
|
||||||
|
percentage="$(cat < repo.yml | yq e .triggerNotification.stagingPercentage)"
|
||||||
|
|
||||||
|
find dist -type f -maxdepth 1 \
|
||||||
|
-name "latest*.yml" \
|
||||||
|
-exec yq -i e .version=\"${{ steps.package_release.outputs.version }}\" {} \;
|
||||||
|
|
||||||
|
find dist -type f -maxdepth 1 \
|
||||||
|
-name "latest*.yml" \
|
||||||
|
-exec yq -i e .stagingPercentage=\"$percentage\" {} \;
|
||||||
|
|
||||||
|
- name: Upload sourcemap to Sentry
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: |
|
||||||
|
VERSION=${{ steps.package_release.outputs.version }} npm run uploadSourcemap
|
||||||
|
env:
|
||||||
|
SENTRY_AUTH_TOKEN: ${{ fromJSON(inputs.secrets).SENTRY_AUTH_TOKEN }}
|
||||||
|
npm_config_SENTRY_ORG: balenaEtcher
|
||||||
|
npm_config_SENTRY_PROJECT: balenaetcher
|
||||||
|
npm_config_SENTRY_VERSION: ${{ steps.package_release.outputs.version }}
|
||||||
|
|
||||||
|
- name: Upload artifacts
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: gh-release-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}
|
||||||
|
path: dist
|
||||||
|
retention-days: 1
|
||||||
58
.github/actions/test/action.yml
vendored
Normal file
58
.github/actions/test/action.yml
vendored
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
---
|
||||||
|
name: test release
|
||||||
|
# https://github.com/product-os/flowzone/tree/master/.github/actions
|
||||||
|
inputs:
|
||||||
|
json:
|
||||||
|
description: "JSON stringified object containing all the inputs from the calling workflow"
|
||||||
|
required: true
|
||||||
|
secrets:
|
||||||
|
description: "JSON stringified object containing all the secrets from the calling workflow"
|
||||||
|
required: true
|
||||||
|
|
||||||
|
# --- custom environment
|
||||||
|
NODE_VERSION:
|
||||||
|
type: string
|
||||||
|
default: "14.x"
|
||||||
|
VERBOSE:
|
||||||
|
type: string
|
||||||
|
default: "true"
|
||||||
|
|
||||||
|
runs:
|
||||||
|
# https://docs.github.com/en/actions/creating-actions/creating-a-composite-action
|
||||||
|
using: "composite"
|
||||||
|
steps:
|
||||||
|
# https://github.com/actions/setup-node#caching-global-packages-data
|
||||||
|
- name: Setup Node.js
|
||||||
|
uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: ${{ inputs.NODE_VERSION }}
|
||||||
|
cache: npm
|
||||||
|
|
||||||
|
- name: Test release
|
||||||
|
shell: bash --noprofile --norc -eo pipefail -x {0}
|
||||||
|
run: |
|
||||||
|
set -ea
|
||||||
|
|
||||||
|
[[ '${{ inputs.VERBOSE }}' =~ on|On|Yes|yes|true|True ]] && set -x
|
||||||
|
|
||||||
|
runner_os="$(echo "${RUNNER_OS}" | tr '[:upper:]' '[:lower:]')"
|
||||||
|
|
||||||
|
npm run flowzone-preinstall-${runner_os}
|
||||||
|
npm ci
|
||||||
|
npm run build
|
||||||
|
npm run test-${runner_os}
|
||||||
|
|
||||||
|
env:
|
||||||
|
# https://www.electronjs.org/docs/latest/api/environment-variables
|
||||||
|
ELECTRON_NO_ATTACH_CONSOLE: true
|
||||||
|
|
||||||
|
- name: Compress custom source
|
||||||
|
shell: pwsh
|
||||||
|
run: tar -acf ${{ runner.temp }}/custom.tgz .
|
||||||
|
|
||||||
|
- name: Upload custom artifact
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: custom-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ runner.os }}
|
||||||
|
path: ${{ runner.temp }}/custom.tgz
|
||||||
|
retention-days: 1
|
||||||
29
.github/workflows/flowzone.yml
vendored
Normal file
29
.github/workflows/flowzone.yml
vendored
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
name: Flowzone
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
types: [opened, synchronize, closed]
|
||||||
|
branches: [main, master]
|
||||||
|
# allow external contributions to use secrets within trusted code
|
||||||
|
pull_request_target:
|
||||||
|
types: [opened, synchronize, closed]
|
||||||
|
branches: [main, master]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
flowzone:
|
||||||
|
name: Flowzone
|
||||||
|
uses: product-os/flowzone/.github/workflows/flowzone.yml@master
|
||||||
|
# prevent duplicate workflows and only allow one `pull_request` or `pull_request_target` for
|
||||||
|
# internal or external contributions respectively
|
||||||
|
if: |
|
||||||
|
(github.event.pull_request.head.repo.full_name == github.repository && github.event_name == 'pull_request') ||
|
||||||
|
(github.event.pull_request.head.repo.full_name != github.repository && github.event_name == 'pull_request_target')
|
||||||
|
secrets: inherit
|
||||||
|
with:
|
||||||
|
tests_run_on: '["ubuntu-20.04","macos-latest","windows-2019"]'
|
||||||
|
restrict_custom_actions: false
|
||||||
|
github_prerelease: true
|
||||||
|
repo_config: true
|
||||||
|
repo_description: "Flash OS images to SD cards & USB drives, safely and easily."
|
||||||
|
repo_homepage: https://etcher.io/
|
||||||
|
repo_enable_wiki: true
|
||||||
13
.github/workflows/winget.yml
vendored
Normal file
13
.github/workflows/winget.yml
vendored
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
name: Publish to WinGet
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [released]
|
||||||
|
jobs:
|
||||||
|
publish:
|
||||||
|
runs-on: windows-latest # action can only be run on windows
|
||||||
|
steps:
|
||||||
|
- uses: vedantmgoyal2009/winget-releaser@v1
|
||||||
|
with:
|
||||||
|
identifier: Balena.Etcher
|
||||||
|
installers-regex: 'balenaEtcher-Setup.*.exe$'
|
||||||
|
token: ${{ secrets.WINGET_PAT }}
|
||||||
10
.gitignore
vendored
10
.gitignore
vendored
@@ -47,3 +47,13 @@ node_modules
|
|||||||
# OSX files
|
# OSX files
|
||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
|
||||||
|
# VSCode files
|
||||||
|
|
||||||
|
.vscode
|
||||||
|
.gitsecret/keys/random_seed
|
||||||
|
!*.secret
|
||||||
|
secrets/APPLE_SIGNING_PASSWORD.txt
|
||||||
|
secrets/WINDOWS_SIGNING_PASSWORD.txt
|
||||||
|
secrets/XCODE_APP_LOADER_PASSWORD.txt
|
||||||
|
secrets/WINDOWS_SIGNING.pfx
|
||||||
|
|||||||
BIN
.gitsecret/keys/pubring.kbx
Normal file
BIN
.gitsecret/keys/pubring.kbx
Normal file
Binary file not shown.
BIN
.gitsecret/keys/pubring.kbx~
Normal file
BIN
.gitsecret/keys/pubring.kbx~
Normal file
Binary file not shown.
BIN
.gitsecret/keys/trustdb.gpg
Normal file
BIN
.gitsecret/keys/trustdb.gpg
Normal file
Binary file not shown.
5
.gitsecret/paths/mapping.cfg
Normal file
5
.gitsecret/paths/mapping.cfg
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
secrets/APPLE_SIGNING_PASSWORD.txt:5c9cfeb1ea5142b547bc842cc6e0b4a932641ae9811ee47abe2c3953f2a4de5d
|
||||||
|
secrets/WINDOWS_SIGNING_PASSWORD.txt:852e431628494f2559793c39cf09c34e9406dd79bb15b90c9f88194020470568
|
||||||
|
secrets/XCODE_APP_LOADER_PASSWORD.txt:005eb9a3c7035c77232973c9355468fc396b94e62783fb8e6dce16bce95b94a1
|
||||||
|
secrets/WINDOWS_SIGNING.pfx:929f401db38733ffc41572539de7c0d938023af51ed06c205a72a71c1f815714
|
||||||
|
secrets/APPLE_SIGNING.p12:61abf7b4ff2eec76ce889d71bcdd568b99a6a719b4947ac20f03966265b0946a
|
||||||
124
.resinci.json
124
.resinci.json
@@ -1,124 +0,0 @@
|
|||||||
{
|
|
||||||
"electron": {
|
|
||||||
"npm_version": "6.7.0",
|
|
||||||
"dependencies": {
|
|
||||||
"linux": [
|
|
||||||
"libudev-dev",
|
|
||||||
"libusb-1.0-0-dev",
|
|
||||||
"libyaml-dev",
|
|
||||||
"libgtk-3-0",
|
|
||||||
"libatk-bridge2.0-0",
|
|
||||||
"libdbus-1-3",
|
|
||||||
"libc6"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"builder": {
|
|
||||||
"appId": "io.balena.etcher",
|
|
||||||
"copyright": "Copyright 2016-2019 Balena Ltd",
|
|
||||||
"productName": "balenaEtcher",
|
|
||||||
"nodeGypRebuild": true,
|
|
||||||
"files": [
|
|
||||||
"!node_modules/**/*.js.map",
|
|
||||||
"!node_modules/**/*.h",
|
|
||||||
"!node_modules/**/*.hpp",
|
|
||||||
"!node_modules/**/*.cpp",
|
|
||||||
"!node_modules/**/*.md",
|
|
||||||
"!node_modules/**/*.ts",
|
|
||||||
"!node_modules/**/*.coffee",
|
|
||||||
"!node_modules/**/*.scss",
|
|
||||||
"!node_modules/**/*.less",
|
|
||||||
"!node_modules/**/*.hbs",
|
|
||||||
"!node_modules/**/*.mkd",
|
|
||||||
"!node_modules/**/LICENSE",
|
|
||||||
"!node_modules/**/LICENCE",
|
|
||||||
"!node_modules/**/license",
|
|
||||||
"!node_modules/**/License",
|
|
||||||
"!node_modules/**/LICENSE.txt",
|
|
||||||
"!node_modules/**/Makefile",
|
|
||||||
"!node_modules/**/.editorconfig",
|
|
||||||
"!node_modules/**/.babelrc",
|
|
||||||
"!node_modules/**/.prettierrc",
|
|
||||||
"!node_modules/**/.prettierrc-*",
|
|
||||||
"!node_modules/**/.eslintrc.yml",
|
|
||||||
"!node_modules/**/.eslintignore",
|
|
||||||
"!node_modules/**/.publishrc",
|
|
||||||
"assets/icon.png",
|
|
||||||
"build/**/*.node",
|
|
||||||
"lib",
|
|
||||||
"!lib/gui/app",
|
|
||||||
"lib/gui/app/index.html",
|
|
||||||
"generated",
|
|
||||||
"!node_modules/chart.js/dist/docs",
|
|
||||||
"!node_modules/ext2fs/config",
|
|
||||||
"!node_modules/ext2fs/deps",
|
|
||||||
"!node_modules/ext2fs/LICENSE",
|
|
||||||
"!node_modules/ext2fs/src",
|
|
||||||
"!node_modules/winusb-driver-generator/src",
|
|
||||||
"!node_modules/winusb-driver-generator/deps",
|
|
||||||
"!node_modules/winusb-driver-generator/ci",
|
|
||||||
"!node_modules/rendition/__screenshots__",
|
|
||||||
"!node_modules/polished/docs",
|
|
||||||
"!node_modules/mermaid/src",
|
|
||||||
"!node_modules/mermaid/dist",
|
|
||||||
"node_modules/mermaid/dist/mermaid.core.js",
|
|
||||||
"!node_modules/raven-js/src",
|
|
||||||
"!node_modules/raven-js/dist",
|
|
||||||
"node_modules/raven-js/dist/raven.js",
|
|
||||||
"!node_modules/raven-js/plugins",
|
|
||||||
"!node_modules/react-jsonschema-form/dist",
|
|
||||||
"!node_modules/xxhash/deps",
|
|
||||||
"!node_modules/xxhash/src",
|
|
||||||
"!node_modules/unzip-stream/testData*",
|
|
||||||
"!node_modules/usb",
|
|
||||||
"node_modules/usb/usb.js",
|
|
||||||
"node_modules/usb/package.json",
|
|
||||||
"node_modules/usb/build",
|
|
||||||
"node_modules/usb/src/binding",
|
|
||||||
"!node_modules/roboto-fontface/fonts",
|
|
||||||
"node_modules/roboto-fontface/fonts/roboto/Roboto-Thin.woff",
|
|
||||||
"node_modules/roboto-fontface/fonts/roboto/Roboto-Light.woff",
|
|
||||||
"node_modules/roboto-fontface/fonts/roboto/Roboto-Regular.woff",
|
|
||||||
"node_modules/roboto-fontface/fonts/roboto/Roboto-Medium.woff",
|
|
||||||
"node_modules/roboto-fontface/fonts/roboto/Roboto-Bold.woff"
|
|
||||||
],
|
|
||||||
"afterSign": "./afterSignHook.js",
|
|
||||||
"mac": {
|
|
||||||
"asar": false,
|
|
||||||
"category": "public.app-category.developer-tools",
|
|
||||||
"hardenedRuntime": true,
|
|
||||||
"entitlements": "entitlements.mac.plist",
|
|
||||||
"entitlementsInherit": "entitlements.mac.plist"
|
|
||||||
},
|
|
||||||
"dmg": {
|
|
||||||
"iconSize": 110,
|
|
||||||
"contents": [
|
|
||||||
{
|
|
||||||
"x": 140,
|
|
||||||
"y": 245
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"x": 415,
|
|
||||||
"y": 245,
|
|
||||||
"type": "link",
|
|
||||||
"path": "/Applications"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"window": {
|
|
||||||
"width": 544,
|
|
||||||
"height": 407
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"linux": {
|
|
||||||
"category": "Utility",
|
|
||||||
"packageCategory": "utils",
|
|
||||||
"synopsis": "balenaEtcher is a powerful OS image flasher built with web technologies to ensure flashing an SDCard or USB drive is a pleasant and safe experience. It protects you from accidentally writing to your hard-drives, ensures every byte of data was written correctly and much more."
|
|
||||||
},
|
|
||||||
"deb": {
|
|
||||||
"priority": "optional",
|
|
||||||
"depends": [
|
|
||||||
"polkit-1-auth-agent | policykit-1-gnome | polkit-kde-1"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
# sass-lint config generated by make-sass-lint-config v0.1.2
|
|
||||||
|
|
||||||
files:
|
|
||||||
include: lib/gui/scss/**/*.scss
|
|
||||||
options:
|
|
||||||
formatter: stylish
|
|
||||||
merge-default-rules: false
|
|
||||||
rules:
|
|
||||||
no-css-comments: 0
|
|
||||||
no-important: 0
|
|
||||||
no-qualifying-elements: 0
|
|
||||||
placeholder-in-extend: 0
|
|
||||||
property-sort-order: 0
|
|
||||||
quotes:
|
|
||||||
- 1
|
|
||||||
- style: double
|
|
||||||
|
|
||||||
12860
.versionbot/CHANGELOG.yml
Normal file
12860
.versionbot/CHANGELOG.yml
Normal file
File diff suppressed because it is too large
Load Diff
1592
CHANGELOG.md
1592
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
@@ -1,2 +0,0 @@
|
|||||||
* @thundron @zvin @jviotti
|
|
||||||
/scripts @nazrhom
|
|
||||||
4
FAQ.md
4
FAQ.md
@@ -37,10 +37,10 @@ modules=xwayland.so
|
|||||||
Sometimes, things might go wrong, and you end up with a half-flashed drive that is unusable by your operating systems, and common graphical tools might even refuse to get it back to a normal state.
|
Sometimes, things might go wrong, and you end up with a half-flashed drive that is unusable by your operating systems, and common graphical tools might even refuse to get it back to a normal state.
|
||||||
To solve these kinds of problems, we've collected [a list of fail-proof methods](https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md#recovering-broken-drives) to completely erase your drive in major operating systems.
|
To solve these kinds of problems, we've collected [a list of fail-proof methods](https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md#recovering-broken-drives) to completely erase your drive in major operating systems.
|
||||||
|
|
||||||
## I receive ”No polkit authentication agent found” error in GNU/Linux
|
## I receive "No polkit authentication agent found" error in GNU/Linux
|
||||||
|
|
||||||
Etcher requires an available [polkit authentication agent](https://wiki.archlinux.org/index.php/Polkit#Authentication_agents) in your system in order to show a secure password prompt dialog to perform elevation. Make sure you have one installed for the desktop environment of your choice.
|
Etcher requires an available [polkit authentication agent](https://wiki.archlinux.org/index.php/Polkit#Authentication_agents) in your system in order to show a secure password prompt dialog to perform elevation. Make sure you have one installed for the desktop environment of your choice.
|
||||||
|
|
||||||
## May I run Etcher in older macOS versions?
|
## May I run Etcher in older macOS versions?
|
||||||
|
|
||||||
Etcher GUI is based on the [Electron](http://electron.atom.io/) framework, [which only supports macOS 10.9 and newer versions](https://github.com/electron/electron/blob/master/docs/tutorial/support.md#supported-platforms).
|
Etcher GUI is based on the [Electron](http://electron.atom.io/) framework, [which only supports macOS 10.10 and newer versions](https://github.com/electron/electron/blob/master/docs/tutorial/support.md#supported-platforms).
|
||||||
|
|||||||
90
Makefile
90
Makefile
@@ -3,18 +3,12 @@
|
|||||||
# ---------------------------------------------------------------------
|
# ---------------------------------------------------------------------
|
||||||
|
|
||||||
RESIN_SCRIPTS ?= ./scripts/resin
|
RESIN_SCRIPTS ?= ./scripts/resin
|
||||||
export NPM_VERSION ?= 6.7.0
|
export NPM_VERSION ?= 6.14.8
|
||||||
S3_BUCKET = artifacts.ci.balena-cloud.com
|
S3_BUCKET = artifacts.ci.balena-cloud.com
|
||||||
|
|
||||||
# This directory will be completely deleted by the `clean` rule
|
# This directory will be completely deleted by the `clean` rule
|
||||||
BUILD_DIRECTORY ?= dist
|
BUILD_DIRECTORY ?= dist
|
||||||
|
|
||||||
# See http://stackoverflow.com/a/20763842/1641422
|
|
||||||
BUILD_DIRECTORY_PARENT = $(dir $(BUILD_DIRECTORY))
|
|
||||||
ifeq ($(wildcard $(BUILD_DIRECTORY_PARENT).),)
|
|
||||||
$(error $(BUILD_DIRECTORY_PARENT) does not exist)
|
|
||||||
endif
|
|
||||||
|
|
||||||
BUILD_TEMPORARY_DIRECTORY = $(BUILD_DIRECTORY)/.tmp
|
BUILD_TEMPORARY_DIRECTORY = $(BUILD_DIRECTORY)/.tmp
|
||||||
|
|
||||||
$(BUILD_DIRECTORY):
|
$(BUILD_DIRECTORY):
|
||||||
@@ -23,9 +17,7 @@ $(BUILD_DIRECTORY):
|
|||||||
$(BUILD_TEMPORARY_DIRECTORY): | $(BUILD_DIRECTORY)
|
$(BUILD_TEMPORARY_DIRECTORY): | $(BUILD_DIRECTORY)
|
||||||
mkdir $@
|
mkdir $@
|
||||||
|
|
||||||
# See https://stackoverflow.com/a/13468229/1641422
|
|
||||||
SHELL := /bin/bash
|
SHELL := /bin/bash
|
||||||
PATH := $(shell pwd)/node_modules/.bin:$(PATH)
|
|
||||||
|
|
||||||
# ---------------------------------------------------------------------
|
# ---------------------------------------------------------------------
|
||||||
# Operating system and architecture detection
|
# Operating system and architecture detection
|
||||||
@@ -74,6 +66,9 @@ else
|
|||||||
ifeq ($(shell uname -m),x86_64)
|
ifeq ($(shell uname -m),x86_64)
|
||||||
HOST_ARCH = x64
|
HOST_ARCH = x64
|
||||||
endif
|
endif
|
||||||
|
ifeq ($(shell uname -m),arm64)
|
||||||
|
HOST_ARCH = aarch64
|
||||||
|
endif
|
||||||
endif
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
@@ -93,12 +88,10 @@ TARGET_ARCH ?= $(HOST_ARCH)
|
|||||||
# ---------------------------------------------------------------------
|
# ---------------------------------------------------------------------
|
||||||
# Electron
|
# Electron
|
||||||
# ---------------------------------------------------------------------
|
# ---------------------------------------------------------------------
|
||||||
electron-develop: | $(BUILD_TEMPORARY_DIRECTORY)
|
electron-develop:
|
||||||
$(RESIN_SCRIPTS)/electron/install.sh \
|
git submodule update --init && \
|
||||||
-b $(shell pwd) \
|
npm ci && \
|
||||||
-r $(TARGET_ARCH) \
|
npm run webpack
|
||||||
-s $(PLATFORM) \
|
|
||||||
-m $(NPM_VERSION)
|
|
||||||
|
|
||||||
electron-test:
|
electron-test:
|
||||||
$(RESIN_SCRIPTS)/electron/test.sh \
|
$(RESIN_SCRIPTS)/electron/test.sh \
|
||||||
@@ -114,8 +107,7 @@ electron-build: assets/dmg/background.tiff | $(BUILD_TEMPORARY_DIRECTORY)
|
|||||||
-r $(TARGET_ARCH) \
|
-r $(TARGET_ARCH) \
|
||||||
-s $(PLATFORM) \
|
-s $(PLATFORM) \
|
||||||
-v production \
|
-v production \
|
||||||
-n $(BUILD_TEMPORARY_DIRECTORY)/npm \
|
-n $(BUILD_TEMPORARY_DIRECTORY)/npm
|
||||||
-w $(BUILD_TEMPORARY_DIRECTORY)
|
|
||||||
|
|
||||||
# ---------------------------------------------------------------------
|
# ---------------------------------------------------------------------
|
||||||
# Phony targets
|
# Phony targets
|
||||||
@@ -125,70 +117,20 @@ TARGETS = \
|
|||||||
help \
|
help \
|
||||||
info \
|
info \
|
||||||
lint \
|
lint \
|
||||||
lint-js \
|
|
||||||
lint-sass \
|
|
||||||
lint-cpp \
|
|
||||||
lint-html \
|
|
||||||
lint-spell \
|
|
||||||
test-spectron \
|
|
||||||
test-gui \
|
|
||||||
test \
|
test \
|
||||||
sanity-checks \
|
|
||||||
clean \
|
clean \
|
||||||
distclean \
|
distclean \
|
||||||
webpack \
|
|
||||||
electron-develop \
|
electron-develop \
|
||||||
electron-test \
|
electron-test \
|
||||||
electron-build
|
electron-build
|
||||||
|
|
||||||
webpack:
|
|
||||||
./node_modules/.bin/webpack
|
|
||||||
|
|
||||||
.PHONY: $(TARGETS)
|
.PHONY: $(TARGETS)
|
||||||
|
|
||||||
sass:
|
lint:
|
||||||
npm rebuild node-sass
|
npm run lint
|
||||||
node-sass lib/gui/app/scss/main.scss > lib/gui/css/main.css
|
|
||||||
|
|
||||||
lint-ts:
|
test:
|
||||||
resin-lint --typescript lib
|
npm run test
|
||||||
|
|
||||||
lint-js:
|
|
||||||
eslint --ignore-pattern scripts/resin/**/*.js lib tests scripts bin webpack.config.js
|
|
||||||
|
|
||||||
lint-sass:
|
|
||||||
sass-lint lib/gui/scss
|
|
||||||
|
|
||||||
lint-cpp:
|
|
||||||
cpplint --recursive src
|
|
||||||
|
|
||||||
lint-html:
|
|
||||||
node scripts/html-lint.js
|
|
||||||
|
|
||||||
lint-spell:
|
|
||||||
codespell \
|
|
||||||
--dictionary - \
|
|
||||||
--dictionary dictionary.txt \
|
|
||||||
--skip *.svg *.gz,*.bz2,*.xz,*.zip,*.img,*.dmg,*.iso,*.rpi-sdcard,*.wic,.DS_Store,*.dtb,*.dtbo,*.dat,*.elf,*.bin,*.foo,xz-without-extension \
|
|
||||||
lib tests docs Makefile *.md LICENSE
|
|
||||||
|
|
||||||
lint: lint-ts lint-js lint-sass lint-cpp lint-html lint-spell
|
|
||||||
|
|
||||||
MOCHA_OPTIONS=--recursive --reporter spec --require ts-node/register
|
|
||||||
|
|
||||||
# See https://github.com/electron/spectron/issues/127
|
|
||||||
ETCHER_SPECTRON_ENTRYPOINT ?= $(shell node -e 'console.log(require("electron"))')
|
|
||||||
test-spectron:
|
|
||||||
ETCHER_SPECTRON_ENTRYPOINT="$(ETCHER_SPECTRON_ENTRYPOINT)" mocha $(MOCHA_OPTIONS) tests/spectron
|
|
||||||
|
|
||||||
test-gui:
|
|
||||||
electron-mocha $(MOCHA_OPTIONS) --renderer tests/gui
|
|
||||||
|
|
||||||
test-sdk:
|
|
||||||
electron-mocha $(MOCHA_OPTIONS) \
|
|
||||||
tests/shared
|
|
||||||
|
|
||||||
test: test-gui test-sdk test-spectron
|
|
||||||
|
|
||||||
help:
|
help:
|
||||||
@echo "Available targets: $(TARGETS)"
|
@echo "Available targets: $(TARGETS)"
|
||||||
@@ -198,17 +140,11 @@ info:
|
|||||||
@echo "Host arch : $(HOST_ARCH)"
|
@echo "Host arch : $(HOST_ARCH)"
|
||||||
@echo "Target arch : $(TARGET_ARCH)"
|
@echo "Target arch : $(TARGET_ARCH)"
|
||||||
|
|
||||||
sanity-checks:
|
|
||||||
./scripts/ci/ensure-staged-sass.sh
|
|
||||||
./scripts/ci/ensure-npm-dependencies-compatibility.sh
|
|
||||||
./scripts/ci/ensure-all-file-extensions-in-gitattributes.sh
|
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
rm -rf $(BUILD_DIRECTORY)
|
rm -rf $(BUILD_DIRECTORY)
|
||||||
|
|
||||||
distclean: clean
|
distclean: clean
|
||||||
rm -rf node_modules
|
rm -rf node_modules
|
||||||
rm -rf build
|
|
||||||
rm -rf dist
|
rm -rf dist
|
||||||
rm -rf generated
|
rm -rf generated
|
||||||
rm -rf $(BUILD_TEMPORARY_DIRECTORY)
|
rm -rf $(BUILD_TEMPORARY_DIRECTORY)
|
||||||
|
|||||||
159
README.md
159
README.md
@@ -5,16 +5,15 @@
|
|||||||
Etcher is a powerful OS image flasher built with web technologies to ensure
|
Etcher is a powerful OS image flasher built with web technologies to ensure
|
||||||
flashing an SDCard or USB drive is a pleasant and safe experience. It protects
|
flashing an SDCard or USB drive is a pleasant and safe experience. It protects
|
||||||
you from accidentally writing to your hard-drives, ensures every byte of data
|
you from accidentally writing to your hard-drives, ensures every byte of data
|
||||||
was written correctly and much more. It can also flash directly Raspberry Pi devices that support the usbboot protocol
|
was written correctly, and much more. It can also directly flash Raspberry Pi devices that support [USB device boot mode](https://www.raspberrypi.com/documentation/computers/raspberry-pi.html#usb-device-boot-mode).
|
||||||
|
|
||||||
[](https://balena.io/etcher)
|
[](https://balena.io/etcher)
|
||||||
[](https://github.com/balena-io/etcher/blob/master/LICENSE)
|
[](https://github.com/balena-io/etcher/blob/master/LICENSE)
|
||||||
[](https://david-dm.org/balena-io/etcher)
|
|
||||||
[](https://forums.balena.io/c/etcher)
|
[](https://forums.balena.io/c/etcher)
|
||||||
|
|
||||||
***
|
---
|
||||||
|
|
||||||
[**Download**][etcher] | [**Support**][SUPPORT] | [**Documentation**][USER-DOCUMENTATION] | [**Contributing**][CONTRIBUTING] | [**Roadmap**][milestones]
|
[**Download**][etcher] | [**Support**][support] | [**Documentation**][user-documentation] | [**Contributing**][contributing] | [**Roadmap**][milestones]
|
||||||
|
|
||||||
## Supported Operating Systems
|
## Supported Operating Systems
|
||||||
|
|
||||||
@@ -22,7 +21,7 @@ was written correctly and much more. It can also flash directly Raspberry Pi dev
|
|||||||
- macOS 10.10 (Yosemite) and later
|
- macOS 10.10 (Yosemite) and later
|
||||||
- Microsoft Windows 7 and later
|
- Microsoft Windows 7 and later
|
||||||
|
|
||||||
Note that Etcher will run on any platform officially supported by
|
**Note**: Etcher will run on any platform officially supported by
|
||||||
[Electron][electron]. Read more in their
|
[Electron][electron]. Read more in their
|
||||||
[documentation][electron-supported-platforms].
|
[documentation][electron-supported-platforms].
|
||||||
|
|
||||||
@@ -31,66 +30,118 @@ Note that Etcher will run on any platform officially supported by
|
|||||||
Refer to the [downloads page][etcher] for the latest pre-made
|
Refer to the [downloads page][etcher] for the latest pre-made
|
||||||
installers for all supported operating systems.
|
installers for all supported operating systems.
|
||||||
|
|
||||||
|
## Packages
|
||||||
|
|
||||||
|
> [](https://cloudsmith.com) \
|
||||||
|
Package repository hosting is graciously provided by [Cloudsmith](https://cloudsmith.com).
|
||||||
|
Cloudsmith is the only fully hosted, cloud-native, universal package management solution, that
|
||||||
|
enables your organization to create, store and share packages in any format, to any place, with total
|
||||||
|
confidence.
|
||||||
|
|
||||||
#### Debian and Ubuntu based Package Repository (GNU/Linux x86/x64)
|
#### Debian and Ubuntu based Package Repository (GNU/Linux x86/x64)
|
||||||
|
|
||||||
1. Add Etcher debian repository:
|
> Detailed or alternative steps in the [instructions by Cloudsmith](https://cloudsmith.io/~balena/repos/etcher/setup/#formats-deb)
|
||||||
|
|
||||||
```sh
|
1. Add Etcher Debian repository:
|
||||||
echo "deb https://deb.etcher.io stable etcher" | sudo tee /etc/apt/sources.list.d/balena-etcher.list
|
|
||||||
```
|
|
||||||
|
|
||||||
2. Trust Bintray.com's GPG key:
|
```sh
|
||||||
|
curl -1sLf \
|
||||||
|
'https://dl.cloudsmith.io/public/balena/etcher/setup.deb.sh' \
|
||||||
|
| sudo -E bash
|
||||||
|
```
|
||||||
|
|
||||||
```sh
|
2. Update and install:
|
||||||
sudo apt-key adv --keyserver keyserver.ubuntu.com --recv-keys 379CE192D401AB61
|
|
||||||
```
|
|
||||||
|
|
||||||
3. Update and install:
|
```sh
|
||||||
|
sudo apt-get update
|
||||||
```sh
|
sudo apt-get install balena-etcher-electron
|
||||||
sudo apt-get update
|
```
|
||||||
sudo apt-get install balena-etcher-electron
|
|
||||||
```
|
|
||||||
|
|
||||||
##### Uninstall
|
##### Uninstall
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo apt-get remove balena-etcher-electron
|
sudo apt-get remove balena-etcher-electron
|
||||||
sudo rm /etc/apt/sources.list.d/balena-etcher.list
|
rm /etc/apt/sources.list.d/balena-etcher.list
|
||||||
sudo apt-get update
|
apt-get clean
|
||||||
|
rm -rf /var/lib/apt/lists/*
|
||||||
|
apt-get update
|
||||||
```
|
```
|
||||||
#### Redhat (RHEL) and Fedora based Package Repository (GNU/Linux x86/x64)
|
|
||||||
|
#### Redhat (RHEL) and Fedora-based Package Repository (GNU/Linux x86/x64)
|
||||||
|
|
||||||
|
> Detailed or alternative steps in the [instructions by Cloudsmith](https://cloudsmith.io/~balena/repos/etcher/setup/#formats-rpm)
|
||||||
|
|
||||||
|
|
||||||
|
##### DNF
|
||||||
|
|
||||||
1. Add Etcher rpm repository:
|
1. Add Etcher rpm repository:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo wget https://balena.io/etcher/static/etcher-rpm.repo -O /etc/yum.repos.d/etcher-rpm.repo
|
curl -1sLf \
|
||||||
```
|
'https://dl.cloudsmith.io/public/balena/etcher/setup.rpm.sh' \
|
||||||
|
| sudo -E bash
|
||||||
|
```
|
||||||
|
|
||||||
2. Update and install:
|
2. Update and install:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo yum install -y balena-etcher-electron
|
sudo dnf install -y balena-etcher-electron
|
||||||
```
|
```
|
||||||
or
|
|
||||||
```sh
|
###### Uninstall
|
||||||
sudo dnf install -y balena-etcher-electron
|
|
||||||
```
|
```sh
|
||||||
|
rm /etc/yum.repos.d/balena-etcher.repo
|
||||||
|
rm /etc/yum.repos.d/balena-etcher-source.repo
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Yum
|
||||||
|
|
||||||
|
1. Add Etcher rpm repository:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
curl -1sLf \
|
||||||
|
'https://dl.cloudsmith.io/public/balena/etcher/setup.rpm.sh' \
|
||||||
|
| sudo -E bash
|
||||||
|
```
|
||||||
|
|
||||||
|
2. Update and install:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
sudo yum install -y balena-etcher-electron
|
||||||
|
```
|
||||||
|
|
||||||
|
###### Uninstall
|
||||||
|
|
||||||
|
```sh
|
||||||
|
sudo yum remove -y balena-etcher-electron
|
||||||
|
rm /etc/yum.repos.d/balena-etcher.repo
|
||||||
|
rm /etc/yum.repos.d/balena-etcher-source.repo
|
||||||
|
```
|
||||||
|
|
||||||
|
#### OpenSUSE LEAP & Tumbleweed install (zypper)
|
||||||
|
|
||||||
|
1. Add the repo
|
||||||
|
|
||||||
|
```sh
|
||||||
|
curl -1sLf \
|
||||||
|
'https://dl.cloudsmith.io/public/balena/etcher/setup.rpm.sh' \
|
||||||
|
| sudo -E bash
|
||||||
|
```
|
||||||
|
2. Update and install
|
||||||
|
|
||||||
|
```sh
|
||||||
|
sudo zypper up
|
||||||
|
sudo zypper install balena-etcher-electron
|
||||||
|
```
|
||||||
|
|
||||||
##### Uninstall
|
##### Uninstall
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
sudo yum remove -y balena-etcher-electron
|
sudo zypper rm balena-etcher-electron
|
||||||
sudo rm /etc/yum.repos.d/etcher-rpm.repo
|
# remove the repo
|
||||||
sudo yum clean all
|
sudo zypper rr balena-etcher
|
||||||
sudo yum makecache fast
|
sudo zypper rr balena-etcher-source
|
||||||
```
|
|
||||||
or
|
|
||||||
```sh
|
|
||||||
sudo dnf remove -y balena-etcher-electron
|
|
||||||
sudo rm /etc/yum.repos.d/etcher-rpm.repo
|
|
||||||
sudo dnf clean all
|
|
||||||
sudo dnf makecache
|
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Solus (GNU/Linux x64)
|
#### Solus (GNU/Linux x64)
|
||||||
@@ -105,20 +156,18 @@ sudo eopkg it etcher
|
|||||||
sudo eopkg rm etcher
|
sudo eopkg rm etcher
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Brew Cask (macOS)
|
#### Arch/Manjaro Linux (GNU/Linux x64)
|
||||||
|
|
||||||
Note that the Etcher Cask has to be updated manually to point to new versions,
|
Etcher is offered through the Arch User Repository and can be installed on both Manjaro and Arch systems. You can compile it from the source code in this repository using [`balena-etcher`](https://aur.archlinux.org/packages/balena-etcher/). The following example uses a common AUR helper to install the latest release:
|
||||||
so it might not refer to the latest version immediately after an Etcher
|
|
||||||
release.
|
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
brew cask install balenaetcher
|
yay -S balena-etcher
|
||||||
```
|
```
|
||||||
|
|
||||||
##### Uninstall
|
##### Uninstall
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
brew cask uninstall balenaetcher
|
yay -R balena-etcher
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Chocolatey (Windows)
|
#### Chocolatey (Windows)
|
||||||
@@ -138,20 +187,22 @@ choco uninstall etcher
|
|||||||
|
|
||||||
## Support
|
## Support
|
||||||
|
|
||||||
If you're having any problem, please [raise an issue][newissue] on GitHub and
|
If you're having any problem, please [raise an issue][newissue] on GitHub, and
|
||||||
the balena.io team will be happy to help.
|
the balena.io team will be happy to help.
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
Etcher is free software, and may be redistributed under the terms specified in
|
Etcher is free software and may be redistributed under the terms specified in
|
||||||
the [license].
|
the [license].
|
||||||
|
|
||||||
[etcher]: https://balena.io/etcher
|
[etcher]: https://balena.io/etcher
|
||||||
[electron]: https://electronjs.org/
|
[electron]: https://electronjs.org/
|
||||||
[electron-supported-platforms]: https://electronjs.org/docs/tutorial/support#supported-platforms
|
[electron-supported-platforms]: https://electronjs.org/docs/tutorial/support#supported-platforms
|
||||||
[SUPPORT]: https://github.com/balena-io/etcher/blob/master/SUPPORT.md
|
[support]: https://github.com/balena-io/etcher/blob/master/SUPPORT.md
|
||||||
[CONTRIBUTING]: https://github.com/balena-io/etcher/blob/master/docs/CONTRIBUTING.md
|
[contributing]: https://github.com/balena-io/etcher/blob/master/docs/CONTRIBUTING.md
|
||||||
[USER-DOCUMENTATION]: https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md
|
[user-documentation]: https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md
|
||||||
[milestones]: https://github.com/balena-io/etcher/milestones
|
[milestones]: https://github.com/balena-io/etcher/milestones
|
||||||
[newissue]: https://github.com/balena-io/etcher/issues/new
|
[newissue]: https://github.com/balena-io/etcher/issues/new
|
||||||
[license]: https://github.com/balena-io/etcher/blob/master/LICENSE
|
[license]: https://github.com/balena-io/etcher/blob/master/LICENSE
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
17
SUPPORT.md
17
SUPPORT.md
@@ -1,9 +1,16 @@
|
|||||||
Getting help with Etcher
|
Getting help with BalenaEtcher
|
||||||
========================
|
===============================
|
||||||
|
|
||||||
There are various ways to get support for Etcher if you experience an issue or
|
There are various ways to get support for Etcher if you experience an issue or
|
||||||
have an idea you'd like to share with us.
|
have an idea you'd like to share with us.
|
||||||
|
|
||||||
|
Documentation
|
||||||
|
------
|
||||||
|
|
||||||
|
We have answers to a variety of frequently asked questions in the [user
|
||||||
|
documentation][documentation] and also in the [FAQs][faq] on the Etcher website.
|
||||||
|
|
||||||
|
|
||||||
Forums
|
Forums
|
||||||
------
|
------
|
||||||
|
|
||||||
@@ -15,7 +22,7 @@ a look at the existing threads before opening a new one!
|
|||||||
Make sure to mention the following information to help us provide better
|
Make sure to mention the following information to help us provide better
|
||||||
support:
|
support:
|
||||||
|
|
||||||
- The Etcher version you're running.
|
- The BalenaEtcher version you're running.
|
||||||
|
|
||||||
- The operating system you're running Etcher in.
|
- The operating system you're running Etcher in.
|
||||||
|
|
||||||
@@ -25,10 +32,12 @@ support:
|
|||||||
GitHub
|
GitHub
|
||||||
------
|
------
|
||||||
|
|
||||||
If you encounter an issue or have a suggestion, head on over to Etcher's [issue
|
If you encounter an issue or have a suggestion, head on over to BalenaEtcher's [issue
|
||||||
tracker][issues] and if there isn't a ticket covering it, [create
|
tracker][issues] and if there isn't a ticket covering it, [create
|
||||||
one][new-issue].
|
one][new-issue].
|
||||||
|
|
||||||
[discourse]: https://forums.balena.io/c/etcher
|
[discourse]: https://forums.balena.io/c/etcher
|
||||||
[issues]: https://github.com/balena-io/etcher/issues
|
[issues]: https://github.com/balena-io/etcher/issues
|
||||||
[new-issue]: https://github.com/balena-io/etcher/issues/new
|
[new-issue]: https://github.com/balena-io/etcher/issues/new
|
||||||
|
[documentation]: https://github.com/balena-io/etcher/blob/master/docs/USER-DOCUMENTATION.md
|
||||||
|
[faq]: https://etcher.io
|
||||||
|
|||||||
11
after-install.tpl
Normal file
11
after-install.tpl
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Link to the binary
|
||||||
|
# Must hardcode balenaEtcher directory; no variable available
|
||||||
|
ln -sf '/opt/balenaEtcher/${executable}' '/usr/bin/${executable}'
|
||||||
|
|
||||||
|
# SUID chrome-sandbox for Electron 5+
|
||||||
|
chmod 4755 '/opt/balenaEtcher/chrome-sandbox' || true
|
||||||
|
|
||||||
|
update-mime-database /usr/share/mime || true
|
||||||
|
update-desktop-database /usr/share/applications || true
|
||||||
31
afterPack.js
Normal file
31
afterPack.js
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
'use strict'
|
||||||
|
|
||||||
|
const cp = require('child_process')
|
||||||
|
const fs = require('fs')
|
||||||
|
const outdent = require('outdent')
|
||||||
|
const path = require('path')
|
||||||
|
|
||||||
|
exports.default = function(context) {
|
||||||
|
if (context.packager.platform.name !== 'linux') {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const scriptPath = path.join(context.appOutDir, context.packager.executableName)
|
||||||
|
const binPath = scriptPath + '.bin'
|
||||||
|
cp.execFileSync('mv', [scriptPath, binPath])
|
||||||
|
fs.writeFileSync(
|
||||||
|
scriptPath,
|
||||||
|
outdent({trimTrailingNewline: false})`
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Resolve symlinks. Warning, readlink -f doesn't work on MacOS/BSD
|
||||||
|
script_dir="$(dirname "$(readlink -f "\${BASH_SOURCE[0]}")")"
|
||||||
|
|
||||||
|
if [[ $EUID -ne 0 ]] || [[ $ELECTRON_RUN_AS_NODE ]]; then
|
||||||
|
"\${script_dir}"/${context.packager.executableName}.bin "$@"
|
||||||
|
else
|
||||||
|
"\${script_dir}"/${context.packager.executableName}.bin "$@" --no-sandbox
|
||||||
|
fi
|
||||||
|
`
|
||||||
|
)
|
||||||
|
cp.execFileSync('chmod', ['+x', scriptPath])
|
||||||
|
}
|
||||||
@@ -1,21 +1,24 @@
|
|||||||
'use strict'
|
'use strict'
|
||||||
|
|
||||||
const { notarize } = require('electron-notarize')
|
const { notarize } = require('electron-notarize')
|
||||||
|
const { ELECTRON_SKIP_NOTARIZATION } = process.env
|
||||||
|
|
||||||
async function main(context) {
|
async function main(context) {
|
||||||
const { electronPlatformName, appOutDir } = context
|
const { electronPlatformName, appOutDir } = context
|
||||||
if (electronPlatformName !== 'darwin') {
|
if (electronPlatformName !== 'darwin' || ELECTRON_SKIP_NOTARIZATION === 'true') {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
const appName = context.packager.appInfo.productFilename
|
const appName = context.packager.appInfo.productFilename
|
||||||
const appleId = 'accounts+apple@balena.io'
|
const appleId = process.env.XCODE_APP_LOADER_EMAIL || 'accounts+apple@balena.io'
|
||||||
|
const appleIdPassword = process.env.XCODE_APP_LOADER_PASSWORD
|
||||||
|
|
||||||
|
// https://github.com/electron/notarize/blob/main/README.md
|
||||||
await notarize({
|
await notarize({
|
||||||
appBundleId: 'io.balena.etcher',
|
appBundleId: 'io.balena.etcher',
|
||||||
appPath: `${appOutDir}/${appName}.app`,
|
appPath: `${appOutDir}/${appName}.app`,
|
||||||
appleId,
|
appleId,
|
||||||
appleIdPassword: `@keychain:Application Loader: ${appleId}`
|
appleIdPassword
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
BIN
assets/icon.icns
BIN
assets/icon.icns
Binary file not shown.
35
binding.gyp
35
binding.gyp
@@ -1,35 +0,0 @@
|
|||||||
{
|
|
||||||
"targets": [
|
|
||||||
{
|
|
||||||
"target_name": "elevator",
|
|
||||||
"include_dirs" : [
|
|
||||||
"src",
|
|
||||||
"<!(node -e \"require('nan')\")"
|
|
||||||
],
|
|
||||||
'conditions': [
|
|
||||||
|
|
||||||
[ 'OS=="win"', {
|
|
||||||
"sources": [
|
|
||||||
"src/utils/v8utils.cpp",
|
|
||||||
"src/os/win32/elevate.cpp",
|
|
||||||
"src/elevator_init.cpp",
|
|
||||||
],
|
|
||||||
"libraries": [
|
|
||||||
"-lShell32.lib",
|
|
||||||
],
|
|
||||||
} ],
|
|
||||||
|
|
||||||
[ 'OS=="mac"', {
|
|
||||||
"xcode_settings": {
|
|
||||||
"OTHER_CPLUSPLUSFLAGS": [
|
|
||||||
"-stdlib=libc++"
|
|
||||||
],
|
|
||||||
"OTHER_LDFLAGS": [
|
|
||||||
"-stdlib=libc++"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
} ]
|
|
||||||
],
|
|
||||||
}
|
|
||||||
],
|
|
||||||
}
|
|
||||||
@@ -12,12 +12,9 @@ technologies used in Etcher that you should become familiar with:
|
|||||||
|
|
||||||
- [Electron][electron]
|
- [Electron][electron]
|
||||||
- [NodeJS][nodejs]
|
- [NodeJS][nodejs]
|
||||||
- [AngularJS][angularjs]
|
|
||||||
- [Redux][redux]
|
- [Redux][redux]
|
||||||
- [ImmutableJS][immutablejs]
|
- [ImmutableJS][immutablejs]
|
||||||
- [Bootstrap][bootstrap]
|
|
||||||
- [Sass][sass]
|
- [Sass][sass]
|
||||||
- [Flexbox Grid][flexbox-grid]
|
|
||||||
- [Mocha][mocha]
|
- [Mocha][mocha]
|
||||||
- [JSDoc][jsdoc]
|
- [JSDoc][jsdoc]
|
||||||
|
|
||||||
@@ -66,11 +63,8 @@ be documented instead!
|
|||||||
[gui-dir]: https://github.com/balena-io/etcher/tree/master/lib/gui
|
[gui-dir]: https://github.com/balena-io/etcher/tree/master/lib/gui
|
||||||
[electron]: http://electron.atom.io
|
[electron]: http://electron.atom.io
|
||||||
[nodejs]: https://nodejs.org
|
[nodejs]: https://nodejs.org
|
||||||
[angularjs]: https://angularjs.org
|
|
||||||
[redux]: http://redux.js.org
|
[redux]: http://redux.js.org
|
||||||
[immutablejs]: http://facebook.github.io/immutable-js/
|
[immutablejs]: http://facebook.github.io/immutable-js/
|
||||||
[bootstrap]: http://getbootstrap.com
|
|
||||||
[sass]: http://sass-lang.com
|
[sass]: http://sass-lang.com
|
||||||
[flexbox-grid]: http://flexboxgrid.com
|
|
||||||
[mocha]: http://mochajs.org
|
[mocha]: http://mochajs.org
|
||||||
[jsdoc]: http://usejsdoc.org
|
[jsdoc]: http://usejsdoc.org
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ make electron-develop
|
|||||||
|
|
||||||
```sh
|
```sh
|
||||||
# Build the GUI
|
# Build the GUI
|
||||||
make webpack
|
npm run webpack
|
||||||
# Start Electron
|
# Start Electron
|
||||||
npm start
|
npm start
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -159,6 +159,18 @@ pre-installed in all modern Windows versions.
|
|||||||
|
|
||||||
- Run `clean`. This command will completely clean your drive by erasing any
|
- Run `clean`. This command will completely clean your drive by erasing any
|
||||||
existent filesystem.
|
existent filesystem.
|
||||||
|
|
||||||
|
- Run `create partition primary`. This command will create a new partition.
|
||||||
|
|
||||||
|
- Run `active`. This command will active the partition.
|
||||||
|
|
||||||
|
- Run `list partition`. This command will show available partition.
|
||||||
|
|
||||||
|
- Run `select partition N`, where `N` corresponds to the id of the newly available partition.
|
||||||
|
|
||||||
|
- Run `format override quick`. This command will format the partition. You can choose a specific formatting by adding `FS=xx` where `xx` could be `NTFS or FAT or FAT32` after `format`. Example : `format FS=NTFS override quick`
|
||||||
|
|
||||||
|
- Run `exit` to quit diskpart.
|
||||||
|
|
||||||
### OS X
|
### OS X
|
||||||
|
|
||||||
@@ -166,7 +178,7 @@ Run the following command in `Terminal.app`, replacing `N` by the corresponding
|
|||||||
disk number, which you can find by running `diskutil list`:
|
disk number, which you can find by running `diskutil list`:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
diskutil eraseDisk free UNTITLED /dev/diskN
|
diskutil eraseDisk FAT32 UNTITLED MBRFormat /dev/diskN
|
||||||
```
|
```
|
||||||
|
|
||||||
### GNU/Linux
|
### GNU/Linux
|
||||||
|
|||||||
@@ -1,23 +1,23 @@
|
|||||||
|
# https://www.electron.build/configuration/configuration
|
||||||
appId: io.balena.etcher
|
appId: io.balena.etcher
|
||||||
copyright: Copyright 2016-2019 Balena Ltd
|
copyright: Copyright 2016-2023 Balena Ltd
|
||||||
productName: balenaEtcher
|
productName: balenaEtcher
|
||||||
npmRebuild: true
|
afterPack: ./afterPack.js
|
||||||
nodeGypRebuild: true
|
afterSign: ./afterSignHook.js
|
||||||
publish: null
|
asar: false
|
||||||
files:
|
files:
|
||||||
- lib
|
|
||||||
- lib/gui/app/index.html
|
|
||||||
- generated
|
- generated
|
||||||
- build/**/*.node
|
- lib/shared/catalina-sudo/sudo-askpass.osascript-zh.js
|
||||||
- assets/icon.png
|
- lib/shared/catalina-sudo/sudo-askpass.osascript-en.js
|
||||||
- node_modules/**/*
|
|
||||||
mac:
|
mac:
|
||||||
asar: false
|
|
||||||
icon: assets/icon.icns
|
icon: assets/icon.icns
|
||||||
category: public.app-category.developer-tools
|
category: public.app-category.developer-tools
|
||||||
hardenedRuntime: true
|
hardenedRuntime: true
|
||||||
entitlements: "entitlements.mac.plist"
|
entitlements: "entitlements.mac.plist"
|
||||||
entitlementsInherit: "entitlements.mac.plist"
|
entitlementsInherit: "entitlements.mac.plist"
|
||||||
|
artifactName: "${productName}-${version}.${ext}"
|
||||||
|
target:
|
||||||
|
- dmg
|
||||||
dmg:
|
dmg:
|
||||||
background: assets/dmg/background.tiff
|
background: assets/dmg/background.tiff
|
||||||
icon: assets/icon.icns
|
icon: assets/icon.icns
|
||||||
@@ -34,6 +34,10 @@ dmg:
|
|||||||
height: 405
|
height: 405
|
||||||
win:
|
win:
|
||||||
icon: assets/icon.ico
|
icon: assets/icon.ico
|
||||||
|
target:
|
||||||
|
- zip
|
||||||
|
- nsis
|
||||||
|
- portable
|
||||||
nsis:
|
nsis:
|
||||||
oneClick: true
|
oneClick: true
|
||||||
runAfterFinish: true
|
runAfterFinish: true
|
||||||
@@ -46,17 +50,23 @@ portable:
|
|||||||
artifactName: "${productName}-Portable-${version}.${ext}"
|
artifactName: "${productName}-Portable-${version}.${ext}"
|
||||||
requestExecutionLevel: user
|
requestExecutionLevel: user
|
||||||
linux:
|
linux:
|
||||||
|
icon: assets/iconset
|
||||||
|
target:
|
||||||
|
- AppImage
|
||||||
|
- rpm
|
||||||
|
- deb
|
||||||
category: Utility
|
category: Utility
|
||||||
packageCategory: utils
|
packageCategory: utils
|
||||||
executableName: balena-etcher-electron
|
executableName: balena-etcher
|
||||||
synopsis: balenaEtcher is a powerful OS image flasher built with web technologies to ensure flashing an SDCard or USB drive is a pleasant and safe experience. It protects you from accidentally writing to your hard-drives, ensures every byte of data was written correctly and much more.
|
synopsis: balenaEtcher is a powerful OS image flasher built with web technologies to ensure flashing an SDCard or USB drive is a pleasant and safe experience. It protects you from accidentally writing to your hard-drives, ensures every byte of data was written correctly and much more.
|
||||||
icon: assets/iconset
|
appImage:
|
||||||
|
artifactName: ${productName}-${version}-${env.ELECTRON_BUILDER_ARCHITECTURE}.${ext}
|
||||||
deb:
|
deb:
|
||||||
priority: optional
|
priority: optional
|
||||||
|
compression: bzip2
|
||||||
depends:
|
depends:
|
||||||
- gconf2
|
|
||||||
- gconf-service
|
- gconf-service
|
||||||
- libappindicator1
|
- gconf2
|
||||||
- libasound2
|
- libasound2
|
||||||
- libatk1.0-0
|
- libatk1.0-0
|
||||||
- libc6
|
- libc6
|
||||||
@@ -66,6 +76,7 @@ deb:
|
|||||||
- libexpat1
|
- libexpat1
|
||||||
- libfontconfig1
|
- libfontconfig1
|
||||||
- libfreetype6
|
- libfreetype6
|
||||||
|
- libgbm1
|
||||||
- libgcc1
|
- libgcc1
|
||||||
- libgconf-2-4
|
- libgconf-2-4
|
||||||
- libgdk-pixbuf2.0-0
|
- libgdk-pixbuf2.0-0
|
||||||
@@ -75,7 +86,7 @@ deb:
|
|||||||
- libnotify4
|
- libnotify4
|
||||||
- libnspr4
|
- libnspr4
|
||||||
- libnss3
|
- libnss3
|
||||||
- libpango1.0-0
|
- libpango1.0-0 | libpango-1.0-0
|
||||||
- libstdc++6
|
- libstdc++6
|
||||||
- libx11-6
|
- libx11-6
|
||||||
- libxcomposite1
|
- libxcomposite1
|
||||||
@@ -89,7 +100,11 @@ deb:
|
|||||||
- libxss1
|
- libxss1
|
||||||
- libxtst6
|
- libxtst6
|
||||||
- polkit-1-auth-agent | policykit-1-gnome | polkit-kde-1
|
- polkit-1-auth-agent | policykit-1-gnome | polkit-kde-1
|
||||||
|
afterInstall: "./after-install.tpl"
|
||||||
rpm:
|
rpm:
|
||||||
depends:
|
depends:
|
||||||
- lsb
|
- util-linux
|
||||||
- libXScrnSaver
|
protocols:
|
||||||
|
name: etcher
|
||||||
|
schemes:
|
||||||
|
- etcher
|
||||||
|
|||||||
@@ -1,508 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/* eslint-disable no-var */
|
|
||||||
|
|
||||||
var angular = require('angular')
|
|
||||||
|
|
||||||
/* eslint-enable no-var */
|
|
||||||
|
|
||||||
const electron = require('electron')
|
|
||||||
const sdk = require('etcher-sdk')
|
|
||||||
const _ = require('lodash')
|
|
||||||
const uuidV4 = require('uuid/v4')
|
|
||||||
|
|
||||||
const EXIT_CODES = require('../../shared/exit-codes')
|
|
||||||
const messages = require('../../shared/messages')
|
|
||||||
const store = require('./models/store')
|
|
||||||
const packageJSON = require('../../../package.json')
|
|
||||||
const flashState = require('./models/flash-state')
|
|
||||||
const settings = require('./models/settings')
|
|
||||||
const windowProgress = require('./os/window-progress')
|
|
||||||
const analytics = require('./modules/analytics')
|
|
||||||
const availableDrives = require('./models/available-drives')
|
|
||||||
const selectionState = require('./models/selection-state')
|
|
||||||
const driveScanner = require('./modules/drive-scanner')
|
|
||||||
const osDialog = require('./os/dialog')
|
|
||||||
const exceptionReporter = require('./modules/exception-reporter')
|
|
||||||
const updateLock = require('./modules/update-lock')
|
|
||||||
|
|
||||||
/* eslint-disable lodash/prefer-lodash-method,lodash/prefer-get */
|
|
||||||
|
|
||||||
// Enable debug information from all modules that use `debug`
|
|
||||||
// See https://github.com/visionmedia/debug#browser-support
|
|
||||||
//
|
|
||||||
// Enable drivelist debugging information
|
|
||||||
// See https://github.com/resin-io-modules/drivelist
|
|
||||||
process.env.DRIVELIST_DEBUG = /drivelist|^\*$/i.test(process.env.DEBUG) ? '1' : ''
|
|
||||||
window.localStorage.debug = process.env.DEBUG
|
|
||||||
|
|
||||||
window.addEventListener('unhandledrejection', (event) => {
|
|
||||||
// Promise: event.reason
|
|
||||||
// Bluebird: event.detail.reason
|
|
||||||
// Anything else: event
|
|
||||||
const error = event.reason || (event.detail && event.detail.reason) || event
|
|
||||||
analytics.logException(error)
|
|
||||||
event.preventDefault()
|
|
||||||
})
|
|
||||||
|
|
||||||
// Set application session UUID
|
|
||||||
store.dispatch({
|
|
||||||
type: store.Actions.SET_APPLICATION_SESSION_UUID,
|
|
||||||
data: uuidV4()
|
|
||||||
})
|
|
||||||
|
|
||||||
// Set first flashing workflow UUID
|
|
||||||
store.dispatch({
|
|
||||||
type: store.Actions.SET_FLASHING_WORKFLOW_UUID,
|
|
||||||
data: uuidV4()
|
|
||||||
})
|
|
||||||
|
|
||||||
const applicationSessionUuid = store.getState().toJS().applicationSessionUuid
|
|
||||||
const flashingWorkflowUuid = store.getState().toJS().flashingWorkflowUuid
|
|
||||||
|
|
||||||
const app = angular.module('Etcher', [
|
|
||||||
require('angular-ui-router'),
|
|
||||||
require('angular-ui-bootstrap'),
|
|
||||||
require('angular-if-state'),
|
|
||||||
|
|
||||||
// Components
|
|
||||||
require('./components/svg-icon'),
|
|
||||||
require('./components/warning-modal/warning-modal'),
|
|
||||||
require('./components/safe-webview'),
|
|
||||||
require('./components/file-selector'),
|
|
||||||
|
|
||||||
// Pages
|
|
||||||
require('./pages/main/main'),
|
|
||||||
require('./pages/finish/finish'),
|
|
||||||
require('./pages/settings/settings'),
|
|
||||||
|
|
||||||
// OS
|
|
||||||
require('./os/open-external/open-external'),
|
|
||||||
require('./os/dropzone/dropzone'),
|
|
||||||
|
|
||||||
// Utils
|
|
||||||
require('./utils/manifest-bind/manifest-bind')
|
|
||||||
])
|
|
||||||
|
|
||||||
app.run(() => {
|
|
||||||
console.log([
|
|
||||||
' _____ _ _',
|
|
||||||
'| ___| | | |',
|
|
||||||
'| |__ | |_ ___| |__ ___ _ __',
|
|
||||||
'| __|| __/ __| \'_ \\ / _ \\ \'__|',
|
|
||||||
'| |___| || (__| | | | __/ |',
|
|
||||||
'\\____/ \\__\\___|_| |_|\\___|_|',
|
|
||||||
'',
|
|
||||||
'Interested in joining the Etcher team?',
|
|
||||||
'Drop us a line at join+etcher@balena.io',
|
|
||||||
'',
|
|
||||||
`Version = ${packageJSON.version}, Type = ${packageJSON.packageType}`
|
|
||||||
].join('\n'))
|
|
||||||
})
|
|
||||||
|
|
||||||
app.run(() => {
|
|
||||||
const currentVersion = packageJSON.version
|
|
||||||
|
|
||||||
analytics.logEvent('Application start', {
|
|
||||||
packageType: packageJSON.packageType,
|
|
||||||
version: currentVersion,
|
|
||||||
applicationSessionUuid
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
app.run(() => {
|
|
||||||
store.observe(() => {
|
|
||||||
if (!flashState.isFlashing()) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const currentFlashState = flashState.getFlashState()
|
|
||||||
const stateType = !currentFlashState.flashing && currentFlashState.verifying
|
|
||||||
? `Verifying ${currentFlashState.verifying}`
|
|
||||||
: `Flashing ${currentFlashState.flashing}`
|
|
||||||
|
|
||||||
// NOTE: There is usually a short time period between the `isFlashing()`
|
|
||||||
// property being set, and the flashing actually starting, which
|
|
||||||
// might cause some non-sense flashing state logs including
|
|
||||||
// `undefined` values.
|
|
||||||
analytics.logDebug(
|
|
||||||
`${stateType} devices, ` +
|
|
||||||
`${currentFlashState.percentage}% at ${currentFlashState.speed} MB/s ` +
|
|
||||||
`(total ${currentFlashState.totalSpeed} MB/s) ` +
|
|
||||||
`eta in ${currentFlashState.eta}s ` +
|
|
||||||
`with ${currentFlashState.failed} failed devices`
|
|
||||||
)
|
|
||||||
|
|
||||||
windowProgress.set(currentFlashState)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary The radix used by USB ID numbers
|
|
||||||
* @type {Number}
|
|
||||||
* @constant
|
|
||||||
*/
|
|
||||||
const USB_ID_RADIX = 16
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary The expected length of a USB ID number
|
|
||||||
* @type {Number}
|
|
||||||
* @constant
|
|
||||||
*/
|
|
||||||
const USB_ID_LENGTH = 4
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Convert a USB id (e.g. product/vendor) to a string
|
|
||||||
* @function
|
|
||||||
* @private
|
|
||||||
*
|
|
||||||
* @param {Number} id - USB id
|
|
||||||
* @returns {String} string id
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* console.log(usbIdToString(2652))
|
|
||||||
* > '0x0a5c'
|
|
||||||
*/
|
|
||||||
const usbIdToString = (id) => {
|
|
||||||
return `0x${_.padStart(id.toString(USB_ID_RADIX), USB_ID_LENGTH, '0')}`
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Product ID of BCM2708
|
|
||||||
* @type {Number}
|
|
||||||
* @constant
|
|
||||||
*/
|
|
||||||
const USB_PRODUCT_ID_BCM2708_BOOT = 0x2763
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Product ID of BCM2710
|
|
||||||
* @type {Number}
|
|
||||||
* @constant
|
|
||||||
*/
|
|
||||||
const USB_PRODUCT_ID_BCM2710_BOOT = 0x2764
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Compute module descriptions
|
|
||||||
* @type {Object}
|
|
||||||
* @constant
|
|
||||||
*/
|
|
||||||
const COMPUTE_MODULE_DESCRIPTIONS = {
|
|
||||||
[USB_PRODUCT_ID_BCM2708_BOOT]: 'Compute Module 1',
|
|
||||||
[USB_PRODUCT_ID_BCM2710_BOOT]: 'Compute Module 3'
|
|
||||||
}
|
|
||||||
|
|
||||||
app.run(($timeout) => {
|
|
||||||
const BLACKLISTED_DRIVES = settings.has('driveBlacklist')
|
|
||||||
? settings.get('driveBlacklist').split(',')
|
|
||||||
: []
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc
|
|
||||||
const driveIsAllowed = (drive) => {
|
|
||||||
return !(
|
|
||||||
BLACKLISTED_DRIVES.includes(drive.devicePath) ||
|
|
||||||
BLACKLISTED_DRIVES.includes(drive.device) ||
|
|
||||||
BLACKLISTED_DRIVES.includes(drive.raw)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc,consistent-return
|
|
||||||
const prepareDrive = (drive) => {
|
|
||||||
if (drive instanceof sdk.sourceDestination.BlockDevice) {
|
|
||||||
return drive.drive
|
|
||||||
} else if (drive instanceof sdk.sourceDestination.UsbbootDrive) {
|
|
||||||
// This is a workaround etcher expecting a device string and a size
|
|
||||||
drive.device = drive.usbDevice.portId
|
|
||||||
drive.size = null
|
|
||||||
drive.progress = 0
|
|
||||||
drive.disabled = true
|
|
||||||
drive.on('progress', (progress) => {
|
|
||||||
updateDriveProgress(drive, progress)
|
|
||||||
})
|
|
||||||
return drive
|
|
||||||
} else if (drive instanceof sdk.sourceDestination.DriverlessDevice) {
|
|
||||||
const description = COMPUTE_MODULE_DESCRIPTIONS[drive.deviceDescriptor.idProduct] || 'Compute Module'
|
|
||||||
return {
|
|
||||||
device: `${usbIdToString(drive.deviceDescriptor.idVendor)}:${usbIdToString(drive.deviceDescriptor.idProduct)}`,
|
|
||||||
displayName: 'Missing drivers',
|
|
||||||
description,
|
|
||||||
mountpoints: [],
|
|
||||||
isReadOnly: false,
|
|
||||||
isSystem: false,
|
|
||||||
disabled: true,
|
|
||||||
icon: 'warning',
|
|
||||||
size: null,
|
|
||||||
link: 'https://www.raspberrypi.org/documentation/hardware/computemodule/cm-emmc-flashing.md',
|
|
||||||
linkCTA: 'Install',
|
|
||||||
linkTitle: 'Install missing drivers',
|
|
||||||
linkMessage: [
|
|
||||||
'Would you like to download the necessary drivers from the Raspberry Pi Foundation?',
|
|
||||||
'This will open your browser.\n\n',
|
|
||||||
'Once opened, download and run the installer from the "Windows Installer" section to install the drivers.'
|
|
||||||
].join(' ')
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc
|
|
||||||
const setDrives = (drives) => {
|
|
||||||
availableDrives.setDrives(_.values(drives))
|
|
||||||
|
|
||||||
// Safely trigger a digest cycle.
|
|
||||||
// In some cases, AngularJS doesn't acknowledge that the
|
|
||||||
// available drives list has changed, and incorrectly
|
|
||||||
// keeps asking the user to "Connect a drive".
|
|
||||||
$timeout()
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc
|
|
||||||
const getDrives = () => {
|
|
||||||
return _.keyBy(availableDrives.getDrives() || [], 'device')
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc
|
|
||||||
const addDrive = (drive) => {
|
|
||||||
const preparedDrive = prepareDrive(drive)
|
|
||||||
if (!driveIsAllowed(preparedDrive)) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const drives = getDrives()
|
|
||||||
drives[preparedDrive.device] = preparedDrive
|
|
||||||
setDrives(drives)
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc
|
|
||||||
const removeDrive = (drive) => {
|
|
||||||
const preparedDrive = prepareDrive(drive)
|
|
||||||
const drives = getDrives()
|
|
||||||
// eslint-disable-next-line prefer-reflect
|
|
||||||
delete drives[preparedDrive.device]
|
|
||||||
setDrives(drives)
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line require-jsdoc
|
|
||||||
const updateDriveProgress = (drive, progress) => {
|
|
||||||
const drives = getDrives()
|
|
||||||
const driveInMap = drives[drive.device]
|
|
||||||
if (driveInMap) {
|
|
||||||
driveInMap.progress = progress
|
|
||||||
setDrives(drives)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
driveScanner.on('attach', addDrive)
|
|
||||||
driveScanner.on('detach', removeDrive)
|
|
||||||
|
|
||||||
driveScanner.on('error', (error) => {
|
|
||||||
// Stop the drive scanning loop in case of errors,
|
|
||||||
// otherwise we risk presenting the same error over
|
|
||||||
// and over again to the user, while also heavily
|
|
||||||
// spamming our error reporting service.
|
|
||||||
driveScanner.stop()
|
|
||||||
|
|
||||||
return exceptionReporter.report(error)
|
|
||||||
})
|
|
||||||
|
|
||||||
driveScanner.start()
|
|
||||||
})
|
|
||||||
|
|
||||||
app.run(($window) => {
|
|
||||||
let popupExists = false
|
|
||||||
|
|
||||||
$window.addEventListener('beforeunload', (event) => {
|
|
||||||
if (!flashState.isFlashing() || popupExists) {
|
|
||||||
analytics.logEvent('Close application', {
|
|
||||||
isFlashing: flashState.isFlashing(),
|
|
||||||
applicationSessionUuid
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't close window while flashing
|
|
||||||
event.returnValue = false
|
|
||||||
|
|
||||||
// Don't open any more popups
|
|
||||||
popupExists = true
|
|
||||||
|
|
||||||
analytics.logEvent('Close attempt while flashing', { applicationSessionUuid, flashingWorkflowUuid })
|
|
||||||
|
|
||||||
osDialog.showWarning({
|
|
||||||
confirmationLabel: 'Yes, quit',
|
|
||||||
rejectionLabel: 'Cancel',
|
|
||||||
title: 'Are you sure you want to close Etcher?',
|
|
||||||
description: messages.warning.exitWhileFlashing()
|
|
||||||
}).then((confirmed) => {
|
|
||||||
if (confirmed) {
|
|
||||||
analytics.logEvent('Close confirmed while flashing', {
|
|
||||||
flashInstanceUuid: flashState.getFlashUuid(),
|
|
||||||
applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
// This circumvents the 'beforeunload' event unlike
|
|
||||||
// electron.remote.app.quit() which does not.
|
|
||||||
electron.remote.process.exit(EXIT_CODES.SUCCESS)
|
|
||||||
}
|
|
||||||
|
|
||||||
analytics.logEvent('Close rejected while flashing', { applicationSessionUuid, flashingWorkflowUuid })
|
|
||||||
popupExists = false
|
|
||||||
}).catch(exceptionReporter.report)
|
|
||||||
})
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Helper fn for events
|
|
||||||
* @function
|
|
||||||
* @private
|
|
||||||
* @example
|
|
||||||
* window.addEventListener('click', extendLock)
|
|
||||||
*/
|
|
||||||
const extendLock = () => {
|
|
||||||
updateLock.extend()
|
|
||||||
}
|
|
||||||
|
|
||||||
$window.addEventListener('click', extendLock)
|
|
||||||
$window.addEventListener('touchstart', extendLock)
|
|
||||||
|
|
||||||
// Initial update lock acquisition
|
|
||||||
extendLock()
|
|
||||||
})
|
|
||||||
|
|
||||||
app.run(($rootScope) => {
|
|
||||||
$rootScope.$on('$stateChangeSuccess', (event, toState, toParams, fromState) => {
|
|
||||||
// Ignore first navigation
|
|
||||||
if (!fromState.name) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
analytics.logEvent('Navigate', {
|
|
||||||
to: toState.name,
|
|
||||||
from: fromState.name,
|
|
||||||
applicationSessionUuid
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
app.config(($urlRouterProvider) => {
|
|
||||||
$urlRouterProvider.otherwise('/main')
|
|
||||||
})
|
|
||||||
|
|
||||||
app.config(($provide) => {
|
|
||||||
$provide.decorator('$exceptionHandler', ($delegate) => {
|
|
||||||
return (exception, cause) => {
|
|
||||||
exceptionReporter.report(exception)
|
|
||||||
$delegate(exception, cause)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
app.config(($locationProvider) => {
|
|
||||||
// NOTE(Shou): this seems to invoke a minor perf decrease when set to true
|
|
||||||
$locationProvider.html5Mode({
|
|
||||||
rewriteLinks: false
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
app.controller('HeaderController', function (OSOpenExternalService) {
|
|
||||||
/**
|
|
||||||
* @summary Open help page
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @description
|
|
||||||
* This application will open either the image's support url, declared
|
|
||||||
* in the archive `manifest.json`, or the default Etcher help page.
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* HeaderController.openHelpPage();
|
|
||||||
*/
|
|
||||||
this.openHelpPage = () => {
|
|
||||||
const DEFAULT_SUPPORT_URL = 'https://github.com/resin-io/etcher/blob/master/SUPPORT.md'
|
|
||||||
const supportUrl = selectionState.getImageSupportUrl() || DEFAULT_SUPPORT_URL
|
|
||||||
OSOpenExternalService.open(supportUrl)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Whether to show the help link
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @returns {Boolean}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* HeaderController.shouldShowHelp()
|
|
||||||
*/
|
|
||||||
this.shouldShowHelp = () => {
|
|
||||||
return !settings.get('disableExternalLinks')
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
app.controller('StateController', function ($rootScope, $scope) {
|
|
||||||
const unregisterStateChange = $rootScope.$on('$stateChangeSuccess', (event, toState, toParams, fromState) => {
|
|
||||||
this.previousName = fromState.name
|
|
||||||
this.currentName = toState.name
|
|
||||||
})
|
|
||||||
|
|
||||||
$scope.$on('$destroy', unregisterStateChange)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Get the previous state name
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @returns {String} previous state name
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* if (StateController.previousName === 'main') {
|
|
||||||
* console.log('We left the main screen!');
|
|
||||||
* }
|
|
||||||
*/
|
|
||||||
this.previousName = null
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Get the current state name
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @returns {String} current state name
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* if (StateController.currentName === 'main') {
|
|
||||||
* console.log('We are on the main screen!');
|
|
||||||
* }
|
|
||||||
*/
|
|
||||||
this.currentName = null
|
|
||||||
})
|
|
||||||
|
|
||||||
// Handle keyboard shortcut to open the settings
|
|
||||||
app.run(($state) => {
|
|
||||||
electron.ipcRenderer.on('menu:preferences', () => {
|
|
||||||
$state.go('settings')
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
// Ensure user settings are loaded before
|
|
||||||
// we bootstrap the Angular.js application
|
|
||||||
angular.element(document).ready(() => {
|
|
||||||
settings.load().then(() => {
|
|
||||||
angular.bootstrap(document, [ 'Etcher' ])
|
|
||||||
}).catch(exceptionReporter.report)
|
|
||||||
})
|
|
||||||
364
lib/gui/app/app.ts
Normal file
364
lib/gui/app/app.ts
Normal file
@@ -0,0 +1,364 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as electron from 'electron';
|
||||||
|
import * as sdk from 'etcher-sdk';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
import outdent from 'outdent';
|
||||||
|
import * as React from 'react';
|
||||||
|
import * as ReactDOM from 'react-dom';
|
||||||
|
import { v4 as uuidV4 } from 'uuid';
|
||||||
|
|
||||||
|
import * as packageJSON from '../../../package.json';
|
||||||
|
import { DrivelistDrive, isSourceDrive } from '../../shared/drive-constraints';
|
||||||
|
import * as EXIT_CODES from '../../shared/exit-codes';
|
||||||
|
import * as messages from '../../shared/messages';
|
||||||
|
import * as availableDrives from './models/available-drives';
|
||||||
|
import * as flashState from './models/flash-state';
|
||||||
|
import { deselectImage, getImage } from './models/selection-state';
|
||||||
|
import * as settings from './models/settings';
|
||||||
|
import { Actions, observe, store } from './models/store';
|
||||||
|
import * as analytics from './modules/analytics';
|
||||||
|
import { scanner as driveScanner } from './modules/drive-scanner';
|
||||||
|
import * as exceptionReporter from './modules/exception-reporter';
|
||||||
|
import * as osDialog from './os/dialog';
|
||||||
|
import * as windowProgress from './os/window-progress';
|
||||||
|
import MainPage from './pages/main/MainPage';
|
||||||
|
import './css/main.css';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
window.addEventListener(
|
||||||
|
'unhandledrejection',
|
||||||
|
(event: PromiseRejectionEvent | any) => {
|
||||||
|
// Promise: event.reason
|
||||||
|
// Anything else: event
|
||||||
|
const error = event.reason || event;
|
||||||
|
analytics.logException(error);
|
||||||
|
event.preventDefault();
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
// Set application session UUID
|
||||||
|
store.dispatch({
|
||||||
|
type: Actions.SET_APPLICATION_SESSION_UUID,
|
||||||
|
data: uuidV4(),
|
||||||
|
});
|
||||||
|
|
||||||
|
// Set first flashing workflow UUID
|
||||||
|
store.dispatch({
|
||||||
|
type: Actions.SET_FLASHING_WORKFLOW_UUID,
|
||||||
|
data: uuidV4(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const applicationSessionUuid = store.getState().toJS().applicationSessionUuid;
|
||||||
|
const flashingWorkflowUuid = store.getState().toJS().flashingWorkflowUuid;
|
||||||
|
|
||||||
|
console.log(outdent`
|
||||||
|
${outdent}
|
||||||
|
_____ _ _
|
||||||
|
| ___| | | |
|
||||||
|
| |__ | |_ ___| |__ ___ _ __
|
||||||
|
| __|| __/ __| '_ \\ / _ \\ '__|
|
||||||
|
| |___| || (__| | | | __/ |
|
||||||
|
\\____/ \\__\\___|_| |_|\\___|_|
|
||||||
|
|
||||||
|
Interested in joining the Etcher team?
|
||||||
|
Drop us a line at join+etcher@balena.io
|
||||||
|
|
||||||
|
Version = ${packageJSON.version}, Type = ${packageJSON.packageType}
|
||||||
|
`);
|
||||||
|
|
||||||
|
const currentVersion = packageJSON.version;
|
||||||
|
|
||||||
|
analytics.logEvent('Application start', {
|
||||||
|
packageType: packageJSON.packageType,
|
||||||
|
version: currentVersion,
|
||||||
|
});
|
||||||
|
|
||||||
|
const debouncedLog = _.debounce(console.log, 1000, { maxWait: 1000 });
|
||||||
|
|
||||||
|
function pluralize(word: string, quantity: number) {
|
||||||
|
return `${quantity} ${word}${quantity === 1 ? '' : 's'}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
observe(() => {
|
||||||
|
if (!flashState.isFlashing()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const currentFlashState = flashState.getFlashState();
|
||||||
|
windowProgress.set(currentFlashState);
|
||||||
|
|
||||||
|
let eta = '';
|
||||||
|
if (currentFlashState.eta !== undefined) {
|
||||||
|
eta = `eta in ${currentFlashState.eta.toFixed(0)}s`;
|
||||||
|
}
|
||||||
|
let active = '';
|
||||||
|
if (currentFlashState.type !== 'decompressing') {
|
||||||
|
active = pluralize('device', currentFlashState.active);
|
||||||
|
}
|
||||||
|
// NOTE: There is usually a short time period between the `isFlashing()`
|
||||||
|
// property being set, and the flashing actually starting, which
|
||||||
|
// might cause some non-sense flashing state logs including
|
||||||
|
// `undefined` values.
|
||||||
|
debouncedLog(outdent({ newline: ' ' })`
|
||||||
|
${_.capitalize(currentFlashState.type)}
|
||||||
|
${active},
|
||||||
|
${currentFlashState.percentage}%
|
||||||
|
at
|
||||||
|
${(currentFlashState.speed || 0).toFixed(2)}
|
||||||
|
MB/s
|
||||||
|
(total ${(currentFlashState.speed * currentFlashState.active).toFixed(2)} MB/s)
|
||||||
|
${eta}
|
||||||
|
with
|
||||||
|
${pluralize('failed device', currentFlashState.failed)}
|
||||||
|
`);
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary The radix used by USB ID numbers
|
||||||
|
*/
|
||||||
|
const USB_ID_RADIX = 16;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary The expected length of a USB ID number
|
||||||
|
*/
|
||||||
|
const USB_ID_LENGTH = 4;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Convert a USB id (e.g. product/vendor) to a string
|
||||||
|
*
|
||||||
|
* @example
|
||||||
|
* console.log(usbIdToString(2652))
|
||||||
|
* > '0x0a5c'
|
||||||
|
*/
|
||||||
|
function usbIdToString(id: number): string {
|
||||||
|
return `0x${_.padStart(id.toString(USB_ID_RADIX), USB_ID_LENGTH, '0')}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Product ID of BCM2708
|
||||||
|
*/
|
||||||
|
const USB_PRODUCT_ID_BCM2708_BOOT = 0x2763;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Product ID of BCM2710
|
||||||
|
*/
|
||||||
|
const USB_PRODUCT_ID_BCM2710_BOOT = 0x2764;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Compute module descriptions
|
||||||
|
*/
|
||||||
|
const COMPUTE_MODULE_DESCRIPTIONS: _.Dictionary<string> = {
|
||||||
|
[USB_PRODUCT_ID_BCM2708_BOOT]: 'Compute Module 1',
|
||||||
|
[USB_PRODUCT_ID_BCM2710_BOOT]: 'Compute Module 3',
|
||||||
|
};
|
||||||
|
|
||||||
|
async function driveIsAllowed(drive: {
|
||||||
|
devicePath: string;
|
||||||
|
device: string;
|
||||||
|
raw: string;
|
||||||
|
}) {
|
||||||
|
const driveBlacklist = (await settings.get('driveBlacklist')) || [];
|
||||||
|
return !(
|
||||||
|
driveBlacklist.includes(drive.devicePath) ||
|
||||||
|
driveBlacklist.includes(drive.device) ||
|
||||||
|
driveBlacklist.includes(drive.raw)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
type Drive =
|
||||||
|
| sdk.sourceDestination.BlockDevice
|
||||||
|
| sdk.sourceDestination.UsbbootDrive
|
||||||
|
| sdk.sourceDestination.DriverlessDevice;
|
||||||
|
|
||||||
|
function prepareDrive(drive: Drive) {
|
||||||
|
if (drive instanceof sdk.sourceDestination.BlockDevice) {
|
||||||
|
// @ts-ignore (BlockDevice.drive is private)
|
||||||
|
return drive.drive;
|
||||||
|
} else if (drive instanceof sdk.sourceDestination.UsbbootDrive) {
|
||||||
|
// This is a workaround etcher expecting a device string and a size
|
||||||
|
// @ts-ignore
|
||||||
|
drive.device = drive.usbDevice.portId;
|
||||||
|
drive.size = null;
|
||||||
|
// @ts-ignore
|
||||||
|
drive.progress = 0;
|
||||||
|
drive.disabled = true;
|
||||||
|
drive.on('progress', (progress) => {
|
||||||
|
updateDriveProgress(drive, progress);
|
||||||
|
});
|
||||||
|
return drive;
|
||||||
|
} else if (drive instanceof sdk.sourceDestination.DriverlessDevice) {
|
||||||
|
const description =
|
||||||
|
COMPUTE_MODULE_DESCRIPTIONS[
|
||||||
|
drive.deviceDescriptor.idProduct.toString()
|
||||||
|
] || 'Compute Module';
|
||||||
|
return {
|
||||||
|
device: `${usbIdToString(
|
||||||
|
drive.deviceDescriptor.idVendor,
|
||||||
|
)}:${usbIdToString(drive.deviceDescriptor.idProduct)}`,
|
||||||
|
displayName: 'Missing drivers',
|
||||||
|
description,
|
||||||
|
mountpoints: [],
|
||||||
|
isReadOnly: false,
|
||||||
|
isSystem: false,
|
||||||
|
disabled: true,
|
||||||
|
icon: 'warning',
|
||||||
|
size: null,
|
||||||
|
link: 'https://www.raspberrypi.com/documentation/computers/compute-module.html#flashing-the-compute-module-emmc',
|
||||||
|
linkCTA: 'Install',
|
||||||
|
linkTitle: 'Install missing drivers',
|
||||||
|
linkMessage: outdent`
|
||||||
|
Would you like to download the necessary drivers from the Raspberry Pi Foundation?
|
||||||
|
This will open your browser.
|
||||||
|
|
||||||
|
|
||||||
|
Once opened, download and run the installer from the "Windows Installer" section to install the drivers
|
||||||
|
`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setDrives(drives: _.Dictionary<DrivelistDrive>) {
|
||||||
|
availableDrives.setDrives(_.values(drives));
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDrives() {
|
||||||
|
return _.keyBy(availableDrives.getDrives(), 'device');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function addDrive(drive: Drive) {
|
||||||
|
const preparedDrive = prepareDrive(drive);
|
||||||
|
if (!(await driveIsAllowed(preparedDrive))) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const drives = getDrives();
|
||||||
|
drives[preparedDrive.device] = preparedDrive;
|
||||||
|
setDrives(drives);
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeDrive(drive: Drive) {
|
||||||
|
if (
|
||||||
|
drive instanceof sdk.sourceDestination.BlockDevice &&
|
||||||
|
// @ts-ignore BlockDevice.drive is private
|
||||||
|
isSourceDrive(drive.drive, getImage())
|
||||||
|
) {
|
||||||
|
// Deselect the image if it was on the drive that was removed.
|
||||||
|
// This will also deselect the image if the drive mountpoints change.
|
||||||
|
deselectImage();
|
||||||
|
}
|
||||||
|
const preparedDrive = prepareDrive(drive);
|
||||||
|
const drives = getDrives();
|
||||||
|
delete drives[preparedDrive.device];
|
||||||
|
setDrives(drives);
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateDriveProgress(
|
||||||
|
drive: sdk.sourceDestination.UsbbootDrive,
|
||||||
|
progress: number,
|
||||||
|
) {
|
||||||
|
const drives = getDrives();
|
||||||
|
// @ts-ignore
|
||||||
|
const driveInMap = drives[drive.device];
|
||||||
|
if (driveInMap) {
|
||||||
|
// @ts-ignore
|
||||||
|
drives[drive.device] = { ...driveInMap, progress };
|
||||||
|
setDrives(drives);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
driveScanner.on('attach', addDrive);
|
||||||
|
driveScanner.on('detach', removeDrive);
|
||||||
|
|
||||||
|
driveScanner.on('error', (error) => {
|
||||||
|
// Stop the drive scanning loop in case of errors,
|
||||||
|
// otherwise we risk presenting the same error over
|
||||||
|
// and over again to the user, while also heavily
|
||||||
|
// spamming our error reporting service.
|
||||||
|
driveScanner.stop();
|
||||||
|
|
||||||
|
return exceptionReporter.report(error);
|
||||||
|
});
|
||||||
|
|
||||||
|
driveScanner.start();
|
||||||
|
|
||||||
|
let popupExists = false;
|
||||||
|
|
||||||
|
window.addEventListener('beforeunload', async (event) => {
|
||||||
|
if (!flashState.isFlashing() || popupExists) {
|
||||||
|
analytics.logEvent('Close application', {
|
||||||
|
isFlashing: flashState.isFlashing(),
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Don't close window while flashing
|
||||||
|
event.returnValue = false;
|
||||||
|
|
||||||
|
// Don't open any more popups
|
||||||
|
popupExists = true;
|
||||||
|
|
||||||
|
analytics.logEvent('Close attempt while flashing');
|
||||||
|
|
||||||
|
try {
|
||||||
|
const confirmed = await osDialog.showWarning({
|
||||||
|
confirmationLabel: i18next.t('yesExit'),
|
||||||
|
rejectionLabel: i18next.t('cancel'),
|
||||||
|
title: i18next.t('reallyExit'),
|
||||||
|
description: messages.warning.exitWhileFlashing(),
|
||||||
|
});
|
||||||
|
if (confirmed) {
|
||||||
|
analytics.logEvent('Close confirmed while flashing', {
|
||||||
|
flashInstanceUuid: flashState.getFlashUuid(),
|
||||||
|
});
|
||||||
|
|
||||||
|
// This circumvents the 'beforeunload' event unlike
|
||||||
|
// electron.remote.app.quit() which does not.
|
||||||
|
electron.remote.process.exit(EXIT_CODES.SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
|
analytics.logEvent('Close rejected while flashing', {
|
||||||
|
applicationSessionUuid,
|
||||||
|
flashingWorkflowUuid,
|
||||||
|
});
|
||||||
|
popupExists = false;
|
||||||
|
} catch (error: any) {
|
||||||
|
exceptionReporter.report(error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
export async function main() {
|
||||||
|
try {
|
||||||
|
const { init: ledsInit } = require('./models/leds');
|
||||||
|
await ledsInit();
|
||||||
|
} catch (error: any) {
|
||||||
|
exceptionReporter.report(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
ReactDOM.render(
|
||||||
|
React.createElement(MainPage),
|
||||||
|
document.getElementById('main'),
|
||||||
|
// callback to set the correct zoomFactor for webviews as well
|
||||||
|
async () => {
|
||||||
|
const fullscreen = await settings.get('fullscreen');
|
||||||
|
const width = fullscreen ? window.screen.width : window.outerWidth;
|
||||||
|
try {
|
||||||
|
electron.webFrame.setZoomFactor(width / settings.DEFAULT_WIDTH);
|
||||||
|
} catch (err) {
|
||||||
|
// noop
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.ConfirmModal
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const MODULE_NAME = 'Etcher.Components.ConfirmModal'
|
|
||||||
const ConfirmModal = angular.module(MODULE_NAME, [
|
|
||||||
require('../modal/modal')
|
|
||||||
])
|
|
||||||
|
|
||||||
ConfirmModal.controller('ConfirmModalController', require('./controllers/confirm-modal'))
|
|
||||||
ConfirmModal.service('ConfirmModalService', require('./services/confirm-modal'))
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
module.exports = function ($uibModalInstance, options) {
|
|
||||||
/**
|
|
||||||
* @summary Modal options
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
this.options = options
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Reject the warning prompt
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* WarningModalController.reject();
|
|
||||||
*/
|
|
||||||
this.reject = () => {
|
|
||||||
$uibModalInstance.close(false)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Accept the warning prompt
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* WarningModalController.accept();
|
|
||||||
*/
|
|
||||||
this.accept = () => {
|
|
||||||
$uibModalInstance.close(true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
|
|
||||||
module.exports = function ($sce, ModalService) {
|
|
||||||
/**
|
|
||||||
* @summary show the confirm modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} options - options
|
|
||||||
* @param {String} options.description - danger message
|
|
||||||
* @param {String} options.confirmationLabel - confirmation button text
|
|
||||||
* @param {String} options.rejectionLabel - rejection button text
|
|
||||||
* @fulfil {Boolean} - whether the user accepted or rejected the confirm
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* ConfirmModalService.show({
|
|
||||||
* description: 'Don\'t do this!',
|
|
||||||
* confirmationLabel: 'Yes, continue!'
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.show = (options = {}) => {
|
|
||||||
options.description = $sce.trustAsHtml(options.description)
|
|
||||||
return ModalService.open({
|
|
||||||
name: 'confirm',
|
|
||||||
template: require('../templates/confirm-modal.tpl.html'),
|
|
||||||
controller: 'ConfirmModalController as modal',
|
|
||||||
size: 'confirm-modal',
|
|
||||||
resolve: {
|
|
||||||
options: _.constant(options)
|
|
||||||
}
|
|
||||||
}).result
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
.modal-confirm-modal .modal-content {
|
|
||||||
width: 350px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-confirm-modal .modal-title .glyphicon {
|
|
||||||
color: $palette-theme-danger-background;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-confirm-modal .modal-body {
|
|
||||||
max-height: 200px;
|
|
||||||
overflow-y: auto;
|
|
||||||
}
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
<div class="modal-header">
|
|
||||||
<h4 class="modal-title">
|
|
||||||
<span>{{ ::modal.options.title }}</span>
|
|
||||||
</h4>
|
|
||||||
<button class="close"
|
|
||||||
tabindex="11"
|
|
||||||
ng-click="modal.reject()">×</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-body">
|
|
||||||
<p>{{ ::modal.options.message }}</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-footer">
|
|
||||||
<div class="modal-menu">
|
|
||||||
<button ng-if="modal.options.rejectionLabel" class="button button-block"
|
|
||||||
tabindex="12"
|
|
||||||
ng-class="{
|
|
||||||
'button-default': modal.options.cancelButton === 'default',
|
|
||||||
'button-primary': modal.options.cancelButton === 'primary',
|
|
||||||
'button-warning': modal.options.cancelButton === 'warning',
|
|
||||||
'button-danger': modal.options.cancelButton === 'danger',
|
|
||||||
}"
|
|
||||||
ng-click="modal.reject()">{{ ::modal.options.rejectionLabel }}</button>
|
|
||||||
<button class="button button-block"
|
|
||||||
tabindex="13"
|
|
||||||
ng-class="{
|
|
||||||
'button-default': modal.options.confirmButton === 'default',
|
|
||||||
'button-primary': modal.options.confirmButton === 'primary',
|
|
||||||
'button-warning': modal.options.confirmButton === 'warning',
|
|
||||||
'button-danger': modal.options.confirmButton === 'danger',
|
|
||||||
}"
|
|
||||||
ng-click="modal.accept()">{{ ::modal.options.confirmationLabel }}</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@@ -1,265 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const _ = require('lodash')
|
|
||||||
const Bluebird = require('bluebird')
|
|
||||||
const constraints = require('../../../../../shared/drive-constraints')
|
|
||||||
const store = require('../../../models/store')
|
|
||||||
const analytics = require('../../../modules/analytics')
|
|
||||||
const availableDrives = require('../../../models/available-drives')
|
|
||||||
const selectionState = require('../../../models/selection-state')
|
|
||||||
const utils = require('../../../../../shared/utils')
|
|
||||||
|
|
||||||
module.exports = function (
|
|
||||||
$q,
|
|
||||||
$uibModalInstance,
|
|
||||||
ConfirmModalService,
|
|
||||||
OSOpenExternalService
|
|
||||||
) {
|
|
||||||
/**
|
|
||||||
* @summary The drive selector state
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
this.state = selectionState
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Static methods to check a drive's properties
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
this.constraints = constraints
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary The drives model
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @description
|
|
||||||
* We expose the whole service instead of the `.drives`
|
|
||||||
* property, which is the one we're interested in since
|
|
||||||
* this allows the property to be automatically updated
|
|
||||||
* when `availableDrives` detects a change in the drives.
|
|
||||||
*/
|
|
||||||
this.drives = availableDrives
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Determine if we can change a drive's selection state
|
|
||||||
* @function
|
|
||||||
* @private
|
|
||||||
*
|
|
||||||
* @param {Object} drive - drive
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorController.shouldChangeDriveSelectionState(drive)
|
|
||||||
* .then((shouldChangeDriveSelectionState) => {
|
|
||||||
* if (shouldChangeDriveSelectionState) doSomething();
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
const shouldChangeDriveSelectionState = (drive) => {
|
|
||||||
return $q.resolve(constraints.isDriveValid(drive, selectionState.getImage()))
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Toggle a drive selection
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} drive - drive
|
|
||||||
* @returns {Promise} - resolved promise
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorController.toggleDrive({
|
|
||||||
* device: '/dev/disk2',
|
|
||||||
* size: 999999999,
|
|
||||||
* name: 'Cruzer USB drive'
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.toggleDrive = (drive) => {
|
|
||||||
return shouldChangeDriveSelectionState(drive).then((canChangeDriveSelectionState) => {
|
|
||||||
if (canChangeDriveSelectionState) {
|
|
||||||
analytics.logEvent('Toggle drive', {
|
|
||||||
drive,
|
|
||||||
previouslySelected: selectionState.isCurrentDrive(drive.device),
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
selectionState.toggleDrive(drive.device)
|
|
||||||
}
|
|
||||||
|
|
||||||
return Bluebird.resolve()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Prompt the user to install missing usbboot drivers
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} drive - drive
|
|
||||||
* @returns {Promise} - resolved promise
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorController.installMissingDrivers({
|
|
||||||
* linkTitle: 'Go to example.com',
|
|
||||||
* linkMessage: 'Examples are great, right?',
|
|
||||||
* linkCTA: 'Call To Action',
|
|
||||||
* link: 'https://example.com'
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.installMissingDrivers = (drive) => {
|
|
||||||
if (drive.link) {
|
|
||||||
analytics.logEvent('Open driver link modal', {
|
|
||||||
url: drive.link,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
return ConfirmModalService.show({
|
|
||||||
confirmationLabel: 'Yes, continue',
|
|
||||||
rejectionLabel: 'Cancel',
|
|
||||||
title: drive.linkTitle,
|
|
||||||
confirmButton: 'primary',
|
|
||||||
message: drive.linkMessage || `Etcher will open ${drive.link} in your browser`
|
|
||||||
}).then((shouldContinue) => {
|
|
||||||
if (shouldContinue) {
|
|
||||||
OSOpenExternalService.open(drive.link)
|
|
||||||
}
|
|
||||||
}).catch((error) => {
|
|
||||||
analytics.logException(error)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return Bluebird.resolve()
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Close the modal and resolve the selected drive
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorController.closeModal();
|
|
||||||
*/
|
|
||||||
this.closeModal = () => {
|
|
||||||
const selectedDrive = selectionState.getCurrentDrive()
|
|
||||||
|
|
||||||
// Sanity check to cover the case where a drive is selected,
|
|
||||||
// the drive is then unplugged from the computer and the modal
|
|
||||||
// is resolved with a non-existent drive.
|
|
||||||
if (!selectedDrive || !_.includes(this.drives.getDrives(), selectedDrive)) {
|
|
||||||
$uibModalInstance.close()
|
|
||||||
} else {
|
|
||||||
$uibModalInstance.close(selectedDrive)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Select a drive and close the modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} drive - drive
|
|
||||||
* @returns {Promise} - resolved promise
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorController.selectDriveAndClose({
|
|
||||||
* device: '/dev/disk2',
|
|
||||||
* size: 999999999,
|
|
||||||
* name: 'Cruzer USB drive'
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.selectDriveAndClose = (drive) => {
|
|
||||||
return shouldChangeDriveSelectionState(drive).then((canChangeDriveSelectionState) => {
|
|
||||||
if (canChangeDriveSelectionState) {
|
|
||||||
selectionState.selectDrive(drive.device)
|
|
||||||
|
|
||||||
analytics.logEvent('Drive selected (double click)', {
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
this.closeModal()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Memoized getDrives function
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @returns {Array<Object>} - memoized list of drives
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* const drives = DriveSelectorController.getDrives()
|
|
||||||
* // Do something with drives
|
|
||||||
*/
|
|
||||||
this.getDrives = utils.memoize(this.drives.getDrives, angular.equals)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Get a drive's compatibility status object(s)
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @description
|
|
||||||
* Given a drive, return its compatibility status with the selected image,
|
|
||||||
* containing the status type (ERROR, WARNING), and accompanying
|
|
||||||
* status message.
|
|
||||||
*
|
|
||||||
* @returns {Object[]} list of objects containing statuses
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* const statuses = DriveSelectorController.getDriveStatuses(drive);
|
|
||||||
*
|
|
||||||
* for ({ type, message } of statuses) {
|
|
||||||
* // do something
|
|
||||||
* }
|
|
||||||
*/
|
|
||||||
this.getDriveStatuses = utils.memoize((drive) => {
|
|
||||||
return this.constraints.getDriveImageCompatibilityStatuses(drive, this.state.getImage())
|
|
||||||
}, angular.equals)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Keyboard event drive toggling
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @description
|
|
||||||
* Keyboard-event specific entry to the toggleDrive function.
|
|
||||||
*
|
|
||||||
* @param {Object} drive - drive
|
|
||||||
* @param {Object} $event - event
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* <div tabindex="1" ng-keypress="this.keyboardToggleDrive(drive, $event)">
|
|
||||||
* Tab-select me and press enter or space!
|
|
||||||
* </div>
|
|
||||||
*/
|
|
||||||
this.keyboardToggleDrive = (drive, $event) => {
|
|
||||||
console.log($event.keyCode)
|
|
||||||
const ENTER = 13
|
|
||||||
const SPACE = 32
|
|
||||||
if (_.includes([ ENTER, SPACE ], $event.keyCode)) {
|
|
||||||
this.toggleDrive(drive)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.DriveSelector
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const MODULE_NAME = 'Etcher.Components.DriveSelector'
|
|
||||||
const DriveSelector = angular.module(MODULE_NAME, [
|
|
||||||
require('../modal/modal'),
|
|
||||||
require('../confirm-modal/confirm-modal'),
|
|
||||||
require('../../utils/byte-size/byte-size'),
|
|
||||||
require('../../os/open-external/open-external')
|
|
||||||
])
|
|
||||||
|
|
||||||
DriveSelector.controller('DriveSelectorController', require('./controllers/drive-selector'))
|
|
||||||
DriveSelector.service('DriveSelectorService', require('./services/drive-selector'))
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
563
lib/gui/app/components/drive-selector/drive-selector.tsx
Normal file
563
lib/gui/app/components/drive-selector/drive-selector.tsx
Normal file
@@ -0,0 +1,563 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2019 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ExclamationTriangleSvg from '@fortawesome/fontawesome-free/svgs/solid/exclamation-triangle.svg';
|
||||||
|
import ChevronDownSvg from '@fortawesome/fontawesome-free/svgs/solid/chevron-down.svg';
|
||||||
|
import * as sourceDestination from 'etcher-sdk/build/source-destination/';
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex, ModalProps, Txt, Badge, Link, TableColumn } from 'rendition';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
import {
|
||||||
|
getDriveImageCompatibilityStatuses,
|
||||||
|
isDriveValid,
|
||||||
|
DriveStatus,
|
||||||
|
DrivelistDrive,
|
||||||
|
isDriveSizeLarge,
|
||||||
|
} from '../../../../shared/drive-constraints';
|
||||||
|
import { compatibility, warning } from '../../../../shared/messages';
|
||||||
|
import * as prettyBytes from 'pretty-bytes';
|
||||||
|
import { getDrives, hasAvailableDrives } from '../../models/available-drives';
|
||||||
|
import { getImage, isDriveSelected } from '../../models/selection-state';
|
||||||
|
import { store } from '../../models/store';
|
||||||
|
import { logEvent, logException } from '../../modules/analytics';
|
||||||
|
import { open as openExternal } from '../../os/open-external/services/open-external';
|
||||||
|
import {
|
||||||
|
Alert,
|
||||||
|
GenericTableProps,
|
||||||
|
Modal,
|
||||||
|
Table,
|
||||||
|
} from '../../styled-components';
|
||||||
|
|
||||||
|
import { SourceMetadata } from '../source-selector/source-selector';
|
||||||
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
interface UsbbootDrive extends sourceDestination.UsbbootDrive {
|
||||||
|
progress: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DriverlessDrive {
|
||||||
|
displayName: string; // added in app.ts
|
||||||
|
description: string;
|
||||||
|
link: string;
|
||||||
|
linkTitle: string;
|
||||||
|
linkMessage: string;
|
||||||
|
linkCTA: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
type Drive = DrivelistDrive | DriverlessDrive | UsbbootDrive;
|
||||||
|
|
||||||
|
function isUsbbootDrive(drive: Drive): drive is UsbbootDrive {
|
||||||
|
return (drive as UsbbootDrive).progress !== undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isDriverlessDrive(drive: Drive): drive is DriverlessDrive {
|
||||||
|
return (drive as DriverlessDrive).link !== undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isDrivelistDrive(drive: Drive): drive is DrivelistDrive {
|
||||||
|
return typeof (drive as DrivelistDrive).size === 'number';
|
||||||
|
}
|
||||||
|
|
||||||
|
const DrivesTable = styled((props: GenericTableProps<Drive>) => (
|
||||||
|
<Table<Drive> {...props} />
|
||||||
|
))`
|
||||||
|
[data-display='table-head'],
|
||||||
|
[data-display='table-body'] {
|
||||||
|
> [data-display='table-row'] > [data-display='table-cell'] {
|
||||||
|
&:nth-child(2) {
|
||||||
|
width: 32%;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:nth-child(3) {
|
||||||
|
width: 15%;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:nth-child(4) {
|
||||||
|
width: 15%;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:nth-child(5) {
|
||||||
|
width: 32%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
function badgeShadeFromStatus(status: string) {
|
||||||
|
switch (status) {
|
||||||
|
case compatibility.containsImage():
|
||||||
|
return 16;
|
||||||
|
case compatibility.system():
|
||||||
|
case compatibility.tooSmall():
|
||||||
|
return 5;
|
||||||
|
default:
|
||||||
|
return 14;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const InitProgress = styled(
|
||||||
|
({
|
||||||
|
value,
|
||||||
|
...props
|
||||||
|
}: {
|
||||||
|
value: number;
|
||||||
|
props?: React.ProgressHTMLAttributes<Element>;
|
||||||
|
}) => {
|
||||||
|
return <progress max="100" value={value} {...props} />;
|
||||||
|
},
|
||||||
|
)`
|
||||||
|
/* Reset the default appearance */
|
||||||
|
appearance: none;
|
||||||
|
|
||||||
|
::-webkit-progress-bar {
|
||||||
|
width: 130px;
|
||||||
|
height: 4px;
|
||||||
|
background-color: #dde1f0;
|
||||||
|
border-radius: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
::-webkit-progress-value {
|
||||||
|
background-color: #1496e1;
|
||||||
|
border-radius: 14px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export interface DriveSelectorProps
|
||||||
|
extends Omit<ModalProps, 'done' | 'cancel' | 'onSelect'> {
|
||||||
|
write: boolean;
|
||||||
|
multipleSelection: boolean;
|
||||||
|
showWarnings?: boolean;
|
||||||
|
cancel: (drives: DrivelistDrive[]) => void;
|
||||||
|
done: (drives: DrivelistDrive[]) => void;
|
||||||
|
titleLabel: string;
|
||||||
|
emptyListLabel: string;
|
||||||
|
emptyListIcon: JSX.Element;
|
||||||
|
selectedList?: DrivelistDrive[];
|
||||||
|
updateSelectedList?: () => DrivelistDrive[];
|
||||||
|
onSelect?: (drive: DrivelistDrive) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DriveSelectorState {
|
||||||
|
drives: Drive[];
|
||||||
|
image?: SourceMetadata;
|
||||||
|
missingDriversModal: { drive?: DriverlessDrive };
|
||||||
|
selectedList: DrivelistDrive[];
|
||||||
|
showSystemDrives: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isSystemDrive(drive: Drive) {
|
||||||
|
return isDrivelistDrive(drive) && drive.isSystem;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DriveSelector extends React.Component<
|
||||||
|
DriveSelectorProps,
|
||||||
|
DriveSelectorState
|
||||||
|
> {
|
||||||
|
private unsubscribe: (() => void) | undefined;
|
||||||
|
tableColumns: Array<TableColumn<Drive>>;
|
||||||
|
originalList: DrivelistDrive[];
|
||||||
|
|
||||||
|
constructor(props: DriveSelectorProps) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
const defaultMissingDriversModalState: { drive?: DriverlessDrive } = {};
|
||||||
|
const selectedList = this.props.selectedList || [];
|
||||||
|
this.originalList = [...(this.props.selectedList || [])];
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
drives: getDrives(),
|
||||||
|
image: getImage(),
|
||||||
|
missingDriversModal: defaultMissingDriversModalState,
|
||||||
|
selectedList,
|
||||||
|
showSystemDrives: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.tableColumns = [
|
||||||
|
{
|
||||||
|
field: 'description',
|
||||||
|
label: i18next.t('drives.name'),
|
||||||
|
render: (description: string, drive: Drive) => {
|
||||||
|
if (isDrivelistDrive(drive)) {
|
||||||
|
const isLargeDrive = isDriveSizeLarge(drive);
|
||||||
|
const hasWarnings =
|
||||||
|
this.props.showWarnings && (isLargeDrive || drive.isSystem);
|
||||||
|
return (
|
||||||
|
<Flex alignItems="center">
|
||||||
|
{hasWarnings && (
|
||||||
|
<ExclamationTriangleSvg
|
||||||
|
height="1em"
|
||||||
|
fill={drive.isSystem ? '#fca321' : '#8f9297'}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<Txt ml={(hasWarnings && 8) || 0}>
|
||||||
|
{middleEllipsis(description, 32)}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return <Txt>{description}</Txt>;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'description',
|
||||||
|
key: 'size',
|
||||||
|
label: i18next.t('drives.size'),
|
||||||
|
render: (_description: string, drive: Drive) => {
|
||||||
|
if (isDrivelistDrive(drive) && drive.size !== null) {
|
||||||
|
return prettyBytes(drive.size);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'description',
|
||||||
|
key: 'link',
|
||||||
|
label: i18next.t('drives.location'),
|
||||||
|
render: (_description: string, drive: Drive) => {
|
||||||
|
return (
|
||||||
|
<Txt>
|
||||||
|
{drive.displayName}
|
||||||
|
{isDriverlessDrive(drive) && (
|
||||||
|
<>
|
||||||
|
{' '}
|
||||||
|
-{' '}
|
||||||
|
<b>
|
||||||
|
<a onClick={() => this.installMissingDrivers(drive)}>
|
||||||
|
{drive.linkCTA}
|
||||||
|
</a>
|
||||||
|
</b>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Txt>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'description',
|
||||||
|
key: 'extra',
|
||||||
|
// We use an empty React fragment otherwise it uses the field name as label
|
||||||
|
label: <></>,
|
||||||
|
render: (_description: string, drive: Drive) => {
|
||||||
|
if (isUsbbootDrive(drive)) {
|
||||||
|
return this.renderProgress(drive.progress);
|
||||||
|
} else if (isDrivelistDrive(drive)) {
|
||||||
|
return this.renderStatuses(drive);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private driveShouldBeDisabled(drive: Drive, image?: SourceMetadata) {
|
||||||
|
return (
|
||||||
|
isUsbbootDrive(drive) ||
|
||||||
|
isDriverlessDrive(drive) ||
|
||||||
|
!isDriveValid(drive, image, this.props.write) ||
|
||||||
|
(this.props.write && drive.isReadOnly)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private getDisplayedDrives(drives: Drive[]): Drive[] {
|
||||||
|
return drives.filter((drive) => {
|
||||||
|
return (
|
||||||
|
isUsbbootDrive(drive) ||
|
||||||
|
isDriverlessDrive(drive) ||
|
||||||
|
isDriveSelected(drive.device) ||
|
||||||
|
this.state.showSystemDrives ||
|
||||||
|
!drive.isSystem
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private getDisabledDrives(drives: Drive[], image?: SourceMetadata): string[] {
|
||||||
|
return drives
|
||||||
|
.filter((drive) => this.driveShouldBeDisabled(drive, image))
|
||||||
|
.map((drive) => drive.displayName);
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderProgress(progress: number) {
|
||||||
|
return (
|
||||||
|
<Flex flexDirection="column">
|
||||||
|
<Txt fontSize={12}>Initializing device</Txt>
|
||||||
|
<InitProgress value={progress} />
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private warningFromStatus(
|
||||||
|
status: string,
|
||||||
|
drive: { device: string; size: number },
|
||||||
|
) {
|
||||||
|
switch (status) {
|
||||||
|
case compatibility.containsImage():
|
||||||
|
return warning.sourceDrive();
|
||||||
|
case compatibility.largeDrive():
|
||||||
|
return warning.largeDriveSize();
|
||||||
|
case compatibility.system():
|
||||||
|
return warning.systemDrive();
|
||||||
|
case compatibility.tooSmall():
|
||||||
|
const size =
|
||||||
|
this.state.image?.recommendedDriveSize || this.state.image?.size || 0;
|
||||||
|
return warning.tooSmall({ size }, drive);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private renderStatuses(drive: DrivelistDrive) {
|
||||||
|
const statuses: DriveStatus[] = getDriveImageCompatibilityStatuses(
|
||||||
|
drive,
|
||||||
|
this.state.image,
|
||||||
|
this.props.write,
|
||||||
|
).slice(0, 2);
|
||||||
|
return (
|
||||||
|
// the column render fn expects a single Element
|
||||||
|
<>
|
||||||
|
{statuses.map((status) => {
|
||||||
|
const badgeShade = badgeShadeFromStatus(status.message);
|
||||||
|
const warningMessage = this.warningFromStatus(status.message, {
|
||||||
|
device: drive.device,
|
||||||
|
size: drive.size || 0,
|
||||||
|
});
|
||||||
|
return (
|
||||||
|
<Badge
|
||||||
|
key={status.message}
|
||||||
|
shade={badgeShade}
|
||||||
|
mr="8px"
|
||||||
|
tooltip={this.props.showWarnings ? warningMessage : ''}
|
||||||
|
>
|
||||||
|
{status.message}
|
||||||
|
</Badge>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private installMissingDrivers(drive: DriverlessDrive) {
|
||||||
|
if (drive.link) {
|
||||||
|
logEvent('Open driver link modal', {
|
||||||
|
url: drive.link,
|
||||||
|
});
|
||||||
|
this.setState({ missingDriversModal: { drive } });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
this.unsubscribe = store.subscribe(() => {
|
||||||
|
const drives = getDrives();
|
||||||
|
const image = getImage();
|
||||||
|
this.setState({
|
||||||
|
drives,
|
||||||
|
image,
|
||||||
|
selectedList:
|
||||||
|
(this.props.updateSelectedList && this.props.updateSelectedList()) ||
|
||||||
|
[],
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
this.unsubscribe?.();
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { cancel, done, ...props } = this.props;
|
||||||
|
const { selectedList, drives, image, missingDriversModal } = this.state;
|
||||||
|
|
||||||
|
const displayedDrives = this.getDisplayedDrives(drives);
|
||||||
|
const disabledDrives = this.getDisabledDrives(drives, image);
|
||||||
|
const numberOfSystemDrives = drives.filter(isSystemDrive).length;
|
||||||
|
const numberOfDisplayedSystemDrives =
|
||||||
|
displayedDrives.filter(isSystemDrive).length;
|
||||||
|
const numberOfHiddenSystemDrives =
|
||||||
|
numberOfSystemDrives - numberOfDisplayedSystemDrives;
|
||||||
|
const hasSystemDrives = selectedList.filter(isSystemDrive).length;
|
||||||
|
const showWarnings = this.props.showWarnings && hasSystemDrives;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
titleElement={
|
||||||
|
<Flex alignItems="baseline" mb={18}>
|
||||||
|
<Txt fontSize={24} align="left">
|
||||||
|
{this.props.titleLabel}
|
||||||
|
</Txt>
|
||||||
|
<Txt
|
||||||
|
fontSize={11}
|
||||||
|
ml={12}
|
||||||
|
color="#5b82a7"
|
||||||
|
style={{ fontWeight: 600 }}
|
||||||
|
>
|
||||||
|
{i18next.t('drives.find', { length: drives.length })}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
}
|
||||||
|
titleDetails={<Txt fontSize={11}>{getDrives().length} found</Txt>}
|
||||||
|
cancel={() => cancel(this.originalList)}
|
||||||
|
done={() => done(selectedList)}
|
||||||
|
action={i18next.t('drives.select', { select: selectedList.length })}
|
||||||
|
primaryButtonProps={{
|
||||||
|
primary: !showWarnings,
|
||||||
|
warning: showWarnings,
|
||||||
|
disabled: !hasAvailableDrives(),
|
||||||
|
}}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{!hasAvailableDrives() ? (
|
||||||
|
<Flex
|
||||||
|
flexDirection="column"
|
||||||
|
justifyContent="center"
|
||||||
|
alignItems="center"
|
||||||
|
width="100%"
|
||||||
|
>
|
||||||
|
{this.props.emptyListIcon}
|
||||||
|
<b>{this.props.emptyListLabel}</b>
|
||||||
|
</Flex>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<DrivesTable
|
||||||
|
refFn={(t) => {
|
||||||
|
if (t !== null) {
|
||||||
|
t.setRowSelection(selectedList);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
checkedRowsNumber={selectedList.length}
|
||||||
|
multipleSelection={this.props.multipleSelection}
|
||||||
|
columns={this.tableColumns}
|
||||||
|
data={displayedDrives}
|
||||||
|
disabledRows={disabledDrives}
|
||||||
|
getRowClass={(row: Drive) =>
|
||||||
|
isDrivelistDrive(row) && row.isSystem ? ['system'] : []
|
||||||
|
}
|
||||||
|
rowKey="displayName"
|
||||||
|
onCheck={(rows: Drive[]) => {
|
||||||
|
let newSelection = rows.filter(isDrivelistDrive);
|
||||||
|
if (this.props.multipleSelection) {
|
||||||
|
if (rows.length === 0) {
|
||||||
|
newSelection = [];
|
||||||
|
}
|
||||||
|
const deselecting = selectedList.filter(
|
||||||
|
(selected) =>
|
||||||
|
newSelection.filter(
|
||||||
|
(row) => row.device === selected.device,
|
||||||
|
).length === 0,
|
||||||
|
);
|
||||||
|
const selecting = newSelection.filter(
|
||||||
|
(row) =>
|
||||||
|
selectedList.filter(
|
||||||
|
(selected) => row.device === selected.device,
|
||||||
|
).length === 0,
|
||||||
|
);
|
||||||
|
deselecting.concat(selecting).forEach((row) => {
|
||||||
|
if (this.props.onSelect) {
|
||||||
|
this.props.onSelect(row);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.setState({
|
||||||
|
selectedList: newSelection,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.props.onSelect) {
|
||||||
|
this.props.onSelect(newSelection[newSelection.length - 1]);
|
||||||
|
}
|
||||||
|
this.setState({
|
||||||
|
selectedList: newSelection.slice(newSelection.length - 1),
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
onRowClick={(row: Drive) => {
|
||||||
|
if (
|
||||||
|
!isDrivelistDrive(row) ||
|
||||||
|
this.driveShouldBeDisabled(row, image)
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.props.onSelect) {
|
||||||
|
this.props.onSelect(row);
|
||||||
|
}
|
||||||
|
const index = selectedList.findIndex(
|
||||||
|
(d) => d.device === row.device,
|
||||||
|
);
|
||||||
|
const newList = this.props.multipleSelection
|
||||||
|
? [...selectedList]
|
||||||
|
: [];
|
||||||
|
if (index === -1) {
|
||||||
|
newList.push(row);
|
||||||
|
} else {
|
||||||
|
// Deselect if selected
|
||||||
|
newList.splice(index, 1);
|
||||||
|
}
|
||||||
|
this.setState({
|
||||||
|
selectedList: newList,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
{numberOfHiddenSystemDrives > 0 && (
|
||||||
|
<Link
|
||||||
|
mt={15}
|
||||||
|
mb={15}
|
||||||
|
fontSize="14px"
|
||||||
|
onClick={() => this.setState({ showSystemDrives: true })}
|
||||||
|
>
|
||||||
|
<Flex alignItems="center">
|
||||||
|
<ChevronDownSvg height="1em" fill="currentColor" />
|
||||||
|
<Txt ml={8}>
|
||||||
|
{i18next.t('drives.showHidden', {
|
||||||
|
num: numberOfHiddenSystemDrives,
|
||||||
|
})}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{this.props.showWarnings && hasSystemDrives ? (
|
||||||
|
<Alert className="system-drive-alert" style={{ width: '67%' }}>
|
||||||
|
{i18next.t('drives.systemDriveDanger')}
|
||||||
|
</Alert>
|
||||||
|
) : null}
|
||||||
|
|
||||||
|
{missingDriversModal.drive !== undefined && (
|
||||||
|
<Modal
|
||||||
|
width={400}
|
||||||
|
title={missingDriversModal.drive.linkTitle}
|
||||||
|
cancel={() => this.setState({ missingDriversModal: {} })}
|
||||||
|
done={() => {
|
||||||
|
try {
|
||||||
|
if (missingDriversModal.drive !== undefined) {
|
||||||
|
openExternal(missingDriversModal.drive.link);
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
logException(error);
|
||||||
|
} finally {
|
||||||
|
this.setState({ missingDriversModal: {} });
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
action={i18next.t('yesContinue')}
|
||||||
|
cancelButtonProps={{
|
||||||
|
children: i18next.t('cancel'),
|
||||||
|
}}
|
||||||
|
children={
|
||||||
|
missingDriversModal.drive.linkMessage ||
|
||||||
|
i18next.t('drives.openInBrowser', {
|
||||||
|
link: missingDriversModal.drive.link,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2019 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.TargetSelector
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.TargetSelector'
|
|
||||||
const SelectTargetButton = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
SelectTargetButton.component(
|
|
||||||
'targetSelector',
|
|
||||||
react2angular(require('./target-selector.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
module.exports = function (ModalService, $q) {
|
|
||||||
let modal = null
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Open the drive selector widget
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @fulfil {(Object|Undefined)} - selected drive
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorService.open().then((drive) => {
|
|
||||||
* console.log(drive);
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.open = () => {
|
|
||||||
modal = ModalService.open({
|
|
||||||
name: 'drive-selector',
|
|
||||||
template: require('../templates/drive-selector-modal.tpl.html'),
|
|
||||||
controller: 'DriveSelectorController as modal',
|
|
||||||
size: 'drive-selector-modal'
|
|
||||||
})
|
|
||||||
|
|
||||||
return modal.result
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Close the drive selector widget
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @fulfil {Undefined}
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorService.close();
|
|
||||||
*/
|
|
||||||
this.close = () => {
|
|
||||||
if (modal) {
|
|
||||||
return modal.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Resolve `undefined` if the modal
|
|
||||||
// was already closed for consistency
|
|
||||||
return $q.resolve()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,110 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
.modal-drive-selector-modal .modal-content {
|
|
||||||
width: 315px;
|
|
||||||
height: 320px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-drive-selector-modal .modal-body {
|
|
||||||
padding-top: 0;
|
|
||||||
padding-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-drive-selector-modal .list-group-item[disabled] {
|
|
||||||
cursor: not-allowed;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-drive-selector-modal {
|
|
||||||
|
|
||||||
.list-group-item-footer:has(span) {
|
|
||||||
margin-top: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-group-item-heading,
|
|
||||||
.list-group-item-text {
|
|
||||||
word-break: break-all;
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-group {
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-group-item {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
border-left: 0;
|
|
||||||
border-right: 0;
|
|
||||||
border-radius: 0;
|
|
||||||
border-color: darken($palette-theme-light-background, 7%);
|
|
||||||
padding: 12px 0;
|
|
||||||
|
|
||||||
.list-group-item-section-expanded {
|
|
||||||
flex-grow: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-group-item-section + .list-group-item-section {
|
|
||||||
margin-left: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
> .tick {
|
|
||||||
font-size: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:first-child {
|
|
||||||
border-top: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
&[disabled] .list-group-item-heading {
|
|
||||||
color: $palette-theme-light-soft-foreground;
|
|
||||||
}
|
|
||||||
|
|
||||||
progress {
|
|
||||||
appearance: none;
|
|
||||||
width: 100%;
|
|
||||||
height: 2.5px;
|
|
||||||
border: none;
|
|
||||||
border-radius: 50% 50%;
|
|
||||||
}
|
|
||||||
|
|
||||||
progress::-webkit-progress-bar {
|
|
||||||
background-color: $palette-theme-default-background;
|
|
||||||
border: none;
|
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
progress::-webkit-progress-value {
|
|
||||||
border-bottom: 1px solid darken($palette-theme-primary-background, 15);
|
|
||||||
background-color: $palette-theme-primary-background;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-group-item-heading {
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.list-group-item-text {
|
|
||||||
line-height: 1;
|
|
||||||
font-size: 11px;
|
|
||||||
color: $palette-theme-light-soft-foreground;
|
|
||||||
}
|
|
||||||
|
|
||||||
.word-keep {
|
|
||||||
word-break: keep-all;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,166 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2019 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* eslint-disable no-magic-numbers */
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
// eslint-disable-next-line no-unused-vars
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const { default: styled } = require('styled-components')
|
|
||||||
const {
|
|
||||||
ChangeButton,
|
|
||||||
DetailsText,
|
|
||||||
StepButton,
|
|
||||||
StepNameButton,
|
|
||||||
ThemedProvider
|
|
||||||
} = require('./../../styled-components')
|
|
||||||
const { Txt } = require('rendition')
|
|
||||||
const middleEllipsis = require('./../../utils/middle-ellipsis')
|
|
||||||
const { bytesToClosestUnit } = require('./../../../../shared/units')
|
|
||||||
|
|
||||||
const TargetDetail = styled((props) => (
|
|
||||||
<Txt.span {...props}>
|
|
||||||
</Txt.span>
|
|
||||||
)) `
|
|
||||||
float: ${({ float }) => float}
|
|
||||||
`
|
|
||||||
|
|
||||||
const TargetDisplayText = ({
|
|
||||||
description,
|
|
||||||
size,
|
|
||||||
...props
|
|
||||||
}) => {
|
|
||||||
return (
|
|
||||||
<Txt.span {...props}>
|
|
||||||
<TargetDetail
|
|
||||||
float='left'>
|
|
||||||
{description}
|
|
||||||
</TargetDetail>
|
|
||||||
<TargetDetail
|
|
||||||
float='right'
|
|
||||||
>
|
|
||||||
{size}
|
|
||||||
</TargetDetail>
|
|
||||||
</Txt.span>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
const TargetSelector = (props) => {
|
|
||||||
const targets = props.selection.getSelectedDrives()
|
|
||||||
|
|
||||||
if (targets.length === 1) {
|
|
||||||
const target = targets[0]
|
|
||||||
return (
|
|
||||||
<ThemedProvider>
|
|
||||||
<StepNameButton
|
|
||||||
plain
|
|
||||||
tooltip={props.tooltip}
|
|
||||||
>
|
|
||||||
{/* eslint-disable no-magic-numbers */}
|
|
||||||
{ middleEllipsis(target.description, 20) }
|
|
||||||
</StepNameButton>
|
|
||||||
{ !props.flashing &&
|
|
||||||
<ChangeButton
|
|
||||||
plain
|
|
||||||
mb={14}
|
|
||||||
onClick={props.reselectDrive}
|
|
||||||
>
|
|
||||||
Change
|
|
||||||
</ChangeButton>
|
|
||||||
}
|
|
||||||
<DetailsText>
|
|
||||||
{ props.constraints.hasListDriveImageCompatibilityStatus(targets, props.image) &&
|
|
||||||
<Txt.span className='glyphicon glyphicon-exclamation-sign'
|
|
||||||
ml={2}
|
|
||||||
tooltip={
|
|
||||||
props.constraints.getListDriveImageCompatibilityStatuses(targets, props.image)[0].message
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
{ bytesToClosestUnit(target.size) }
|
|
||||||
</DetailsText>
|
|
||||||
</ThemedProvider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (targets.length > 1) {
|
|
||||||
const targetsTemplate = []
|
|
||||||
for (const target of targets) {
|
|
||||||
targetsTemplate.push((
|
|
||||||
<DetailsText
|
|
||||||
key={target.device}
|
|
||||||
tooltip={
|
|
||||||
`${target.description} ${target.displayName} ${bytesToClosestUnit(target.size)}`
|
|
||||||
}
|
|
||||||
px={21}
|
|
||||||
>
|
|
||||||
<TargetDisplayText
|
|
||||||
description={middleEllipsis(target.description, 14)}
|
|
||||||
size={bytesToClosestUnit(target.size)}
|
|
||||||
>
|
|
||||||
</TargetDisplayText>
|
|
||||||
</DetailsText>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<ThemedProvider>
|
|
||||||
<StepNameButton
|
|
||||||
plain
|
|
||||||
tooltip={props.tooltip}
|
|
||||||
>
|
|
||||||
{targets.length} Targets
|
|
||||||
</StepNameButton>
|
|
||||||
{ !props.flashing &&
|
|
||||||
<ChangeButton
|
|
||||||
plain
|
|
||||||
onClick={props.reselectDrive}
|
|
||||||
mb={14}
|
|
||||||
>
|
|
||||||
Change
|
|
||||||
</ChangeButton>
|
|
||||||
}
|
|
||||||
{targetsTemplate}
|
|
||||||
</ThemedProvider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ThemedProvider>
|
|
||||||
<StepButton
|
|
||||||
tabindex={(targets.length > 0) ? -1 : 2 }
|
|
||||||
disabled={props.disabled}
|
|
||||||
onClick={props.openDriveSelector}
|
|
||||||
>
|
|
||||||
Select target
|
|
||||||
</StepButton>
|
|
||||||
</ThemedProvider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
TargetSelector.propTypes = {
|
|
||||||
disabled: propTypes.bool,
|
|
||||||
openDriveSelector: propTypes.func,
|
|
||||||
selection: propTypes.object,
|
|
||||||
reselectDrive: propTypes.func,
|
|
||||||
flashing: propTypes.bool,
|
|
||||||
constraints: propTypes.object,
|
|
||||||
show: propTypes.bool,
|
|
||||||
tooltip: propTypes.string
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = TargetSelector
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
<div class="modal-header">
|
|
||||||
<h4 class="modal-title">Select a Drive</h4>
|
|
||||||
<button tabindex="14" class="close" ng-click="modal.closeModal()">×</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-body">
|
|
||||||
<ul class="list-group">
|
|
||||||
<li class="list-group-item" ng-repeat="drive in modal.getDrives() track by drive.device"
|
|
||||||
ng-disabled="!modal.constraints.isDriveValid(drive, modal.state.getImage())"
|
|
||||||
ng-dblclick="modal.selectDriveAndClose(drive)"
|
|
||||||
ng-click="modal.toggleDrive(drive)">
|
|
||||||
<img class="list-group-item-section" alt="Drive device type logo"
|
|
||||||
ng-if="drive.icon"
|
|
||||||
ng-src="../assets/{{drive.icon}}.svg"
|
|
||||||
width="25"
|
|
||||||
height="30">
|
|
||||||
<div
|
|
||||||
class="list-group-item-section list-group-item-section-expanded"
|
|
||||||
tabindex="{{ 15 + $index }}"
|
|
||||||
ng-keypress="modal.keyboardToggleDrive(drive, $event)">
|
|
||||||
|
|
||||||
<h4 class="list-group-item-heading">{{ drive.description }}
|
|
||||||
<span class="word-keep"
|
|
||||||
ng-show="drive.size"> - {{ drive.size | closestUnit }}</span>
|
|
||||||
</h4>
|
|
||||||
<p class="list-group-item-text" ng-if="!drive.link">{{ drive.displayName }}</p>
|
|
||||||
<p class="list-group-item-text" ng-if="drive.link">{{ drive.displayName }} - <b><a ng-click="modal.installMissingDrivers(drive)">{{ drive.linkCTA }}</a></b></p>
|
|
||||||
|
|
||||||
<footer class="list-group-item-footer">
|
|
||||||
|
|
||||||
<span class="label" ng-repeat="status in modal.getDriveStatuses(drive)"
|
|
||||||
ng-class="{
|
|
||||||
'label-warning': status.type === modal.constraints.COMPATIBILITY_STATUS_TYPES.WARNING,
|
|
||||||
'label-danger': status.type === modal.constraints.COMPATIBILITY_STATUS_TYPES.ERROR
|
|
||||||
}">{{ status.message }}</span>
|
|
||||||
|
|
||||||
</footer>
|
|
||||||
<progress ng-if="drive.progress" value="{{ drive.progress }}" max="100"></progress>
|
|
||||||
</div>
|
|
||||||
<span class="list-group-item-section tick tick--success"
|
|
||||||
ng-show="modal.constraints.isDriveValid(drive, modal.state.getImage())"
|
|
||||||
ng-disabled="!modal.state.isDriveSelected(drive.device)"></span>
|
|
||||||
</li>
|
|
||||||
<li class="list-group-item"
|
|
||||||
ng-show="!modal.drives.hasAvailableDrives()">
|
|
||||||
<div>
|
|
||||||
<b>Connect a drive!</b>
|
|
||||||
<div>No removable drive detected.</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-footer">
|
|
||||||
<button class="button button-primary"
|
|
||||||
tabindex="{{ 15 + modal.getDrives().length }}"
|
|
||||||
ng-class="{
|
|
||||||
'button-warning': modal.constraints.hasListDriveImageCompatibilityStatus(modal.state.getSelectedDrives(), modal.state.getImage())
|
|
||||||
}"
|
|
||||||
ng-click="modal.closeModal()"
|
|
||||||
ng-disabled="!modal.state.hasDrive()">Continue</button>
|
|
||||||
</div>
|
|
||||||
@@ -0,0 +1,83 @@
|
|||||||
|
import ExclamationTriangleSvg from '@fortawesome/fontawesome-free/svgs/solid/exclamation-triangle.svg';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Badge, Flex, Txt, ModalProps } from 'rendition';
|
||||||
|
import { Modal, ScrollableFlex } from '../../styled-components';
|
||||||
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
||||||
|
|
||||||
|
import * as prettyBytes from 'pretty-bytes';
|
||||||
|
import { DriveWithWarnings } from '../../pages/main/Flash';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
const DriveStatusWarningModal = ({
|
||||||
|
done,
|
||||||
|
cancel,
|
||||||
|
isSystem,
|
||||||
|
drivesWithWarnings,
|
||||||
|
}: ModalProps & {
|
||||||
|
isSystem: boolean;
|
||||||
|
drivesWithWarnings: DriveWithWarnings[];
|
||||||
|
}) => {
|
||||||
|
let warningSubtitle = i18next.t('drives.largeDriveWarning');
|
||||||
|
let warningCta = i18next.t('drives.largeDriveWarningMsg');
|
||||||
|
|
||||||
|
if (isSystem) {
|
||||||
|
warningSubtitle = i18next.t('drives.systemDriveWarning');
|
||||||
|
warningCta = i18next.t('drives.systemDriveWarningMsg');
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
footerShadow={false}
|
||||||
|
reverseFooterButtons={true}
|
||||||
|
done={done}
|
||||||
|
cancel={cancel}
|
||||||
|
cancelButtonProps={{
|
||||||
|
primary: false,
|
||||||
|
warning: true,
|
||||||
|
children: i18next.t('drives.changeTarget'),
|
||||||
|
}}
|
||||||
|
action={i18next.t('sure')}
|
||||||
|
primaryButtonProps={{
|
||||||
|
primary: false,
|
||||||
|
outline: true,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Flex
|
||||||
|
flexDirection="column"
|
||||||
|
alignItems="center"
|
||||||
|
justifyContent="center"
|
||||||
|
width="100%"
|
||||||
|
>
|
||||||
|
<Flex flexDirection="column">
|
||||||
|
<ExclamationTriangleSvg height="2em" fill="#fca321" />
|
||||||
|
<Txt fontSize="24px" color="#fca321">
|
||||||
|
{i18next.t('warning')}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
<Txt fontSize="24px">{warningSubtitle}</Txt>
|
||||||
|
<ScrollableFlex
|
||||||
|
flexDirection="column"
|
||||||
|
backgroundColor="#fff5e6"
|
||||||
|
m="2em 0"
|
||||||
|
p="1em 2em"
|
||||||
|
width="420px"
|
||||||
|
maxHeight="100px"
|
||||||
|
>
|
||||||
|
{drivesWithWarnings.map((drive, i, array) => (
|
||||||
|
<>
|
||||||
|
<Flex justifyContent="space-between" alignItems="baseline">
|
||||||
|
<strong>{middleEllipsis(drive.description, 28)}</strong>{' '}
|
||||||
|
{drive.size && prettyBytes(drive.size) + ' '}
|
||||||
|
<Badge shade={5}>{drive.statuses[0].message}</Badge>
|
||||||
|
</Flex>
|
||||||
|
{i !== array.length - 1 ? <hr style={{ width: '100%' }} /> : null}
|
||||||
|
</>
|
||||||
|
))}
|
||||||
|
</ScrollableFlex>
|
||||||
|
<Txt style={{ fontWeight: 600 }}>{warningCta}</Txt>
|
||||||
|
</Flex>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DriveStatusWarningModal;
|
||||||
@@ -1,57 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const SafeWebview = require('../safe-webview/safe-webview.jsx')
|
|
||||||
const settings = require('../../models/settings')
|
|
||||||
const analytics = require('../../modules/analytics')
|
|
||||||
|
|
||||||
class FeaturedProject extends React.Component {
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
|
|
||||||
this.state = {
|
|
||||||
endpoint: null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount () {
|
|
||||||
return settings.load()
|
|
||||||
.then(() => {
|
|
||||||
const endpoint = settings.get('featuredProjectEndpoint') || 'https://assets.balena.io/etcher-featured/index.html'
|
|
||||||
this.setState({ endpoint })
|
|
||||||
})
|
|
||||||
.catch(analytics.logException)
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (this.state.endpoint) ? (
|
|
||||||
<SafeWebview
|
|
||||||
src={this.state.endpoint}
|
|
||||||
{...this.props}>
|
|
||||||
</SafeWebview>
|
|
||||||
) : null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
FeaturedProject.propTypes = {
|
|
||||||
onWebviewShow: propTypes.func
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = FeaturedProject
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.FeaturedProject
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.FeaturedProject'
|
|
||||||
const FeaturedProject = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
FeaturedProject.component(
|
|
||||||
'featuredProject',
|
|
||||||
react2angular(require('./featured-project.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,72 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
const os = require('os')
|
|
||||||
const settings = require('../../../models/settings')
|
|
||||||
const utils = require('../../../../../shared/utils')
|
|
||||||
const angular = require('angular')
|
|
||||||
|
|
||||||
/* eslint-disable lodash/prefer-lodash-method */
|
|
||||||
|
|
||||||
module.exports = function (
|
|
||||||
$uibModalInstance
|
|
||||||
) {
|
|
||||||
/**
|
|
||||||
* @summary Close the modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* FileSelectorController.close();
|
|
||||||
*/
|
|
||||||
this.close = () => {
|
|
||||||
$uibModalInstance.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Folder to constrain the file picker to
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @returns {String} - folder to constrain by
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* FileSelectorController.getFolderConstraint()
|
|
||||||
*/
|
|
||||||
this.getFolderConstraint = utils.memoize(() => {
|
|
||||||
return settings.has('fileBrowserConstraintPath')
|
|
||||||
? settings.get('fileBrowserConstraintPath')
|
|
||||||
: ''
|
|
||||||
}, angular.equals)
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Get initial path
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @returns {String} - path
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* <file-selector path="FileSelectorController.getPath()"></file-selector>
|
|
||||||
*/
|
|
||||||
this.getPath = () => {
|
|
||||||
const constraintFolderPath = this.getFolderConstraint()
|
|
||||||
return _.isEmpty(constraintFolderPath) ? os.homedir() : constraintFolderPath
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,321 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const rendition = require('rendition')
|
|
||||||
const colors = require('./colors')
|
|
||||||
|
|
||||||
const prettyBytes = require('pretty-bytes')
|
|
||||||
const files = require('../../../models/files')
|
|
||||||
const middleEllipsis = require('../../../utils/middle-ellipsis')
|
|
||||||
const supportedFormats = require('../../../../../shared/supported-formats')
|
|
||||||
|
|
||||||
const debug = require('debug')('etcher:gui:file-selector')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Character limit of a filename before a middle-ellipsis is added
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
const FILENAME_CHAR_LIMIT = 20
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Pattern to match all supported formats for highlighting
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
const SUPPORTED_FORMATS_PATTERN = new RegExp(`^\\.(${supportedFormats.getAllExtensions().join('|')})$`, 'i')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Flex styled component
|
|
||||||
* @function
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
const Flex = styled.div`
|
|
||||||
display: flex;
|
|
||||||
flex: ${ props => props.flex };
|
|
||||||
flex-direction: ${ props => props.direction };
|
|
||||||
justify-content: ${ props => props.justifyContent };
|
|
||||||
align-items: ${ props => props.alignItems };
|
|
||||||
flex-wrap: ${ props => props.wrap };
|
|
||||||
flex-grow: ${ props => props.grow };
|
|
||||||
`
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Anchor flex styled component
|
|
||||||
* @function
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
const ClickableFlex = styled.a`
|
|
||||||
display: flex;
|
|
||||||
flex: ${ props => props.flex };
|
|
||||||
flex-direction: ${ props => props.direction };
|
|
||||||
justify-content: ${ props => props.justifyContent };
|
|
||||||
align-items: ${ props => props.alignItems };
|
|
||||||
flex-wrap: ${ props => props.wrap };
|
|
||||||
flex-grow: ${ props => props.grow };
|
|
||||||
`
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary FileList scroll wrapper element
|
|
||||||
* @class
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
class UnstyledFileListWrap extends React.PureComponent {
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
this.scrollElem = null
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<Flex className={ this.props.className }
|
|
||||||
ref={ ::this.setScrollElem }
|
|
||||||
wrap="wrap">
|
|
||||||
{ this.props.children }
|
|
||||||
</Flex>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
setScrollElem (element) {
|
|
||||||
this.scrollElem = element
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidUpdate (prevProps) {
|
|
||||||
if (this.scrollElem) {
|
|
||||||
this.scrollElem.scrollTop = 0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary FileList scroll wrapper element
|
|
||||||
* @class
|
|
||||||
* @type {StyledComponent}
|
|
||||||
*/
|
|
||||||
const FileListWrap = styled(UnstyledFileListWrap)`
|
|
||||||
overflow-x: hidden;
|
|
||||||
overflow-y: auto;
|
|
||||||
padding: 0 20px;
|
|
||||||
`
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary File element
|
|
||||||
* @class
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
class UnstyledFile extends React.PureComponent {
|
|
||||||
|
|
||||||
static getFileIconClass (file) {
|
|
||||||
return file.isDirectory
|
|
||||||
? 'fas fa-folder'
|
|
||||||
: 'fas fa-file-alt'
|
|
||||||
}
|
|
||||||
|
|
||||||
onHighlight (event) {
|
|
||||||
event.preventDefault()
|
|
||||||
this.props.onHighlight(this.props.file)
|
|
||||||
}
|
|
||||||
|
|
||||||
onSelect (event) {
|
|
||||||
event.preventDefault()
|
|
||||||
this.props.onSelect(this.props.file)
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
const file = this.props.file
|
|
||||||
return (
|
|
||||||
<ClickableFlex
|
|
||||||
data-path={ file.path }
|
|
||||||
href={ `file://${file.path}` }
|
|
||||||
direction="column"
|
|
||||||
alignItems="stretch"
|
|
||||||
className={ this.props.className }
|
|
||||||
onClick={ ::this.onHighlight }
|
|
||||||
onDoubleClick={ ::this.onSelect }>
|
|
||||||
<span className={ UnstyledFile.getFileIconClass(file) } />
|
|
||||||
<span>{ middleEllipsis(file.basename, FILENAME_CHAR_LIMIT) }</span>
|
|
||||||
<div>{ file.isDirectory ? '' : prettyBytes(file.size || 0) }</div>
|
|
||||||
</ClickableFlex>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary File element
|
|
||||||
* @class
|
|
||||||
* @type {StyledComponent}
|
|
||||||
*/
|
|
||||||
const File = styled(UnstyledFile)`
|
|
||||||
width: 100px;
|
|
||||||
min-height: 100px;
|
|
||||||
max-height: 128px;
|
|
||||||
margin: 5px 10px;
|
|
||||||
padding: 5px;
|
|
||||||
background-color: none;
|
|
||||||
transition: 0.05s background-color ease-out;
|
|
||||||
color: ${ colors.primary.color };
|
|
||||||
cursor: pointer;
|
|
||||||
border-radius: 5px;
|
|
||||||
word-break: break-word;
|
|
||||||
|
|
||||||
> span:first-of-type {
|
|
||||||
align-self: center;
|
|
||||||
line-height: 1;
|
|
||||||
margin-bottom: 6px;
|
|
||||||
font-size: 48px;
|
|
||||||
color: ${ props => props.disabled ? colors.primary.faded : colors.soft.color };
|
|
||||||
}
|
|
||||||
|
|
||||||
> span:last-of-type {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
text-align: center;
|
|
||||||
font-size: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
> div:last-child {
|
|
||||||
background-color: none;
|
|
||||||
color: ${ colors.primary.subColor };
|
|
||||||
text-align: center;
|
|
||||||
font-size: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
:hover, :visited {
|
|
||||||
color: ${ colors.primary.color };
|
|
||||||
}
|
|
||||||
|
|
||||||
:focus,
|
|
||||||
:active {
|
|
||||||
color: ${ colors.highlight.color };
|
|
||||||
background-color: ${ colors.highlight.background };
|
|
||||||
}
|
|
||||||
|
|
||||||
:focus > span:first-of-type,
|
|
||||||
:active > span:first-of-type {
|
|
||||||
color: ${ colors.highlight.color };
|
|
||||||
}
|
|
||||||
|
|
||||||
:focus > div:last-child,
|
|
||||||
:active > div:last-child {
|
|
||||||
color: ${ colors.highlight.color };
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary FileList element
|
|
||||||
* @class
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
class FileList extends React.Component {
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
|
|
||||||
this.state = {
|
|
||||||
path: props.path,
|
|
||||||
highlighted: null,
|
|
||||||
files: [],
|
|
||||||
}
|
|
||||||
|
|
||||||
debug('FileList', props)
|
|
||||||
}
|
|
||||||
|
|
||||||
readdir (dirname) {
|
|
||||||
debug('FileList:readdir', dirname)
|
|
||||||
|
|
||||||
if (this.props.constraintPath && dirname === '/') {
|
|
||||||
if (this.props.constraint) {
|
|
||||||
const mountpoints = this.props.constraint.mountpoints.map(( mount ) => {
|
|
||||||
const entry = new files.FileEntry(mount.path, {
|
|
||||||
size: 0,
|
|
||||||
isFile: () => false,
|
|
||||||
isDirectory: () => true
|
|
||||||
})
|
|
||||||
entry.name = mount.label
|
|
||||||
return entry
|
|
||||||
})
|
|
||||||
debug('FileList:readdir', mountpoints)
|
|
||||||
window.requestAnimationFrame(() => {
|
|
||||||
this.setState({ files: mountpoints })
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
files.readdirAsync(dirname).then((files) => {
|
|
||||||
window.requestAnimationFrame(() => {
|
|
||||||
this.setState({ files: files })
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount () {
|
|
||||||
process.nextTick(() => {
|
|
||||||
this.readdir(this.state.path)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
onHighlight (file) {
|
|
||||||
debug('FileList:onHighlight', file)
|
|
||||||
this.props.onHighlight(file)
|
|
||||||
}
|
|
||||||
|
|
||||||
onSelect (file) {
|
|
||||||
debug('FileList:onSelect', file.path, file.isDirectory)
|
|
||||||
this.props.onSelect(file)
|
|
||||||
}
|
|
||||||
|
|
||||||
shouldComponentUpdate (nextProps, nextState) {
|
|
||||||
const shouldUpdate = (this.state.files !== nextState.files)
|
|
||||||
debug('FileList:shouldComponentUpdate', shouldUpdate)
|
|
||||||
if (this.props.path !== nextProps.path || this.props.constraint !== nextProps.constraint) {
|
|
||||||
process.nextTick(() => {
|
|
||||||
this.readdir(nextProps.path)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return shouldUpdate
|
|
||||||
}
|
|
||||||
|
|
||||||
static isSelectable (file) {
|
|
||||||
return file.isDirectory || !file.ext ||
|
|
||||||
SUPPORTED_FORMATS_PATTERN.test(file.ext)
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<FileListWrap wrap="wrap">
|
|
||||||
{
|
|
||||||
this.state.files.map((file) => {
|
|
||||||
return (
|
|
||||||
<File key={ file.path }
|
|
||||||
file={ file }
|
|
||||||
disabled={ !FileList.isSelectable(file) }
|
|
||||||
onSelect={ ::this.onSelect }
|
|
||||||
onHighlight={ ::this.onHighlight }/>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</FileListWrap>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = FileList
|
|
||||||
@@ -1,358 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const path = require('path')
|
|
||||||
const sdk = require('etcher-sdk')
|
|
||||||
|
|
||||||
const Bluebird = require('bluebird')
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const rendition = require('rendition')
|
|
||||||
const colors = require('./colors')
|
|
||||||
|
|
||||||
const Breadcrumbs = require('./path-breadcrumbs')
|
|
||||||
const FileList = require('./file-list')
|
|
||||||
const RecentFiles = require('./recent-files')
|
|
||||||
const files = require('../../../models/files')
|
|
||||||
|
|
||||||
const selectionState = require('../../../models/selection-state')
|
|
||||||
const store = require('../../../models/store')
|
|
||||||
const osDialog = require('../../../os/dialog')
|
|
||||||
const exceptionReporter = require('../../../modules/exception-reporter')
|
|
||||||
const messages = require('../../../../../shared/messages')
|
|
||||||
const errors = require('../../../../../shared/errors')
|
|
||||||
const supportedFormats = require('../../../../../shared/supported-formats')
|
|
||||||
const analytics = require('../../../modules/analytics')
|
|
||||||
|
|
||||||
const debug = require('debug')('etcher:gui:file-selector')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Flex styled component
|
|
||||||
* @function
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
const Flex = styled.div`
|
|
||||||
display: flex;
|
|
||||||
flex: ${ props => props.flex };
|
|
||||||
flex-direction: ${ props => props.direction };
|
|
||||||
justify-content: ${ props => props.justifyContent };
|
|
||||||
align-items: ${ props => props.alignItems };
|
|
||||||
flex-wrap: ${ props => props.wrap };
|
|
||||||
flex-grow: ${ props => props.grow };
|
|
||||||
overflow: ${ props => props.overflow };
|
|
||||||
`
|
|
||||||
|
|
||||||
const Header = styled(Flex) `
|
|
||||||
padding: 10px 15px 0;
|
|
||||||
border-bottom: 1px solid ${ colors.primary.faded };
|
|
||||||
|
|
||||||
> * {
|
|
||||||
margin: 5px;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
const Main = styled(Flex) ``
|
|
||||||
|
|
||||||
const Footer = styled(Flex) `
|
|
||||||
padding: 10px;
|
|
||||||
flex: 0 0 auto;
|
|
||||||
border-top: 1px solid ${ colors.primary.faded };
|
|
||||||
|
|
||||||
> * {
|
|
||||||
margin: 0 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
> button {
|
|
||||||
flex-grow: 0;
|
|
||||||
flex-shrink: 0;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
class UnstyledFilePath extends React.PureComponent {
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<div className={ this.props.className }>
|
|
||||||
<span>{
|
|
||||||
this.props.file && !this.props.file.isDirectory
|
|
||||||
? this.props.file.basename
|
|
||||||
: ''
|
|
||||||
}</span>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const FilePath = styled(UnstyledFilePath)`
|
|
||||||
display: flex;
|
|
||||||
flex-grow: 1;
|
|
||||||
align-items: center;
|
|
||||||
overflow: hidden;
|
|
||||||
|
|
||||||
> span {
|
|
||||||
font-size: 16px;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
class FileSelector extends React.PureComponent {
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
|
|
||||||
this.state = {
|
|
||||||
path: props.path,
|
|
||||||
highlighted: null,
|
|
||||||
constraint: null,
|
|
||||||
files: [],
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount() {
|
|
||||||
if (this.props.constraintpath) {
|
|
||||||
const device = files.getConstraintDevice(this.props.constraintpath)
|
|
||||||
debug('FileSelector:getConstraintDevice', device)
|
|
||||||
if (device !== undefined) {
|
|
||||||
this.setState({ constraint: device.drive })
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
confirmSelection () {
|
|
||||||
if (this.state.highlighted) {
|
|
||||||
this.selectFile(this.state.highlighted)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
close () {
|
|
||||||
this.props.close()
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidUpdate () {
|
|
||||||
debug('FileSelector:componentDidUpdate')
|
|
||||||
}
|
|
||||||
|
|
||||||
containPath (newPath) {
|
|
||||||
if (this.state.constraint) {
|
|
||||||
const isContained = this.state.constraint.mountpoints.some((mount) => {
|
|
||||||
return !path.relative(mount.path, newPath).startsWith('..')
|
|
||||||
})
|
|
||||||
if (!isContained) {
|
|
||||||
return '/'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return newPath
|
|
||||||
}
|
|
||||||
|
|
||||||
navigate (newPath) {
|
|
||||||
debug('FileSelector:navigate', newPath)
|
|
||||||
this.setState({ path: this.containPath(newPath) })
|
|
||||||
}
|
|
||||||
|
|
||||||
navigateUp () {
|
|
||||||
let newPath = this.containPath(path.join(this.state.path, '..'))
|
|
||||||
debug('FileSelector:navigateUp', this.state.path, '->', newPath)
|
|
||||||
this.setState({ path: newPath })
|
|
||||||
}
|
|
||||||
|
|
||||||
selectImage (image) {
|
|
||||||
debug('FileSelector:selectImage', image)
|
|
||||||
|
|
||||||
if (!supportedFormats.isSupportedImage(image.path)) {
|
|
||||||
const invalidImageError = errors.createUserError({
|
|
||||||
title: 'Invalid image',
|
|
||||||
description: messages.error.invalidImage(image.path)
|
|
||||||
})
|
|
||||||
|
|
||||||
osDialog.showError(invalidImageError)
|
|
||||||
analytics.logEvent('Invalid image', {
|
|
||||||
image,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
return Bluebird.resolve()
|
|
||||||
}
|
|
||||||
|
|
||||||
return Bluebird.try(() => {
|
|
||||||
let message = null
|
|
||||||
|
|
||||||
if (supportedFormats.looksLikeWindowsImage(image.path)) {
|
|
||||||
analytics.logEvent('Possibly Windows image', {
|
|
||||||
image,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
message = messages.warning.looksLikeWindowsImage()
|
|
||||||
} else if (!image.hasMBR) {
|
|
||||||
analytics.logEvent('Missing partition table', {
|
|
||||||
image,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
message = messages.warning.missingPartitionTable()
|
|
||||||
}
|
|
||||||
|
|
||||||
if (message) {
|
|
||||||
// TODO: `Continue` should be on a red background (dangerous action) instead of `Change`.
|
|
||||||
// We want `X` to act as `Continue`, that's why `Continue` is the `rejectionLabel`
|
|
||||||
return osDialog.showWarning({
|
|
||||||
confirmationLabel: 'Change',
|
|
||||||
rejectionLabel: 'Continue',
|
|
||||||
title: 'Warning',
|
|
||||||
description: message
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}).then((shouldChange) => {
|
|
||||||
if (shouldChange) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
selectionState.selectImage(image)
|
|
||||||
|
|
||||||
this.close()
|
|
||||||
|
|
||||||
// An easy way so we can quickly identify if we're making use of
|
|
||||||
// certain features without printing pages of text to DevTools.
|
|
||||||
image.logo = Boolean(image.logo)
|
|
||||||
image.blockMap = Boolean(image.blockMap)
|
|
||||||
|
|
||||||
analytics.logEvent('Select image', {
|
|
||||||
image,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
}).catch(exceptionReporter.report)
|
|
||||||
}
|
|
||||||
|
|
||||||
selectFile (file) {
|
|
||||||
debug('FileSelector:selectFile', file)
|
|
||||||
|
|
||||||
if (file.isDirectory) {
|
|
||||||
this.navigate(file.path)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!supportedFormats.isSupportedImage(file.path)) {
|
|
||||||
const invalidImageError = errors.createUserError({
|
|
||||||
title: 'Invalid image',
|
|
||||||
description: messages.error.invalidImage(file.path)
|
|
||||||
})
|
|
||||||
|
|
||||||
osDialog.showError(invalidImageError)
|
|
||||||
analytics.logEvent('Invalid image', { path: file.path })
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
debug('FileSelector:getImageMetadata', file)
|
|
||||||
|
|
||||||
const source = new sdk.sourceDestination.File(file.path, sdk.sourceDestination.File.OpenFlags.Read)
|
|
||||||
source.getInnerSource()
|
|
||||||
.then((innerSource) => {
|
|
||||||
return innerSource.getMetadata()
|
|
||||||
.then((imageMetadata) => {
|
|
||||||
debug('FileSelector:getImageMetadata', imageMetadata)
|
|
||||||
imageMetadata.path = file.path
|
|
||||||
imageMetadata.extension = path.extname(file.path).slice(1)
|
|
||||||
return innerSource.getPartitionTable()
|
|
||||||
.then((partitionTable) => {
|
|
||||||
if (partitionTable !== undefined) {
|
|
||||||
imageMetadata.hasMBR = true
|
|
||||||
imageMetadata.partitions = partitionTable.partitions
|
|
||||||
}
|
|
||||||
return this.selectImage(imageMetadata)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.catch((error) => {
|
|
||||||
debug('FileSelector:getImageMetadata', error)
|
|
||||||
const imageError = errors.createUserError({
|
|
||||||
title: 'Error opening image',
|
|
||||||
description: messages.error.openImage(path.basename(file.path), error.message)
|
|
||||||
})
|
|
||||||
|
|
||||||
osDialog.showError(imageError)
|
|
||||||
analytics.logException(error)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
onHighlight (file) {
|
|
||||||
this.setState({ highlighted: file })
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
const styles = {
|
|
||||||
display: 'flex',
|
|
||||||
height: 'calc(100vh - 20px)',
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<rendition.Provider style={ styles }>
|
|
||||||
{/*<RecentFiles flex="0 0 auto"
|
|
||||||
selectFile={ ::this.selectFile }
|
|
||||||
navigate={ ::this.navigate } />*/}
|
|
||||||
<Flex direction="column" grow="1" overflow="auto">
|
|
||||||
<Header flex="0 0 auto" alignItems="baseline">
|
|
||||||
<rendition.Button
|
|
||||||
bg={ colors.secondary.background }
|
|
||||||
color={ colors.primary.color }
|
|
||||||
onClick={ ::this.navigateUp }>
|
|
||||||
<span className="fas fa-angle-left" />
|
|
||||||
Back
|
|
||||||
</rendition.Button>
|
|
||||||
<span className="fas fa-hdd" />
|
|
||||||
<Breadcrumbs
|
|
||||||
path={ this.state.path }
|
|
||||||
navigate={ ::this.navigate }
|
|
||||||
constraintPath={ this.props.constraintpath }
|
|
||||||
constraint={ this.state.constraint }
|
|
||||||
/>
|
|
||||||
</Header>
|
|
||||||
<Main flex="1">
|
|
||||||
<Flex direction="column" grow="1">
|
|
||||||
<FileList path={ this.state.path }
|
|
||||||
constraintPath={ this.props.constraintpath }
|
|
||||||
constraint={ this.state.constraint }
|
|
||||||
onHighlight={ ::this.onHighlight }
|
|
||||||
onSelect={ ::this.selectFile }></FileList>
|
|
||||||
</Flex>
|
|
||||||
</Main>
|
|
||||||
<Footer justifyContent="flex-end">
|
|
||||||
<FilePath file={ this.state.highlighted }></FilePath>
|
|
||||||
<rendition.Button onClick={ ::this.close }>Cancel</rendition.Button>
|
|
||||||
<rendition.Button
|
|
||||||
primary
|
|
||||||
onClick={ ::this.confirmSelection }>
|
|
||||||
Select file
|
|
||||||
</rendition.Button>
|
|
||||||
</Footer>
|
|
||||||
</Flex>
|
|
||||||
</rendition.Provider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
FileSelector.propTypes = {
|
|
||||||
path: propTypes.string,
|
|
||||||
close: propTypes.func,
|
|
||||||
constraintpath: propTypes.string,
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = FileSelector
|
|
||||||
@@ -1,119 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const path = require('path')
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const rendition = require('rendition')
|
|
||||||
|
|
||||||
const middleEllipsis = require('../../../utils/middle-ellipsis')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary How many directories to show with the breadcrumbs
|
|
||||||
* @type {Number}
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
const MAX_DIR_CRUMBS = 3
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Character limit of a filename before a middle-ellipsis is added
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
*/
|
|
||||||
const FILENAME_CHAR_LIMIT_SHORT = 15
|
|
||||||
|
|
||||||
function splitComponents(dirname, root) {
|
|
||||||
const components = []
|
|
||||||
let basename = null
|
|
||||||
root = root || path.parse(dirname).root
|
|
||||||
while( dirname !== root ) {
|
|
||||||
basename = path.basename(dirname)
|
|
||||||
components.unshift({
|
|
||||||
path: dirname,
|
|
||||||
basename: basename,
|
|
||||||
name: basename
|
|
||||||
})
|
|
||||||
dirname = path.join( dirname, '..' )
|
|
||||||
}
|
|
||||||
if (components.length < MAX_DIR_CRUMBS) {
|
|
||||||
components.unshift({
|
|
||||||
path: root,
|
|
||||||
basename: root,
|
|
||||||
name: 'Root'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return components
|
|
||||||
}
|
|
||||||
|
|
||||||
class Crumb extends React.PureComponent {
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<rendition.Button
|
|
||||||
onClick={ ::this.navigate }
|
|
||||||
plain={ true }>
|
|
||||||
<rendition.Txt bold={ this.props.bold }>
|
|
||||||
{ middleEllipsis(this.props.dir.name, FILENAME_CHAR_LIMIT_SHORT) }
|
|
||||||
</rendition.Txt>
|
|
||||||
</rendition.Button>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
navigate () {
|
|
||||||
this.props.navigate(this.props.dir.path)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class UnstyledBreadcrumbs extends React.PureComponent {
|
|
||||||
render () {
|
|
||||||
const components = splitComponents(this.props.path).slice(-MAX_DIR_CRUMBS)
|
|
||||||
return (
|
|
||||||
<div className={ this.props.className }>
|
|
||||||
{
|
|
||||||
components.map((dir, index) => {
|
|
||||||
return (
|
|
||||||
<Crumb
|
|
||||||
key={ dir.path }
|
|
||||||
bold={ index === components.length - 1 }
|
|
||||||
dir={ dir }
|
|
||||||
navigate={ ::this.props.navigate }
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const Breadcrumbs = styled(UnstyledBreadcrumbs)`
|
|
||||||
font-size: 18px;
|
|
||||||
|
|
||||||
& > button:not(:last-child)::after {
|
|
||||||
content: '/';
|
|
||||||
margin: 9px;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
module.exports = Breadcrumbs
|
|
||||||
@@ -1,125 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const rendition = require('rendition')
|
|
||||||
const colors = require('./colors')
|
|
||||||
|
|
||||||
const middleEllipsis = require('../../../utils/middle-ellipsis')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Flex styled component
|
|
||||||
* @function
|
|
||||||
* @type {ReactElement}
|
|
||||||
*/
|
|
||||||
const Flex = styled.div`
|
|
||||||
display: flex;
|
|
||||||
flex: ${ props => props.flex };
|
|
||||||
flex-direction: ${ props => props.direction };
|
|
||||||
justify-content: ${ props => props.justifyContent };
|
|
||||||
align-items: ${ props => props.alignItems };
|
|
||||||
flex-wrap: ${ props => props.wrap };
|
|
||||||
flex-grow: ${ props => props.grow };
|
|
||||||
`
|
|
||||||
|
|
||||||
class RecentFileLink extends React.PureComponent {
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
const file = this.props.file
|
|
||||||
return (
|
|
||||||
<rendition.Button
|
|
||||||
onClick={ ::this.select }
|
|
||||||
plain={ true }>
|
|
||||||
{ middleEllipsis(file.name, FILENAME_CHAR_LIMIT_SHORT) }
|
|
||||||
</rendition.Button>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
select () {
|
|
||||||
this.props.onSelect(this.props.file)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class UnstyledRecentFiles extends React.PureComponent {
|
|
||||||
constructor(props) {
|
|
||||||
super(props)
|
|
||||||
this.state = {
|
|
||||||
recent: [],
|
|
||||||
favorites: []
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
render () {
|
|
||||||
return (
|
|
||||||
<Flex className={ this.props.className }>
|
|
||||||
<h5>Recent</h5>
|
|
||||||
{
|
|
||||||
this.state.recent.map((file) => {
|
|
||||||
<RecentFileLink key={ file.path }
|
|
||||||
file={ file }
|
|
||||||
onSelect={ this.props.selectFile }/>
|
|
||||||
})
|
|
||||||
}
|
|
||||||
<h5>Favorite</h5>
|
|
||||||
{
|
|
||||||
this.state.favorites.map((file) => {
|
|
||||||
<RecentFileLink key={ file.path }
|
|
||||||
file={ file }
|
|
||||||
onSelect={ this.props.navigate }/>
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</Flex>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const RecentFiles = styled(UnstyledRecentFiles)`
|
|
||||||
display: flex;
|
|
||||||
flex: 0 0 auto;
|
|
||||||
flex-direction: column;
|
|
||||||
align-items: flex-start;
|
|
||||||
width: 130px;
|
|
||||||
background-color: ${ colors.secondary.background };
|
|
||||||
padding: 20px;
|
|
||||||
color: ${ colors.secondary.color };
|
|
||||||
|
|
||||||
> h5 {
|
|
||||||
color: ${ colors.secondary.title };
|
|
||||||
font-size: 11px;
|
|
||||||
font-weight: 500;
|
|
||||||
text-transform: uppercase;
|
|
||||||
margin-bottom: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
> h5:last-of-type {
|
|
||||||
margin-top: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
> button {
|
|
||||||
margin-bottom: 10px;
|
|
||||||
text-align: start;
|
|
||||||
font-size: 16px;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
module.exports = RecentFiles
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/* eslint-disable jsdoc/require-example */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.SVGIcon
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const react2angular = require('react2angular').react2angular
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.FileSelector'
|
|
||||||
const angularFileSelector = angular.module(MODULE_NAME, [
|
|
||||||
require('../modal/modal')
|
|
||||||
])
|
|
||||||
|
|
||||||
angularFileSelector.component('fileSelector', react2angular(require('./file-selector/file-selector.jsx')))
|
|
||||||
angularFileSelector.controller('FileSelectorController', require('./controllers/file-selector'))
|
|
||||||
angularFileSelector.service('FileSelectorService', require('./services/file-selector'))
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
module.exports = function (ModalService, $q) {
|
|
||||||
let modal = null
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Open the file selector widget
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorService.open()
|
|
||||||
*/
|
|
||||||
this.open = () => {
|
|
||||||
modal = ModalService.open({
|
|
||||||
name: 'file-selector',
|
|
||||||
template: require('../templates/file-selector-modal.tpl.html'),
|
|
||||||
controller: 'FileSelectorController as selector',
|
|
||||||
size: 'file-selector-modal'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Close the file selector widget
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* DriveSelectorService.close()
|
|
||||||
*/
|
|
||||||
this.close = () => {
|
|
||||||
if (modal) {
|
|
||||||
modal.close()
|
|
||||||
}
|
|
||||||
modal = null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
<file-selector
|
|
||||||
constraintpath="selector.getFolderConstraint()"
|
|
||||||
path="selector.getPath()"
|
|
||||||
close="selector.close"></file-selector>
|
|
||||||
124
lib/gui/app/components/finish/finish.tsx
Normal file
124
lib/gui/app/components/finish/finish.tsx
Normal file
@@ -0,0 +1,124 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2019 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex } from 'rendition';
|
||||||
|
import { v4 as uuidV4 } from 'uuid';
|
||||||
|
|
||||||
|
import * as flashState from '../../models/flash-state';
|
||||||
|
import * as selectionState from '../../models/selection-state';
|
||||||
|
import * as settings from '../../models/settings';
|
||||||
|
import { Actions, store } from '../../models/store';
|
||||||
|
import * as analytics from '../../modules/analytics';
|
||||||
|
import { FlashAnother } from '../flash-another/flash-another';
|
||||||
|
import { FlashResults, FlashError } from '../flash-results/flash-results';
|
||||||
|
import { SafeWebview } from '../safe-webview/safe-webview';
|
||||||
|
|
||||||
|
function restart(goToMain: () => void) {
|
||||||
|
selectionState.deselectAllDrives();
|
||||||
|
analytics.logEvent('Restart');
|
||||||
|
|
||||||
|
// Reset the flashing workflow uuid
|
||||||
|
store.dispatch({
|
||||||
|
type: Actions.SET_FLASHING_WORKFLOW_UUID,
|
||||||
|
data: uuidV4(),
|
||||||
|
});
|
||||||
|
|
||||||
|
goToMain();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getSuccessBannerURL() {
|
||||||
|
return (
|
||||||
|
(await settings.get('successBannerURL')) ??
|
||||||
|
'https://www.balena.io/etcher/success-banner?borderTop=false&darkBackground=true'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function FinishPage({ goToMain }: { goToMain: () => void }) {
|
||||||
|
const [webviewShowing, setWebviewShowing] = React.useState(false);
|
||||||
|
const [successBannerURL, setSuccessBannerURL] = React.useState('');
|
||||||
|
(async () => {
|
||||||
|
setSuccessBannerURL(await getSuccessBannerURL());
|
||||||
|
})();
|
||||||
|
const flashResults = flashState.getFlashResults();
|
||||||
|
const errors: FlashError[] = (
|
||||||
|
store.getState().toJS().failedDeviceErrors || []
|
||||||
|
).map(([, error]: [string, FlashError]) => ({
|
||||||
|
...error,
|
||||||
|
}));
|
||||||
|
const { averageSpeed, blockmappedSize, bytesWritten, failed, size } =
|
||||||
|
flashState.getFlashState();
|
||||||
|
const {
|
||||||
|
skip,
|
||||||
|
results = {
|
||||||
|
bytesWritten,
|
||||||
|
sourceMetadata: {
|
||||||
|
size,
|
||||||
|
blockmappedSize,
|
||||||
|
},
|
||||||
|
averageFlashingSpeed: averageSpeed,
|
||||||
|
devices: { failed, successful: 0 },
|
||||||
|
},
|
||||||
|
} = flashResults;
|
||||||
|
return (
|
||||||
|
<Flex height="100%" justifyContent="space-between">
|
||||||
|
<Flex
|
||||||
|
width={webviewShowing ? '36.2vw' : '100vw'}
|
||||||
|
height="100vh"
|
||||||
|
alignItems="center"
|
||||||
|
justifyContent="center"
|
||||||
|
flexDirection="column"
|
||||||
|
style={{
|
||||||
|
position: 'absolute',
|
||||||
|
top: 0,
|
||||||
|
zIndex: 1,
|
||||||
|
boxShadow: '0 2px 15px 0 rgba(0, 0, 0, 0.2)',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<FlashResults
|
||||||
|
image={selectionState.getImage()?.name}
|
||||||
|
results={results}
|
||||||
|
skip={skip}
|
||||||
|
errors={errors}
|
||||||
|
mb="32px"
|
||||||
|
goToMain={goToMain}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<FlashAnother
|
||||||
|
onClick={() => {
|
||||||
|
restart(goToMain);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Flex>
|
||||||
|
{successBannerURL.length && (
|
||||||
|
<SafeWebview
|
||||||
|
src={successBannerURL}
|
||||||
|
onWebviewShow={setWebviewShowing}
|
||||||
|
style={{
|
||||||
|
display: webviewShowing ? 'flex' : 'none',
|
||||||
|
position: 'absolute',
|
||||||
|
right: 0,
|
||||||
|
bottom: 0,
|
||||||
|
width: '63.8vw',
|
||||||
|
height: '100vh',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default FinishPage;
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
// eslint-disable-next-line no-unused-vars
|
|
||||||
const React = require('react')
|
|
||||||
const PropTypes = require('prop-types')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const { position, right } = require('styled-system')
|
|
||||||
const { BaseButton, ThemedProvider } = require('../../styled-components')
|
|
||||||
|
|
||||||
const Div = styled.div `
|
|
||||||
${position}
|
|
||||||
${right}
|
|
||||||
`
|
|
||||||
|
|
||||||
const FlashAnother = (props) => {
|
|
||||||
return (
|
|
||||||
<ThemedProvider>
|
|
||||||
<Div position='absolute' right='152px'>
|
|
||||||
<BaseButton
|
|
||||||
primary
|
|
||||||
onClick={props.onClick.bind(null, { preserveImage: true })}>
|
|
||||||
Flash Another
|
|
||||||
</BaseButton>
|
|
||||||
</Div>
|
|
||||||
</ThemedProvider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
FlashAnother.propTypes = {
|
|
||||||
onClick: PropTypes.func
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = FlashAnother
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright 2016 resin.io
|
* Copyright 2019 balena.io
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
@@ -14,15 +14,19 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
.modal-warning-modal .modal-content {
|
import * as React from 'react';
|
||||||
width: 350px;
|
|
||||||
|
import { BaseButton } from '../../styled-components';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
export interface FlashAnotherProps {
|
||||||
|
onClick: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
.modal-warning-modal .modal-title .glyphicon {
|
export const FlashAnother = (props: FlashAnotherProps) => {
|
||||||
color: $palette-theme-danger-background;
|
return (
|
||||||
}
|
<BaseButton primary onClick={props.onClick}>
|
||||||
|
{i18next.t('flash.another')}
|
||||||
.modal-warning-modal .modal-body {
|
</BaseButton>
|
||||||
max-height: 200px;
|
);
|
||||||
overflow-y: auto;
|
};
|
||||||
}
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.FlashAnother
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.FlashAnother'
|
|
||||||
const FlashAnother = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
FlashAnother.component(
|
|
||||||
'flashAnother',
|
|
||||||
react2angular(require('./flash-another.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.FlashErrorModal
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const MODULE_NAME = 'Etcher.Components.FlashErrorModal'
|
|
||||||
const FlashErrorModal = angular.module(MODULE_NAME, [
|
|
||||||
require('../warning-modal/warning-modal')
|
|
||||||
])
|
|
||||||
|
|
||||||
FlashErrorModal.service('FlashErrorModalService', require('./services/flash-error-modal'))
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,53 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const flashState = require('../../../models/flash-state')
|
|
||||||
const selectionState = require('../../../models/selection-state')
|
|
||||||
const store = require('../../../models/store')
|
|
||||||
const analytics = require('../../../modules/analytics')
|
|
||||||
|
|
||||||
module.exports = function (WarningModalService) {
|
|
||||||
/**
|
|
||||||
* @summary Open the flash error modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {String} message - flash error message
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* FlashErrorModalService.show('The drive is not large enough!');
|
|
||||||
*/
|
|
||||||
this.show = (message) => {
|
|
||||||
return WarningModalService.display({
|
|
||||||
confirmationLabel: 'Retry',
|
|
||||||
description: message
|
|
||||||
}).then((confirmed) => {
|
|
||||||
flashState.resetState()
|
|
||||||
|
|
||||||
if (confirmed) {
|
|
||||||
analytics.logEvent('Restart after failure', {
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
selectionState.clear()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const PropTypes = require('prop-types')
|
|
||||||
const _ = require('lodash')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const { position, left, top, space } = require('styled-system')
|
|
||||||
const { Underline } = require('./../../styled-components')
|
|
||||||
|
|
||||||
const Div = styled.div `
|
|
||||||
${position}
|
|
||||||
${top}
|
|
||||||
${left}
|
|
||||||
${space}
|
|
||||||
`
|
|
||||||
|
|
||||||
/* eslint-disable no-inline-comments */
|
|
||||||
|
|
||||||
const FlashResults = (props) => {
|
|
||||||
return (
|
|
||||||
<Div position='absolute' left='153px' top='66px'>
|
|
||||||
<div className="inline-flex title">
|
|
||||||
<span className="tick tick--success space-right-medium"></span>
|
|
||||||
<h3>Flash Complete!</h3>
|
|
||||||
</div>
|
|
||||||
<Div className="results" mt='11px' mr='0' mb='0' ml='40px'>
|
|
||||||
<Underline
|
|
||||||
tooltip={props.errors()}>
|
|
||||||
{_.map(props.results.devices, (quantity, type) => {
|
|
||||||
return (quantity) ? (
|
|
||||||
<div key={type} className={`target-status-line target-status-${type}`}>
|
|
||||||
<span className="target-status-dot"></span>
|
|
||||||
<span className="target-status-quantity">{ quantity }</span>
|
|
||||||
<span className="target-status-message">{ props.message[type](quantity) }</span>
|
|
||||||
</div>
|
|
||||||
) : null
|
|
||||||
})}
|
|
||||||
</Underline>
|
|
||||||
</Div>
|
|
||||||
</Div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
FlashResults.propTypes = {
|
|
||||||
results: PropTypes.object,
|
|
||||||
message: PropTypes.object,
|
|
||||||
errors: PropTypes.func
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = FlashResults
|
|
||||||
243
lib/gui/app/components/flash-results/flash-results.tsx
Normal file
243
lib/gui/app/components/flash-results/flash-results.tsx
Normal file
@@ -0,0 +1,243 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2019 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import CircleSvg from '@fortawesome/fontawesome-free/svgs/solid/circle.svg';
|
||||||
|
import CheckCircleSvg from '@fortawesome/fontawesome-free/svgs/solid/check-circle.svg';
|
||||||
|
import TimesCircleSvg from '@fortawesome/fontawesome-free/svgs/solid/times-circle.svg';
|
||||||
|
import outdent from 'outdent';
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex, FlexProps, Link, TableColumn, Txt } from 'rendition';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
import { progress } from '../../../../shared/messages';
|
||||||
|
import { bytesToMegabytes } from '../../../../shared/units';
|
||||||
|
|
||||||
|
import FlashSvg from '../../../assets/flash.svg';
|
||||||
|
import { getDrives } from '../../models/available-drives';
|
||||||
|
import { resetState } from '../../models/flash-state';
|
||||||
|
import * as selection from '../../models/selection-state';
|
||||||
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
||||||
|
import { Modal, Table } from '../../styled-components';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
const ErrorsTable = styled((props) => <Table<FlashError> {...props} />)`
|
||||||
|
&&& [data-display='table-head'],
|
||||||
|
&&& [data-display='table-body'] {
|
||||||
|
> [data-display='table-row'] {
|
||||||
|
> [data-display='table-cell'] {
|
||||||
|
&:first-child {
|
||||||
|
width: 30%;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:nth-child(2) {
|
||||||
|
width: 20%;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
width: 50%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
const DoneIcon = (props: {
|
||||||
|
skipped: boolean;
|
||||||
|
color: string;
|
||||||
|
allFailed: boolean;
|
||||||
|
}) => {
|
||||||
|
const svgProps = {
|
||||||
|
width: '28px',
|
||||||
|
fill: props.color,
|
||||||
|
style: {
|
||||||
|
marginTop: '-25px',
|
||||||
|
marginLeft: '13px',
|
||||||
|
zIndex: 1,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
return props.allFailed && !props.skipped ? (
|
||||||
|
<TimesCircleSvg {...svgProps} />
|
||||||
|
) : (
|
||||||
|
<CheckCircleSvg {...svgProps} />
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface FlashError extends Error {
|
||||||
|
description: string;
|
||||||
|
device: string;
|
||||||
|
code: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
function formattedErrors(errors: FlashError[]) {
|
||||||
|
return errors
|
||||||
|
.map((error) => `${error.device}: ${error.message || error.code}`)
|
||||||
|
.join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
|
const columns: Array<TableColumn<FlashError>> = [
|
||||||
|
{
|
||||||
|
field: 'description',
|
||||||
|
label: i18next.t('flash.target'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'device',
|
||||||
|
label: i18next.t('flash.location'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'message',
|
||||||
|
label: i18next.t('flash.error'),
|
||||||
|
render: (message: string, { code }: FlashError) => {
|
||||||
|
return message ?? code;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
function getEffectiveSpeed(results: {
|
||||||
|
sourceMetadata: {
|
||||||
|
size: number;
|
||||||
|
blockmappedSize?: number;
|
||||||
|
};
|
||||||
|
averageFlashingSpeed: number;
|
||||||
|
}) {
|
||||||
|
const flashedSize =
|
||||||
|
results.sourceMetadata.blockmappedSize ?? results.sourceMetadata.size;
|
||||||
|
const timeSpent = flashedSize / results.averageFlashingSpeed;
|
||||||
|
return results.sourceMetadata.size / timeSpent;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FlashResults({
|
||||||
|
goToMain,
|
||||||
|
image = '',
|
||||||
|
errors,
|
||||||
|
results,
|
||||||
|
skip,
|
||||||
|
...props
|
||||||
|
}: {
|
||||||
|
goToMain: () => void;
|
||||||
|
image?: string;
|
||||||
|
errors: FlashError[];
|
||||||
|
skip: boolean;
|
||||||
|
results: {
|
||||||
|
sourceMetadata: {
|
||||||
|
size: number;
|
||||||
|
blockmappedSize?: number;
|
||||||
|
};
|
||||||
|
averageFlashingSpeed: number;
|
||||||
|
devices: { failed: number; successful: number };
|
||||||
|
};
|
||||||
|
} & FlexProps) {
|
||||||
|
const [showErrorsInfo, setShowErrorsInfo] = React.useState(false);
|
||||||
|
const allFailed = !skip && results.devices.successful === 0;
|
||||||
|
const someFailed = results.devices.failed !== 0 || errors.length !== 0;
|
||||||
|
const effectiveSpeed = bytesToMegabytes(getEffectiveSpeed(results)).toFixed(
|
||||||
|
1,
|
||||||
|
);
|
||||||
|
return (
|
||||||
|
<Flex flexDirection="column" {...props}>
|
||||||
|
<Flex alignItems="center" flexDirection="column">
|
||||||
|
<Flex
|
||||||
|
alignItems="center"
|
||||||
|
mt="50px"
|
||||||
|
mb="32px"
|
||||||
|
color="#7e8085"
|
||||||
|
flexDirection="column"
|
||||||
|
>
|
||||||
|
<FlashSvg width="40px" height="40px" className="disabled" />
|
||||||
|
<DoneIcon
|
||||||
|
skipped={skip}
|
||||||
|
allFailed={allFailed}
|
||||||
|
color={allFailed || someFailed ? '#c6c8c9' : '#1ac135'}
|
||||||
|
/>
|
||||||
|
<Txt>{middleEllipsis(image, 24)}</Txt>
|
||||||
|
</Flex>
|
||||||
|
<Txt fontSize={24} color="#fff" mb="17px">
|
||||||
|
{allFailed
|
||||||
|
? i18next.t('flash.flashFailed')
|
||||||
|
: i18next.t('flash.flashCompleted')}
|
||||||
|
</Txt>
|
||||||
|
{skip ? <Txt color="#7e8085">{i18next.t('flash.skip')}</Txt> : null}
|
||||||
|
</Flex>
|
||||||
|
<Flex flexDirection="column" color="#7e8085">
|
||||||
|
{results.devices.successful !== 0 ? (
|
||||||
|
<Flex alignItems="center">
|
||||||
|
<CircleSvg width="14px" fill="#1ac135" />
|
||||||
|
<Txt ml="10px" color="#fff">
|
||||||
|
{results.devices.successful}
|
||||||
|
</Txt>
|
||||||
|
<Txt ml="10px">
|
||||||
|
{progress.successful(results.devices.successful)}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
) : null}
|
||||||
|
{errors.length !== 0 ? (
|
||||||
|
<Flex alignItems="center">
|
||||||
|
<CircleSvg width="14px" fill="#ff4444" />
|
||||||
|
<Txt ml="10px" color="#fff">
|
||||||
|
{errors.length}
|
||||||
|
</Txt>
|
||||||
|
<Txt ml="10px" tooltip={formattedErrors(errors)}>
|
||||||
|
{progress.failed(errors.length)}
|
||||||
|
</Txt>
|
||||||
|
<Link ml="10px" onClick={() => setShowErrorsInfo(true)}>
|
||||||
|
{i18next.t('flash.moreInfo')}
|
||||||
|
</Link>
|
||||||
|
</Flex>
|
||||||
|
) : null}
|
||||||
|
{!allFailed && (
|
||||||
|
<Txt
|
||||||
|
fontSize="10px"
|
||||||
|
style={{
|
||||||
|
fontWeight: 500,
|
||||||
|
textAlign: 'center',
|
||||||
|
}}
|
||||||
|
tooltip={i18next.t('flash.speedTip')}
|
||||||
|
>
|
||||||
|
{i18next.t('flash.speed', { speed: effectiveSpeed })}
|
||||||
|
</Txt>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
|
||||||
|
{showErrorsInfo && (
|
||||||
|
<Modal
|
||||||
|
titleElement={
|
||||||
|
<Flex alignItems="baseline" mb={18}>
|
||||||
|
<Txt fontSize={24} align="left">
|
||||||
|
{i18next.t('failedTarget')}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
}
|
||||||
|
action={i18next.t('failedRetry')}
|
||||||
|
cancel={() => setShowErrorsInfo(false)}
|
||||||
|
done={() => {
|
||||||
|
setShowErrorsInfo(false);
|
||||||
|
resetState();
|
||||||
|
getDrives()
|
||||||
|
.map((drive) => {
|
||||||
|
selection.deselectDrive(drive.device);
|
||||||
|
return drive.device;
|
||||||
|
})
|
||||||
|
.filter((driveDevice) =>
|
||||||
|
errors.some((error) => error.device === driveDevice),
|
||||||
|
)
|
||||||
|
.forEach((driveDevice) => selection.selectDrive(driveDevice));
|
||||||
|
goToMain();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<ErrorsTable columns={columns} data={errors} />
|
||||||
|
</Modal>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.FlashResults
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.FlashResults'
|
|
||||||
const FlashResults = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
FlashResults.component(
|
|
||||||
'flashResults',
|
|
||||||
react2angular(require('./flash-results.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,98 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/* eslint-disable no-unused-vars */
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
|
|
||||||
const middleEllipsis = require('./../../utils/middle-ellipsis')
|
|
||||||
|
|
||||||
const shared = require('./../../../../shared/units')
|
|
||||||
const {
|
|
||||||
StepButton,
|
|
||||||
StepNameButton,
|
|
||||||
StepSelection,
|
|
||||||
Footer,
|
|
||||||
Underline,
|
|
||||||
DetailsText,
|
|
||||||
ChangeButton,
|
|
||||||
ThemedProvider
|
|
||||||
} = require('./../../styled-components')
|
|
||||||
|
|
||||||
const SelectImageButton = (props) => {
|
|
||||||
if (props.hasImage) {
|
|
||||||
return (
|
|
||||||
<ThemedProvider>
|
|
||||||
<StepNameButton
|
|
||||||
plain
|
|
||||||
onClick={props.showSelectedImageDetails}
|
|
||||||
tooltip={props.imageBasename}
|
|
||||||
>
|
|
||||||
{/* eslint-disable no-magic-numbers */}
|
|
||||||
{ middleEllipsis(props.imageName || props.imageBasename, 20) }
|
|
||||||
</StepNameButton>
|
|
||||||
{ !props.flashing &&
|
|
||||||
<ChangeButton
|
|
||||||
plain
|
|
||||||
mb={14}
|
|
||||||
onClick={props.reselectImage}
|
|
||||||
>
|
|
||||||
Change
|
|
||||||
</ChangeButton>
|
|
||||||
}
|
|
||||||
<DetailsText>
|
|
||||||
{shared.bytesToClosestUnit(props.imageSize)}
|
|
||||||
</DetailsText>
|
|
||||||
</ThemedProvider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
return (
|
|
||||||
<ThemedProvider>
|
|
||||||
<StepSelection>
|
|
||||||
<StepButton
|
|
||||||
onClick={props.openImageSelector}
|
|
||||||
>
|
|
||||||
Select image
|
|
||||||
</StepButton>
|
|
||||||
<Footer>
|
|
||||||
{ props.mainSupportedExtensions.join(', ') }, and{' '}
|
|
||||||
<Underline
|
|
||||||
tooltip={ props.extraSupportedExtensions.join(', ') }
|
|
||||||
>
|
|
||||||
many more
|
|
||||||
</Underline>
|
|
||||||
</Footer>
|
|
||||||
</StepSelection>
|
|
||||||
</ThemedProvider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
SelectImageButton.propTypes = {
|
|
||||||
openImageSelector: propTypes.func,
|
|
||||||
mainSupportedExtensions: propTypes.array,
|
|
||||||
extraSupportedExtensions: propTypes.array,
|
|
||||||
hasImage: propTypes.bool,
|
|
||||||
showSelectedImageDetails: propTypes.func,
|
|
||||||
imageName: propTypes.string,
|
|
||||||
imageBasename: propTypes.string,
|
|
||||||
reselectImage: propTypes.func,
|
|
||||||
flashing: propTypes.bool,
|
|
||||||
imageSize: propTypes.number
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = SelectImageButton
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.ImageSelector
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.ImageSelector'
|
|
||||||
const SelectImageButton = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
SelectImageButton.component(
|
|
||||||
'imageSelector',
|
|
||||||
react2angular(require('./image-selector.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,100 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
const store = require('../../../models/store')
|
|
||||||
const analytics = require('../../../modules/analytics')
|
|
||||||
|
|
||||||
module.exports = function ($uibModal, $q) {
|
|
||||||
/**
|
|
||||||
* @summary Open a modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} options - options
|
|
||||||
* @param {String} options.template - template contents
|
|
||||||
* @param {String} options.controller - controller
|
|
||||||
* @param {String} [options.size='sm'] - modal size
|
|
||||||
* @param {Object} options.resolve - modal resolves
|
|
||||||
* @returns {Object} modal
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* ModalService.open({
|
|
||||||
* name: 'my modal',
|
|
||||||
* template: require('./path/to/modal.tpl.html'),
|
|
||||||
* controller: 'DriveSelectorController as modal',
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.open = (options = {}) => {
|
|
||||||
_.defaults(options, {
|
|
||||||
size: 'sm'
|
|
||||||
})
|
|
||||||
|
|
||||||
analytics.logEvent('Open modal', {
|
|
||||||
name: options.name,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
const modal = $uibModal.open({
|
|
||||||
animation: true,
|
|
||||||
template: options.template,
|
|
||||||
controller: options.controller,
|
|
||||||
size: options.size,
|
|
||||||
resolve: options.resolve,
|
|
||||||
backdrop: 'static'
|
|
||||||
})
|
|
||||||
|
|
||||||
return {
|
|
||||||
close: modal.close,
|
|
||||||
result: $q((resolve, reject) => {
|
|
||||||
modal.result.then((value) => {
|
|
||||||
analytics.logEvent('Modal accepted', {
|
|
||||||
name: options.name,
|
|
||||||
value,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
resolve(value)
|
|
||||||
}).catch((error) => {
|
|
||||||
// Bootstrap doesn't 'resolve' these but cancels the dialog
|
|
||||||
if (error === 'escape key press') {
|
|
||||||
analytics.logEvent('Modal rejected', {
|
|
||||||
name: options.name,
|
|
||||||
method: error,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
return resolve()
|
|
||||||
}
|
|
||||||
|
|
||||||
analytics.logEvent('Modal rejected', {
|
|
||||||
name: options.name,
|
|
||||||
value: error,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
return reject(error)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
.modal-content {
|
|
||||||
background-color: $palette-theme-light-background;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
margin: 0 auto;
|
|
||||||
height: auto;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-header {
|
|
||||||
display: flex;
|
|
||||||
align-items: baseline;
|
|
||||||
font-size: 12px;
|
|
||||||
color: $palette-theme-light-soft-foreground;
|
|
||||||
padding: 11px 20px;
|
|
||||||
flex-grow: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-title {
|
|
||||||
font-size: inherit;
|
|
||||||
flex-grow: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-body {
|
|
||||||
flex-grow: 1;
|
|
||||||
color: $palette-theme-light-foreground;
|
|
||||||
padding: 20px;
|
|
||||||
max-height: 250px;
|
|
||||||
overflow: auto;
|
|
||||||
|
|
||||||
a {
|
|
||||||
color: $palette-theme-primary-background;
|
|
||||||
}
|
|
||||||
|
|
||||||
> p {
|
|
||||||
white-space: pre-line;
|
|
||||||
}
|
|
||||||
|
|
||||||
> p:last-child {
|
|
||||||
margin-bottom: 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-menu {
|
|
||||||
display: flex;
|
|
||||||
|
|
||||||
> * {
|
|
||||||
flex-basis: auto;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// UI Bootstrap adds the `.modal-open` class to the <body>
|
|
||||||
// element and sets its right padding to the width of the
|
|
||||||
// window, causing the window content to overflow and get
|
|
||||||
// pushed to the bottom.
|
|
||||||
// The `!important` flag is needed since UI Bootstrap inlines
|
|
||||||
// the styles programmatically to the element.
|
|
||||||
.modal-open {
|
|
||||||
padding-right: 0 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Disable modal opacity
|
|
||||||
.modal-backdrop.in {
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-footer {
|
|
||||||
flex-grow: 0;
|
|
||||||
border: 0;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal {
|
|
||||||
|
|
||||||
// Center the modal using Flexbox so we can
|
|
||||||
// freely use any height.
|
|
||||||
display: flex !important;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
.button[disabled] {
|
|
||||||
background-color: $palette-theme-light-disabled-background;
|
|
||||||
color: $palette-theme-light-disabled-foreground;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.modal-dialog {
|
|
||||||
margin: 0;
|
|
||||||
position: initial;
|
|
||||||
}
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.ProgressButton
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.ProgressButton'
|
|
||||||
const ProgressButton = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
ProgressButton.component(
|
|
||||||
'progressButton',
|
|
||||||
react2angular(require('./progress-button.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,161 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const Color = require('color')
|
|
||||||
|
|
||||||
const {
|
|
||||||
default: styled,
|
|
||||||
css,
|
|
||||||
keyframes
|
|
||||||
} = require('styled-components')
|
|
||||||
|
|
||||||
const { ProgressBar, Provider } = require('rendition')
|
|
||||||
|
|
||||||
const { colors } = require('./../../theme')
|
|
||||||
const { StepButton, StepSelection } = require('./../../styled-components')
|
|
||||||
|
|
||||||
const darkenForegroundStripes = 0.18
|
|
||||||
const desaturateForegroundStripes = 0.2
|
|
||||||
const progressButtonStripesForegroundColor = Color(colors.primary.background)
|
|
||||||
.darken(darkenForegroundStripes)
|
|
||||||
.desaturate(desaturateForegroundStripes)
|
|
||||||
.string()
|
|
||||||
|
|
||||||
const desaturateBackgroundStripes = 0.05
|
|
||||||
const progressButtonStripesBackgroundColor = Color(colors.primary.background)
|
|
||||||
.desaturate(desaturateBackgroundStripes)
|
|
||||||
.string()
|
|
||||||
|
|
||||||
const ProgressButtonStripes = keyframes `
|
|
||||||
0% {
|
|
||||||
background-position: 0 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
100% {
|
|
||||||
background-position: 20px 20px;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
const ProgressButtonStripesRule = css `
|
|
||||||
${ProgressButtonStripes} 1s linear infinite;
|
|
||||||
`
|
|
||||||
|
|
||||||
const FlashProgressBar = styled(ProgressBar) `
|
|
||||||
> div {
|
|
||||||
width: 200px;
|
|
||||||
height: 48px;
|
|
||||||
color: white !important;
|
|
||||||
text-shadow: none !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
width: 200px;
|
|
||||||
height: 48px;
|
|
||||||
font-size: 16px;
|
|
||||||
line-height: 48px;
|
|
||||||
|
|
||||||
background: ${Color(colors.warning.background).darken(darkenForegroundStripes).string()};
|
|
||||||
`
|
|
||||||
|
|
||||||
const FlashProgressBarValidating = styled(FlashProgressBar) `
|
|
||||||
|
|
||||||
// Notice that we add 0.01 to certain gradient stop positions.
|
|
||||||
// That workarounds a Chrome rendering issue where diagonal
|
|
||||||
// lines look spiky.
|
|
||||||
// See https://github.com/resin-io/etcher/issues/472
|
|
||||||
|
|
||||||
background-image: -webkit-gradient(linear, 0 0, 100% 100%,
|
|
||||||
color-stop(0.25, ${progressButtonStripesForegroundColor}),
|
|
||||||
color-stop(0.26, ${progressButtonStripesBackgroundColor}),
|
|
||||||
color-stop(0.50, ${progressButtonStripesBackgroundColor}),
|
|
||||||
color-stop(0.51, ${progressButtonStripesForegroundColor}),
|
|
||||||
color-stop(0.75, ${progressButtonStripesForegroundColor}),
|
|
||||||
color-stop(0.76 , ${progressButtonStripesBackgroundColor}),
|
|
||||||
to(${progressButtonStripesBackgroundColor}));
|
|
||||||
|
|
||||||
background-color: white;
|
|
||||||
|
|
||||||
animation: ${ProgressButtonStripesRule};
|
|
||||||
overflow: hidden;
|
|
||||||
|
|
||||||
background-size: 20px 20px;
|
|
||||||
`
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Progress Button component
|
|
||||||
*/
|
|
||||||
class ProgressButton extends React.Component {
|
|
||||||
render () {
|
|
||||||
if (this.props.active) {
|
|
||||||
if (this.props.striped) {
|
|
||||||
return (
|
|
||||||
<Provider>
|
|
||||||
<StepSelection>
|
|
||||||
<FlashProgressBarValidating
|
|
||||||
primary
|
|
||||||
emphasized
|
|
||||||
value= { this.props.percentage }
|
|
||||||
>
|
|
||||||
{ this.props.label }
|
|
||||||
</FlashProgressBarValidating>
|
|
||||||
</StepSelection>
|
|
||||||
</Provider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Provider>
|
|
||||||
<StepSelection>
|
|
||||||
<FlashProgressBar
|
|
||||||
warning
|
|
||||||
emphasized
|
|
||||||
value= { this.props.percentage }
|
|
||||||
>
|
|
||||||
{ this.props.label }
|
|
||||||
</FlashProgressBar>
|
|
||||||
</StepSelection>
|
|
||||||
</Provider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Provider>
|
|
||||||
<StepSelection>
|
|
||||||
<StepButton
|
|
||||||
onClick= { this.props.callback }
|
|
||||||
disabled= { this.props.disabled }
|
|
||||||
>
|
|
||||||
{this.props.label}
|
|
||||||
</StepButton>
|
|
||||||
</StepSelection>
|
|
||||||
</Provider>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ProgressButton.propTypes = {
|
|
||||||
striped: propTypes.bool,
|
|
||||||
active: propTypes.bool,
|
|
||||||
percentage: propTypes.number,
|
|
||||||
label: propTypes.string,
|
|
||||||
disabled: propTypes.bool,
|
|
||||||
callback: propTypes.func
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = ProgressButton
|
|
||||||
136
lib/gui/app/components/progress-button/progress-button.tsx
Normal file
136
lib/gui/app/components/progress-button/progress-button.tsx
Normal file
@@ -0,0 +1,136 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex, Button, ProgressBar, Txt } from 'rendition';
|
||||||
|
import { default as styled } from 'styled-components';
|
||||||
|
|
||||||
|
import { fromFlashState } from '../../modules/progress-status';
|
||||||
|
import { StepButton } from '../../styled-components';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
const FlashProgressBar = styled(ProgressBar)`
|
||||||
|
> div {
|
||||||
|
width: 100%;
|
||||||
|
height: 12px;
|
||||||
|
color: white !important;
|
||||||
|
text-shadow: none !important;
|
||||||
|
transition-duration: 0s;
|
||||||
|
|
||||||
|
> div {
|
||||||
|
transition-duration: 0s;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
width: 100%;
|
||||||
|
height: 12px;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
border-radius: 14px;
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 48px;
|
||||||
|
|
||||||
|
background: #2f3033;
|
||||||
|
`;
|
||||||
|
|
||||||
|
interface ProgressButtonProps {
|
||||||
|
type: 'decompressing' | 'flashing' | 'verifying';
|
||||||
|
active: boolean;
|
||||||
|
percentage: number;
|
||||||
|
position: number;
|
||||||
|
disabled: boolean;
|
||||||
|
cancel: (type: string) => void;
|
||||||
|
callback: () => void;
|
||||||
|
warning?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const colors = {
|
||||||
|
decompressing: '#00aeef',
|
||||||
|
flashing: '#da60ff',
|
||||||
|
verifying: '#1ac135',
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
const CancelButton = styled(({ type, onClick, ...props }) => {
|
||||||
|
const status = type === 'verifying' ? i18next.t('skip') : i18next.t('cancel');
|
||||||
|
return (
|
||||||
|
<Button plain onClick={() => onClick(status)} {...props}>
|
||||||
|
{status}
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
|
})`
|
||||||
|
font-weight: 600;
|
||||||
|
|
||||||
|
&&& {
|
||||||
|
width: auto;
|
||||||
|
height: auto;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export class ProgressButton extends React.PureComponent<ProgressButtonProps> {
|
||||||
|
public render() {
|
||||||
|
const percentage = this.props.percentage;
|
||||||
|
const warning = this.props.warning;
|
||||||
|
const { status, position } = fromFlashState({
|
||||||
|
type: this.props.type,
|
||||||
|
percentage,
|
||||||
|
position: this.props.position,
|
||||||
|
});
|
||||||
|
const type = this.props.type || 'default';
|
||||||
|
if (this.props.active) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Flex
|
||||||
|
alignItems="baseline"
|
||||||
|
justifyContent="space-between"
|
||||||
|
width="100%"
|
||||||
|
style={{
|
||||||
|
marginTop: 42,
|
||||||
|
marginBottom: '6px',
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: 600,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Flex>
|
||||||
|
<Txt color="#fff">{status} </Txt>
|
||||||
|
<Txt color={colors[type]}>{position}</Txt>
|
||||||
|
</Flex>
|
||||||
|
{type && (
|
||||||
|
<CancelButton
|
||||||
|
type={type}
|
||||||
|
onClick={this.props.cancel}
|
||||||
|
color="#00aeef"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
<FlashProgressBar background={colors[type]} value={percentage} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<StepButton
|
||||||
|
primary={!warning}
|
||||||
|
warning={warning}
|
||||||
|
onClick={this.props.callback}
|
||||||
|
disabled={this.props.disabled}
|
||||||
|
style={{
|
||||||
|
marginTop: 30,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{i18next.t('flash.flashNow')}
|
||||||
|
</StepButton>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.ReducedFlashingInfos
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.ReducedFlashingInfos'
|
|
||||||
const ReducedFlashingInfos = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
ReducedFlashingInfos.component(
|
|
||||||
'reducedFlashingInfos',
|
|
||||||
react2angular(require('./reduced-flashing-infos.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,81 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const React = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const styled = require('styled-components').default
|
|
||||||
const { color } = require('styled-system')
|
|
||||||
const SvgIcon = require('../svg-icon/svg-icon.jsx')
|
|
||||||
|
|
||||||
const Div = styled.div `
|
|
||||||
position: absolute;
|
|
||||||
top: 45px;
|
|
||||||
left: 545px;
|
|
||||||
|
|
||||||
> span.step-name {
|
|
||||||
justify-content: flex-start;
|
|
||||||
|
|
||||||
> span {
|
|
||||||
margin-left: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
> span:nth-child(2) {
|
|
||||||
font-weight: 500;
|
|
||||||
}
|
|
||||||
|
|
||||||
> span:nth-child(3) {
|
|
||||||
font-weight: 400;
|
|
||||||
font-style: italic;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.svg-icon[disabled] {
|
|
||||||
opacity: 0.4;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
|
|
||||||
const Span = styled.span `
|
|
||||||
${color}
|
|
||||||
`
|
|
||||||
|
|
||||||
const ReducedFlashingInfos = (props) => {
|
|
||||||
return (props.shouldShow) ? (
|
|
||||||
<Div>
|
|
||||||
<Span className="step-name">
|
|
||||||
<SvgIcon disabled contents={[ props.imageLogo ]} paths={[ '../../assets/image.svg' ]} width='20px'></SvgIcon>
|
|
||||||
<Span>{ props.imageName }</Span>
|
|
||||||
<Span color='#7e8085'>{ props.imageSize }</Span>
|
|
||||||
</Span>
|
|
||||||
|
|
||||||
<Span className="step-name">
|
|
||||||
<SvgIcon disabled paths={[ '../../assets/drive.svg' ]} width='20px'></SvgIcon>
|
|
||||||
<Span>{ props.driveTitle }</Span>
|
|
||||||
</Span>
|
|
||||||
</Div>
|
|
||||||
) : null
|
|
||||||
}
|
|
||||||
|
|
||||||
ReducedFlashingInfos.propTypes = {
|
|
||||||
imageLogo: propTypes.string,
|
|
||||||
imageName: propTypes.string,
|
|
||||||
imageSize: propTypes.string,
|
|
||||||
driveTitle: propTypes.string,
|
|
||||||
shouldShow: propTypes.bool
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = ReducedFlashingInfos
|
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex, Txt } from 'rendition';
|
||||||
|
|
||||||
|
import DriveSvg from '../../../assets/drive.svg';
|
||||||
|
import ImageSvg from '../../../assets/image.svg';
|
||||||
|
import { SVGIcon } from '../svg-icon/svg-icon';
|
||||||
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
||||||
|
|
||||||
|
interface ReducedFlashingInfosProps {
|
||||||
|
imageLogo?: string;
|
||||||
|
imageName?: string;
|
||||||
|
imageSize: string;
|
||||||
|
driveTitle: string;
|
||||||
|
driveLabel: string;
|
||||||
|
style?: React.CSSProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ReducedFlashingInfos extends React.Component<ReducedFlashingInfosProps> {
|
||||||
|
constructor(props: ReducedFlashingInfosProps) {
|
||||||
|
super(props);
|
||||||
|
this.state = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
public render() {
|
||||||
|
const { imageName = '' } = this.props;
|
||||||
|
return (
|
||||||
|
<Flex
|
||||||
|
flexDirection="column"
|
||||||
|
style={this.props.style ? this.props.style : undefined}
|
||||||
|
>
|
||||||
|
<Flex mb={16}>
|
||||||
|
<SVGIcon
|
||||||
|
disabled
|
||||||
|
width="21px"
|
||||||
|
height="21px"
|
||||||
|
contents={this.props.imageLogo}
|
||||||
|
fallback={ImageSvg}
|
||||||
|
style={{ marginRight: '9px' }}
|
||||||
|
/>
|
||||||
|
<Txt
|
||||||
|
style={{ marginRight: '9px' }}
|
||||||
|
tooltip={{ text: imageName, placement: 'right' }}
|
||||||
|
>
|
||||||
|
{middleEllipsis(imageName, 16)}
|
||||||
|
</Txt>
|
||||||
|
<Txt color="#7e8085">{this.props.imageSize}</Txt>
|
||||||
|
</Flex>
|
||||||
|
|
||||||
|
<Flex>
|
||||||
|
<DriveSvg width="21px" height="21px" style={{ marginRight: '9px' }} />
|
||||||
|
<Txt tooltip={{ text: this.props.driveLabel, placement: 'right' }}>
|
||||||
|
{middleEllipsis(this.props.driveTitle, 16)}
|
||||||
|
</Txt>
|
||||||
|
</Flex>
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.SafeWebview
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const { react2angular } = require('react2angular')
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.SafeWebview'
|
|
||||||
const SafeWebview = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
SafeWebview.component(
|
|
||||||
'safeWebview',
|
|
||||||
react2angular(require('./safe-webview.jsx'))
|
|
||||||
)
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,263 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2017 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/* eslint-disable jsdoc/require-example */
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
const electron = require('electron')
|
|
||||||
const react = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const analytics = require('../../modules/analytics')
|
|
||||||
const store = require('../../models/store')
|
|
||||||
const settings = require('../../models/settings')
|
|
||||||
const packageJSON = require('../../../../../package.json')
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Electron session identifier
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
* @type {String}
|
|
||||||
*/
|
|
||||||
const ELECTRON_SESSION = 'persist:success-banner'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Etcher version search-parameter key
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
* @type {String}
|
|
||||||
*/
|
|
||||||
const ETCHER_VERSION_PARAM = 'etcher-version'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary API version search-parameter key
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
* @type {String}
|
|
||||||
*/
|
|
||||||
const API_VERSION_PARAM = 'api-version'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Opt-out analytics search-parameter key
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
* @type {String}
|
|
||||||
*/
|
|
||||||
const OPT_OUT_ANALYTICS_PARAM = 'optOutAnalytics'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Webview API version
|
|
||||||
* @constant
|
|
||||||
* @private
|
|
||||||
* @type {String}
|
|
||||||
*
|
|
||||||
* @description
|
|
||||||
* Changing this number represents a departure from an older API and as such
|
|
||||||
* should only be changed when truly necessary as it introduces breaking changes.
|
|
||||||
* This version number is exposed to the banner such that it can determine what
|
|
||||||
* features are safe to utilize.
|
|
||||||
*
|
|
||||||
* See `git blame -L n` where n is the line below for the history of version changes.
|
|
||||||
*/
|
|
||||||
const API_VERSION = 2
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Webviews that hide/show depending on the HTTP status returned
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* <safe-webview src="https://etcher.io/"></safe-webview>
|
|
||||||
*/
|
|
||||||
class SafeWebview extends react.PureComponent {
|
|
||||||
/**
|
|
||||||
* @param {Object} props - React element properties
|
|
||||||
*/
|
|
||||||
constructor (props) {
|
|
||||||
super(props)
|
|
||||||
|
|
||||||
this.state = {
|
|
||||||
shouldShow: true
|
|
||||||
}
|
|
||||||
|
|
||||||
const url = new window.URL(props.src)
|
|
||||||
|
|
||||||
// We set the version GET parameters here.
|
|
||||||
url.searchParams.set(ETCHER_VERSION_PARAM, packageJSON.version)
|
|
||||||
url.searchParams.set(API_VERSION_PARAM, API_VERSION)
|
|
||||||
url.searchParams.set(OPT_OUT_ANALYTICS_PARAM, !settings.get('errorReporting'))
|
|
||||||
|
|
||||||
this.entryHref = url.href
|
|
||||||
|
|
||||||
// Events steal 'this'
|
|
||||||
this.didFailLoad = _.bind(this.didFailLoad, this)
|
|
||||||
this.didGetResponseDetails = _.bind(this.didGetResponseDetails, this)
|
|
||||||
|
|
||||||
const logWebViewMessage = (event) => {
|
|
||||||
console.log('Message from SafeWebview:', event.message);
|
|
||||||
};
|
|
||||||
|
|
||||||
this.eventTuples = [
|
|
||||||
[ 'did-fail-load', this.didFailLoad ],
|
|
||||||
[ 'new-window', this.constructor.newWindow ],
|
|
||||||
[ 'console-message', logWebViewMessage ]
|
|
||||||
]
|
|
||||||
|
|
||||||
// Make a persistent electron session for the webview
|
|
||||||
this.session = electron.remote.session.fromPartition(ELECTRON_SESSION, {
|
|
||||||
|
|
||||||
// Disable the cache for the session such that new content shows up when refreshing
|
|
||||||
cache: false
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @returns {react.Element}
|
|
||||||
*/
|
|
||||||
render () {
|
|
||||||
return react.createElement('webview', {
|
|
||||||
ref: 'webview',
|
|
||||||
partition: ELECTRON_SESSION,
|
|
||||||
style: {
|
|
||||||
flex: this.state.shouldShow ? null : '0 1',
|
|
||||||
width: this.state.shouldShow ? null : '0',
|
|
||||||
height: this.state.shouldShow ? null : '0'
|
|
||||||
}
|
|
||||||
}, [])
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Add the Webview events
|
|
||||||
*/
|
|
||||||
componentDidMount () {
|
|
||||||
// Events React is unaware of have to be handled manually
|
|
||||||
_.map(this.eventTuples, (tuple) => {
|
|
||||||
this.refs.webview.addEventListener(...tuple)
|
|
||||||
})
|
|
||||||
|
|
||||||
this.session.webRequest.onCompleted(this.didGetResponseDetails)
|
|
||||||
|
|
||||||
// It's important that this comes after the partition setting, otherwise it will
|
|
||||||
// use another session and we can't change it without destroying the element again
|
|
||||||
this.refs.webview.src = this.entryHref
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Remove the Webview events
|
|
||||||
*/
|
|
||||||
componentWillUnmount () {
|
|
||||||
// Events that React is unaware of have to be handled manually
|
|
||||||
_.map(this.eventTuples, (tuple) => {
|
|
||||||
this.refs.webview.removeEventListener(...tuple)
|
|
||||||
})
|
|
||||||
this.session.webRequest.onCompleted(null)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Refresh the webview if we are navigating away from the success page
|
|
||||||
* @param {Object} nextProps - upcoming properties
|
|
||||||
*/
|
|
||||||
componentWillReceiveProps (nextProps) {
|
|
||||||
if (nextProps.refreshNow && !this.props.refreshNow) {
|
|
||||||
// Reload the page if it hasn't changed, otherwise reset the source URL,
|
|
||||||
// because reload interferes with 'src' setting, resetting the 'src' attribute
|
|
||||||
// to what it was was just prior.
|
|
||||||
if (this.refs.webview.src === this.entryHref) {
|
|
||||||
this.refs.webview.reload()
|
|
||||||
} else {
|
|
||||||
this.refs.webview.src = this.entryHref
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
shouldShow: true
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Set the element state to hidden
|
|
||||||
*/
|
|
||||||
didFailLoad () {
|
|
||||||
this.setState({
|
|
||||||
shouldShow: false
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Set the element state depending on the HTTP response code
|
|
||||||
* @param {Event} event - Event object
|
|
||||||
*/
|
|
||||||
didGetResponseDetails (event) {
|
|
||||||
// This seems to pick up all requests related to the webview,
|
|
||||||
// only care about this event if it's a request for the main frame
|
|
||||||
if (event.resourceType === 'mainFrame') {
|
|
||||||
const HTTP_OK = 200
|
|
||||||
|
|
||||||
analytics.logEvent('SafeWebview loaded', {
|
|
||||||
event,
|
|
||||||
applicationSessionUuid: store.getState().toJS().applicationSessionUuid,
|
|
||||||
flashingWorkflowUuid: store.getState().toJS().flashingWorkflowUuid
|
|
||||||
})
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
shouldShow: event.statusCode === HTTP_OK
|
|
||||||
})
|
|
||||||
if (this.props.onWebviewShow) {
|
|
||||||
this.props.onWebviewShow(event.statusCode === HTTP_OK)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Open link in browser if it's opened as a 'foreground-tab'
|
|
||||||
* @param {Event} event - event object
|
|
||||||
*/
|
|
||||||
static newWindow (event) {
|
|
||||||
const url = new window.URL(event.url)
|
|
||||||
|
|
||||||
if (_.every([
|
|
||||||
url.protocol === 'http:' || url.protocol === 'https:',
|
|
||||||
event.disposition === 'foreground-tab',
|
|
||||||
|
|
||||||
// Don't open links if they're disabled by the env var
|
|
||||||
!settings.get('disableExternalLinks')
|
|
||||||
])) {
|
|
||||||
electron.shell.openExternal(url.href)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
SafeWebview.propTypes = {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary The website source URL
|
|
||||||
*/
|
|
||||||
src: propTypes.string.isRequired,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Refresh the webview
|
|
||||||
*/
|
|
||||||
refreshNow: propTypes.bool,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Webview lifecycle event
|
|
||||||
*/
|
|
||||||
onWebviewShow: propTypes.func
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = SafeWebview
|
|
||||||
208
lib/gui/app/components/safe-webview/safe-webview.tsx
Normal file
208
lib/gui/app/components/safe-webview/safe-webview.tsx
Normal file
@@ -0,0 +1,208 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2017 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as electron from 'electron';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
import * as React from 'react';
|
||||||
|
|
||||||
|
import * as packageJSON from '../../../../../package.json';
|
||||||
|
import * as settings from '../../models/settings';
|
||||||
|
import * as analytics from '../../modules/analytics';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Electron session identifier
|
||||||
|
*/
|
||||||
|
const ELECTRON_SESSION = 'persist:success-banner';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Etcher version search-parameter key
|
||||||
|
*/
|
||||||
|
const ETCHER_VERSION_PARAM = 'etcher-version';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary API version search-parameter key
|
||||||
|
*/
|
||||||
|
const API_VERSION_PARAM = 'api-version';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Opt-out analytics search-parameter key
|
||||||
|
*/
|
||||||
|
const OPT_OUT_ANALYTICS_PARAM = 'optOutAnalytics';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Webview API version
|
||||||
|
*
|
||||||
|
* @description
|
||||||
|
* Changing this number represents a departure from an older API and as such
|
||||||
|
* should only be changed when truly necessary as it introduces breaking changes.
|
||||||
|
* This version number is exposed to the banner such that it can determine what
|
||||||
|
* features are safe to utilize.
|
||||||
|
*
|
||||||
|
* See `git blame -L n` where n is the line below for the history of version changes.
|
||||||
|
*/
|
||||||
|
const API_VERSION = '2';
|
||||||
|
|
||||||
|
interface SafeWebviewProps {
|
||||||
|
// The website source URL
|
||||||
|
src: string;
|
||||||
|
// Webview lifecycle event
|
||||||
|
onWebviewShow?: (isWebviewShowing: boolean) => void;
|
||||||
|
style?: React.CSSProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SafeWebviewState {
|
||||||
|
shouldShow: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Webviews that hide/show depending on the HTTP status returned
|
||||||
|
*/
|
||||||
|
export class SafeWebview extends React.PureComponent<
|
||||||
|
SafeWebviewProps,
|
||||||
|
SafeWebviewState
|
||||||
|
> {
|
||||||
|
private entryHref: string;
|
||||||
|
private session: electron.Session;
|
||||||
|
private webviewRef: React.RefObject<electron.WebviewTag>;
|
||||||
|
|
||||||
|
constructor(props: SafeWebviewProps) {
|
||||||
|
super(props);
|
||||||
|
this.webviewRef = React.createRef();
|
||||||
|
this.state = {
|
||||||
|
shouldShow: true,
|
||||||
|
};
|
||||||
|
const url = new window.URL(this.props.src);
|
||||||
|
// We set the version GET parameters here.
|
||||||
|
url.searchParams.set(ETCHER_VERSION_PARAM, packageJSON.version);
|
||||||
|
url.searchParams.set(API_VERSION_PARAM, API_VERSION);
|
||||||
|
url.searchParams.set(
|
||||||
|
OPT_OUT_ANALYTICS_PARAM,
|
||||||
|
(!settings.getSync('errorReporting')).toString(),
|
||||||
|
);
|
||||||
|
this.entryHref = url.href;
|
||||||
|
// Events steal 'this'
|
||||||
|
this.didFailLoad = _.bind(this.didFailLoad, this);
|
||||||
|
this.didGetResponseDetails = _.bind(this.didGetResponseDetails, this);
|
||||||
|
// Make a persistent electron session for the webview
|
||||||
|
this.session = electron.remote.session.fromPartition(ELECTRON_SESSION, {
|
||||||
|
// Disable the cache for the session such that new content shows up when refreshing
|
||||||
|
cache: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private static logWebViewMessage(event: electron.ConsoleMessageEvent) {
|
||||||
|
console.log('Message from SafeWebview:', event.message);
|
||||||
|
}
|
||||||
|
|
||||||
|
public render() {
|
||||||
|
const {
|
||||||
|
style = {
|
||||||
|
flex: this.state.shouldShow ? undefined : '0 1',
|
||||||
|
width: this.state.shouldShow ? undefined : '0',
|
||||||
|
height: this.state.shouldShow ? undefined : '0',
|
||||||
|
},
|
||||||
|
} = this.props;
|
||||||
|
return (
|
||||||
|
<webview
|
||||||
|
ref={this.webviewRef}
|
||||||
|
partition={ELECTRON_SESSION}
|
||||||
|
style={style}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the Webview events
|
||||||
|
public componentDidMount() {
|
||||||
|
// Events React is unaware of have to be handled manually
|
||||||
|
if (this.webviewRef.current !== null) {
|
||||||
|
this.webviewRef.current.addEventListener(
|
||||||
|
'did-fail-load',
|
||||||
|
this.didFailLoad,
|
||||||
|
);
|
||||||
|
this.webviewRef.current.addEventListener(
|
||||||
|
'new-window',
|
||||||
|
SafeWebview.newWindow,
|
||||||
|
);
|
||||||
|
this.webviewRef.current.addEventListener(
|
||||||
|
'console-message',
|
||||||
|
SafeWebview.logWebViewMessage,
|
||||||
|
);
|
||||||
|
this.session.webRequest.onCompleted(this.didGetResponseDetails);
|
||||||
|
// It's important that this comes after the partition setting, otherwise it will
|
||||||
|
// use another session and we can't change it without destroying the element again
|
||||||
|
this.webviewRef.current.src = this.entryHref;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove the Webview events
|
||||||
|
public componentWillUnmount() {
|
||||||
|
// Events that React is unaware of have to be handled manually
|
||||||
|
if (this.webviewRef.current !== null) {
|
||||||
|
this.webviewRef.current.removeEventListener(
|
||||||
|
'did-fail-load',
|
||||||
|
this.didFailLoad,
|
||||||
|
);
|
||||||
|
this.webviewRef.current.removeEventListener(
|
||||||
|
'new-window',
|
||||||
|
SafeWebview.newWindow,
|
||||||
|
);
|
||||||
|
this.webviewRef.current.removeEventListener(
|
||||||
|
'console-message',
|
||||||
|
SafeWebview.logWebViewMessage,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
this.session.webRequest.onCompleted(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the element state to hidden
|
||||||
|
public didFailLoad() {
|
||||||
|
this.setState({
|
||||||
|
shouldShow: false,
|
||||||
|
});
|
||||||
|
if (this.props.onWebviewShow) {
|
||||||
|
this.props.onWebviewShow(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the element state depending on the HTTP response code
|
||||||
|
public didGetResponseDetails(event: electron.OnCompletedListenerDetails) {
|
||||||
|
// This seems to pick up all requests related to the webview,
|
||||||
|
// only care about this event if it's a request for the main frame
|
||||||
|
if (event.resourceType === 'mainFrame') {
|
||||||
|
const HTTP_OK = 200;
|
||||||
|
analytics.logEvent('SafeWebview loaded', { event });
|
||||||
|
this.setState({
|
||||||
|
shouldShow: event.statusCode === HTTP_OK,
|
||||||
|
});
|
||||||
|
if (this.props.onWebviewShow) {
|
||||||
|
this.props.onWebviewShow(event.statusCode === HTTP_OK);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Open link in browser if it's opened as a 'foreground-tab'
|
||||||
|
public static async newWindow(event: electron.NewWindowEvent) {
|
||||||
|
const url = new window.URL(event.url);
|
||||||
|
if (
|
||||||
|
(url.protocol === 'http:' || url.protocol === 'https:') &&
|
||||||
|
event.disposition === 'foreground-tab' &&
|
||||||
|
// Don't open links if they're disabled by the env var
|
||||||
|
!(await settings.get('disableExternalLinks'))
|
||||||
|
) {
|
||||||
|
electron.shell.openExternal(url.href);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
156
lib/gui/app/components/settings/settings.tsx
Normal file
156
lib/gui/app/components/settings/settings.tsx
Normal file
@@ -0,0 +1,156 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2019 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import GithubSvg from '@fortawesome/fontawesome-free/svgs/brands/github.svg';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Box, Checkbox, Flex, TextWithCopy, Txt } from 'rendition';
|
||||||
|
|
||||||
|
import { version, packageType } from '../../../../../package.json';
|
||||||
|
import * as settings from '../../models/settings';
|
||||||
|
import * as analytics from '../../modules/analytics';
|
||||||
|
import { open as openExternal } from '../../os/open-external/services/open-external';
|
||||||
|
import { Modal } from '../../styled-components';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
import { etcherProInfo } from '../../utils/etcher-pro-specific';
|
||||||
|
|
||||||
|
interface Setting {
|
||||||
|
name: string;
|
||||||
|
label: string | JSX.Element;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getSettingsList(): Promise<Setting[]> {
|
||||||
|
const list: Setting[] = [
|
||||||
|
{
|
||||||
|
name: 'errorReporting',
|
||||||
|
label: i18next.t('settings.errorReporting'),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'autoBlockmapping',
|
||||||
|
label: i18next.t('settings.trimExtPartitions'),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
if (['appimage', 'nsis', 'dmg'].includes(packageType)) {
|
||||||
|
list.push({
|
||||||
|
name: 'updatesEnabled',
|
||||||
|
label: i18next.t('settings.autoUpdate'),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SettingsModalProps {
|
||||||
|
toggleModal: (value: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const EPInfo = etcherProInfo();
|
||||||
|
|
||||||
|
const InfoBox = (props: any) => (
|
||||||
|
<Box fontSize={14}>
|
||||||
|
<Txt>{props.label}</Txt>
|
||||||
|
<TextWithCopy code text={props.value} copy={props.value} />
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
|
||||||
|
export function SettingsModal({ toggleModal }: SettingsModalProps) {
|
||||||
|
const [settingsList, setCurrentSettingsList] = React.useState<Setting[]>([]);
|
||||||
|
React.useEffect(() => {
|
||||||
|
(async () => {
|
||||||
|
if (settingsList.length === 0) {
|
||||||
|
setCurrentSettingsList(await getSettingsList());
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
});
|
||||||
|
const [currentSettings, setCurrentSettings] = React.useState<
|
||||||
|
_.Dictionary<boolean>
|
||||||
|
>({});
|
||||||
|
React.useEffect(() => {
|
||||||
|
(async () => {
|
||||||
|
if (_.isEmpty(currentSettings)) {
|
||||||
|
setCurrentSettings(await settings.getAll());
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
});
|
||||||
|
|
||||||
|
const toggleSetting = async (setting: string) => {
|
||||||
|
const value = currentSettings[setting];
|
||||||
|
analytics.logEvent('Toggle setting', { setting, value });
|
||||||
|
await settings.set(setting, !value);
|
||||||
|
setCurrentSettings({
|
||||||
|
...currentSettings,
|
||||||
|
[setting]: !value,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
titleElement={
|
||||||
|
<Txt fontSize={24} mb={24}>
|
||||||
|
{i18next.t('settings.settings')}
|
||||||
|
</Txt>
|
||||||
|
}
|
||||||
|
done={() => toggleModal(false)}
|
||||||
|
>
|
||||||
|
<Flex flexDirection="column">
|
||||||
|
{settingsList.map((setting: Setting, i: number) => {
|
||||||
|
return (
|
||||||
|
<Flex key={setting.name} mb={14}>
|
||||||
|
<Checkbox
|
||||||
|
toggle
|
||||||
|
tabIndex={6 + i}
|
||||||
|
label={setting.label}
|
||||||
|
checked={currentSettings[setting.name]}
|
||||||
|
onChange={() => toggleSetting(setting.name)}
|
||||||
|
/>
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
{EPInfo !== undefined && (
|
||||||
|
<Flex flexDirection="column">
|
||||||
|
<Txt fontSize={24}>{i18next.t('settings.systemInformation')}</Txt>
|
||||||
|
{EPInfo.get_serial() === undefined ? (
|
||||||
|
<InfoBox label="UUID" value={EPInfo.uuid} />
|
||||||
|
) : (
|
||||||
|
<InfoBox label="Serial" value={EPInfo.get_serial()} />
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
)}
|
||||||
|
<Flex
|
||||||
|
mt={18}
|
||||||
|
alignItems="center"
|
||||||
|
color="#00aeef"
|
||||||
|
style={{
|
||||||
|
width: 'fit-content',
|
||||||
|
cursor: 'pointer',
|
||||||
|
fontSize: 14,
|
||||||
|
}}
|
||||||
|
onClick={() =>
|
||||||
|
openExternal(
|
||||||
|
'https://github.com/balena-io/etcher/blob/master/CHANGELOG.md',
|
||||||
|
)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<GithubSvg
|
||||||
|
height="1em"
|
||||||
|
fill="currentColor"
|
||||||
|
style={{ marginRight: 8 }}
|
||||||
|
/>
|
||||||
|
<Txt style={{ borderBottom: '1px solid #00aeef' }}>{version}</Txt>
|
||||||
|
</Flex>
|
||||||
|
</Flex>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
}
|
||||||
879
lib/gui/app/components/source-selector/source-selector.tsx
Normal file
879
lib/gui/app/components/source-selector/source-selector.tsx
Normal file
@@ -0,0 +1,879 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import CopySvg from '@fortawesome/fontawesome-free/svgs/solid/copy.svg';
|
||||||
|
import FileSvg from '@fortawesome/fontawesome-free/svgs/solid/file.svg';
|
||||||
|
import LinkSvg from '@fortawesome/fontawesome-free/svgs/solid/link.svg';
|
||||||
|
import ExclamationTriangleSvg from '@fortawesome/fontawesome-free/svgs/solid/exclamation-triangle.svg';
|
||||||
|
import ChevronDownSvg from '@fortawesome/fontawesome-free/svgs/solid/chevron-down.svg';
|
||||||
|
import ChevronRightSvg from '@fortawesome/fontawesome-free/svgs/solid/chevron-right.svg';
|
||||||
|
import { sourceDestination } from 'etcher-sdk';
|
||||||
|
import { ipcRenderer, IpcRendererEvent } from 'electron';
|
||||||
|
import * as _ from 'lodash';
|
||||||
|
import { GPTPartition, MBRPartition } from 'partitioninfo';
|
||||||
|
import * as path from 'path';
|
||||||
|
import * as prettyBytes from 'pretty-bytes';
|
||||||
|
import * as React from 'react';
|
||||||
|
import {
|
||||||
|
Flex,
|
||||||
|
ButtonProps,
|
||||||
|
Modal as SmallModal,
|
||||||
|
Txt,
|
||||||
|
Card as BaseCard,
|
||||||
|
Input,
|
||||||
|
Spinner,
|
||||||
|
Link,
|
||||||
|
} from 'rendition';
|
||||||
|
import styled from 'styled-components';
|
||||||
|
|
||||||
|
import * as errors from '../../../../shared/errors';
|
||||||
|
import * as messages from '../../../../shared/messages';
|
||||||
|
import * as supportedFormats from '../../../../shared/supported-formats';
|
||||||
|
import * as selectionState from '../../models/selection-state';
|
||||||
|
import { observe } from '../../models/store';
|
||||||
|
import * as analytics from '../../modules/analytics';
|
||||||
|
import * as exceptionReporter from '../../modules/exception-reporter';
|
||||||
|
import * as osDialog from '../../os/dialog';
|
||||||
|
import { replaceWindowsNetworkDriveLetter } from '../../os/windows-network-drives';
|
||||||
|
import {
|
||||||
|
ChangeButton,
|
||||||
|
DetailsText,
|
||||||
|
Modal,
|
||||||
|
StepButton,
|
||||||
|
StepNameButton,
|
||||||
|
ScrollableFlex,
|
||||||
|
} from '../../styled-components';
|
||||||
|
import { colors } from '../../theme';
|
||||||
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
||||||
|
import { SVGIcon } from '../svg-icon/svg-icon';
|
||||||
|
|
||||||
|
import ImageSvg from '../../../assets/image.svg';
|
||||||
|
import SrcSvg from '../../../assets/src.svg';
|
||||||
|
import { DriveSelector } from '../drive-selector/drive-selector';
|
||||||
|
import { DrivelistDrive } from '../../../../shared/drive-constraints';
|
||||||
|
import axios, { AxiosRequestConfig } from 'axios';
|
||||||
|
import { isJson } from '../../../../shared/utils';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
const recentUrlImagesKey = 'recentUrlImages';
|
||||||
|
|
||||||
|
function normalizeRecentUrlImages(urls: any[]): URL[] {
|
||||||
|
if (!Array.isArray(urls)) {
|
||||||
|
urls = [];
|
||||||
|
}
|
||||||
|
urls = urls
|
||||||
|
.map((url) => {
|
||||||
|
try {
|
||||||
|
return new URL(url);
|
||||||
|
} catch (error: any) {
|
||||||
|
// Invalid URL, skip
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.filter((url) => url !== undefined);
|
||||||
|
urls = _.uniqBy(urls, (url) => url.href);
|
||||||
|
return urls.slice(urls.length - 5);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getRecentUrlImages(): URL[] {
|
||||||
|
let urls = [];
|
||||||
|
try {
|
||||||
|
urls = JSON.parse(localStorage.getItem(recentUrlImagesKey) || '[]');
|
||||||
|
} catch {
|
||||||
|
// noop
|
||||||
|
}
|
||||||
|
return normalizeRecentUrlImages(urls);
|
||||||
|
}
|
||||||
|
|
||||||
|
function setRecentUrlImages(urls: URL[]) {
|
||||||
|
const normalized = normalizeRecentUrlImages(urls.map((url: URL) => url.href));
|
||||||
|
localStorage.setItem(recentUrlImagesKey, JSON.stringify(normalized));
|
||||||
|
}
|
||||||
|
|
||||||
|
const isURL = (imagePath: string) =>
|
||||||
|
imagePath.startsWith('https://') || imagePath.startsWith('http://');
|
||||||
|
|
||||||
|
const Card = styled(BaseCard)`
|
||||||
|
hr {
|
||||||
|
margin: 5px 0;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
// TODO move these styles to rendition
|
||||||
|
const ModalText = styled.p`
|
||||||
|
a {
|
||||||
|
color: rgb(0, 174, 239);
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: rgb(0, 139, 191);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
function getState() {
|
||||||
|
const image = selectionState.getImage();
|
||||||
|
return {
|
||||||
|
hasImage: selectionState.hasImage(),
|
||||||
|
imageName: image?.name,
|
||||||
|
imageSize: image?.size,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function isString(value: any): value is string {
|
||||||
|
return typeof value === 'string';
|
||||||
|
}
|
||||||
|
|
||||||
|
const URLSelector = ({
|
||||||
|
done,
|
||||||
|
cancel,
|
||||||
|
}: {
|
||||||
|
done: (imageURL: string, auth?: Authentication) => void;
|
||||||
|
cancel: () => void;
|
||||||
|
}) => {
|
||||||
|
const [imageURL, setImageURL] = React.useState('');
|
||||||
|
const [recentImages, setRecentImages] = React.useState<URL[]>([]);
|
||||||
|
const [loading, setLoading] = React.useState(false);
|
||||||
|
const [showBasicAuth, setShowBasicAuth] = React.useState(false);
|
||||||
|
const [username, setUsername] = React.useState('');
|
||||||
|
const [password, setPassword] = React.useState('');
|
||||||
|
React.useEffect(() => {
|
||||||
|
const fetchRecentUrlImages = async () => {
|
||||||
|
const recentUrlImages: URL[] = await getRecentUrlImages();
|
||||||
|
setRecentImages(recentUrlImages);
|
||||||
|
};
|
||||||
|
fetchRecentUrlImages();
|
||||||
|
}, []);
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
cancel={cancel}
|
||||||
|
primaryButtonProps={{
|
||||||
|
disabled: loading || !imageURL,
|
||||||
|
}}
|
||||||
|
action={loading ? <Spinner /> : i18next.t('ok')}
|
||||||
|
done={async () => {
|
||||||
|
setLoading(true);
|
||||||
|
const urlStrings = recentImages.map((url: URL) => url.href);
|
||||||
|
const normalizedRecentUrls = normalizeRecentUrlImages([
|
||||||
|
...urlStrings,
|
||||||
|
imageURL,
|
||||||
|
]);
|
||||||
|
setRecentUrlImages(normalizedRecentUrls);
|
||||||
|
const auth = username ? { username, password } : undefined;
|
||||||
|
await done(imageURL, auth);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Flex flexDirection="column">
|
||||||
|
<Flex mb={15} style={{ width: '100%' }} flexDirection="column">
|
||||||
|
<Txt mb="10px" fontSize="24px">
|
||||||
|
{i18next.t('source.useSourceURL')}
|
||||||
|
</Txt>
|
||||||
|
<Input
|
||||||
|
value={imageURL}
|
||||||
|
placeholder={i18next.t('source.enterValidURL')}
|
||||||
|
type="text"
|
||||||
|
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
|
setImageURL(evt.target.value)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<Link
|
||||||
|
mt={15}
|
||||||
|
mb={15}
|
||||||
|
fontSize="14px"
|
||||||
|
onClick={() => {
|
||||||
|
if (showBasicAuth) {
|
||||||
|
setUsername('');
|
||||||
|
setPassword('');
|
||||||
|
}
|
||||||
|
setShowBasicAuth(!showBasicAuth);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Flex alignItems="center">
|
||||||
|
{showBasicAuth && (
|
||||||
|
<ChevronDownSvg height="1em" fill="currentColor" />
|
||||||
|
)}
|
||||||
|
{!showBasicAuth && (
|
||||||
|
<ChevronRightSvg height="1em" fill="currentColor" />
|
||||||
|
)}
|
||||||
|
<Txt ml={8}>{i18next.t('source.auth')}</Txt>
|
||||||
|
</Flex>
|
||||||
|
</Link>
|
||||||
|
{showBasicAuth && (
|
||||||
|
<React.Fragment>
|
||||||
|
<Input
|
||||||
|
mb={15}
|
||||||
|
value={username}
|
||||||
|
placeholder={i18next.t('source.username')}
|
||||||
|
type="text"
|
||||||
|
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
|
setUsername(evt.target.value)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
value={password}
|
||||||
|
placeholder={i18next.t('source.password')}
|
||||||
|
type="password"
|
||||||
|
onChange={(evt: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
|
setPassword(evt.target.value)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</React.Fragment>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
{recentImages.length > 0 && (
|
||||||
|
<Flex flexDirection="column" height="78.6%">
|
||||||
|
<Txt fontSize={18}>Recent</Txt>
|
||||||
|
<ScrollableFlex flexDirection="column">
|
||||||
|
<Card
|
||||||
|
p="10px 15px"
|
||||||
|
rows={recentImages
|
||||||
|
.map((recent) => (
|
||||||
|
<Txt
|
||||||
|
key={recent.href}
|
||||||
|
onClick={() => {
|
||||||
|
setImageURL(recent.href);
|
||||||
|
}}
|
||||||
|
style={{
|
||||||
|
overflowWrap: 'break-word',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{recent.pathname.split('/').pop()} - {recent.href}
|
||||||
|
</Txt>
|
||||||
|
))
|
||||||
|
.reverse()}
|
||||||
|
/>
|
||||||
|
</ScrollableFlex>
|
||||||
|
</Flex>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
interface Flow {
|
||||||
|
icon?: JSX.Element;
|
||||||
|
onClick: (evt: React.MouseEvent) => void;
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FlowSelector = styled(
|
||||||
|
({ flow, ...props }: { flow: Flow } & ButtonProps) => (
|
||||||
|
<StepButton
|
||||||
|
plain={!props.primary}
|
||||||
|
primary={props.primary}
|
||||||
|
onClick={(evt: React.MouseEvent<Element, MouseEvent>) =>
|
||||||
|
flow.onClick(evt)
|
||||||
|
}
|
||||||
|
icon={flow.icon}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{flow.label}
|
||||||
|
</StepButton>
|
||||||
|
),
|
||||||
|
)`
|
||||||
|
border-radius: 24px;
|
||||||
|
color: rgba(255, 255, 255, 0.7);
|
||||||
|
|
||||||
|
:enabled:focus,
|
||||||
|
:enabled:focus svg {
|
||||||
|
color: ${colors.primary.foreground} !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
:enabled:hover {
|
||||||
|
background-color: ${colors.primary.background};
|
||||||
|
color: ${colors.primary.foreground};
|
||||||
|
font-weight: 600;
|
||||||
|
|
||||||
|
svg {
|
||||||
|
color: ${colors.primary.foreground} !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
export type Source =
|
||||||
|
| typeof sourceDestination.File
|
||||||
|
| typeof sourceDestination.BlockDevice
|
||||||
|
| typeof sourceDestination.Http;
|
||||||
|
|
||||||
|
export interface SourceMetadata extends sourceDestination.Metadata {
|
||||||
|
hasMBR?: boolean;
|
||||||
|
partitions?: MBRPartition[] | GPTPartition[];
|
||||||
|
path: string;
|
||||||
|
displayName: string;
|
||||||
|
description: string;
|
||||||
|
SourceType: Source;
|
||||||
|
drive?: DrivelistDrive;
|
||||||
|
extension?: string;
|
||||||
|
archiveExtension?: string;
|
||||||
|
auth?: Authentication;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SourceSelectorProps {
|
||||||
|
flashing: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SourceSelectorState {
|
||||||
|
hasImage: boolean;
|
||||||
|
imageName?: string;
|
||||||
|
imageSize?: number;
|
||||||
|
warning: { message: string; title: string | null } | null;
|
||||||
|
showImageDetails: boolean;
|
||||||
|
showURLSelector: boolean;
|
||||||
|
showDriveSelector: boolean;
|
||||||
|
defaultFlowActive: boolean;
|
||||||
|
imageSelectorOpen: boolean;
|
||||||
|
imageLoading: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Authentication {
|
||||||
|
username: string;
|
||||||
|
password: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class SourceSelector extends React.Component<
|
||||||
|
SourceSelectorProps,
|
||||||
|
SourceSelectorState
|
||||||
|
> {
|
||||||
|
private unsubscribe: (() => void) | undefined;
|
||||||
|
|
||||||
|
constructor(props: SourceSelectorProps) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
...getState(),
|
||||||
|
warning: null,
|
||||||
|
showImageDetails: false,
|
||||||
|
showURLSelector: false,
|
||||||
|
showDriveSelector: false,
|
||||||
|
defaultFlowActive: true,
|
||||||
|
imageSelectorOpen: false,
|
||||||
|
imageLoading: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Bind `this` since it's used in an event's callback
|
||||||
|
this.onSelectImage = this.onSelectImage.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public componentDidMount() {
|
||||||
|
this.unsubscribe = observe(() => {
|
||||||
|
this.setState(getState());
|
||||||
|
});
|
||||||
|
ipcRenderer.on('select-image', this.onSelectImage);
|
||||||
|
ipcRenderer.send('source-selector-ready');
|
||||||
|
}
|
||||||
|
|
||||||
|
public componentWillUnmount() {
|
||||||
|
this.unsubscribe?.();
|
||||||
|
ipcRenderer.removeListener('select-image', this.onSelectImage);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async onSelectImage(_event: IpcRendererEvent, imagePath: string) {
|
||||||
|
this.setState({ imageLoading: true });
|
||||||
|
await this.selectSource(
|
||||||
|
imagePath,
|
||||||
|
isURL(this.normalizeImagePath(imagePath))
|
||||||
|
? sourceDestination.Http
|
||||||
|
: sourceDestination.File,
|
||||||
|
).promise;
|
||||||
|
this.setState({ imageLoading: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
private async createSource(
|
||||||
|
selected: string,
|
||||||
|
SourceType: Source,
|
||||||
|
auth?: Authentication,
|
||||||
|
) {
|
||||||
|
try {
|
||||||
|
selected = await replaceWindowsNetworkDriveLetter(selected);
|
||||||
|
} catch (error: any) {
|
||||||
|
analytics.logException(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isJson(decodeURIComponent(selected))) {
|
||||||
|
const config: AxiosRequestConfig = JSON.parse(
|
||||||
|
decodeURIComponent(selected),
|
||||||
|
);
|
||||||
|
return new sourceDestination.Http({
|
||||||
|
url: config.url!,
|
||||||
|
axiosInstance: axios.create(_.omit(config, ['url'])),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (SourceType === sourceDestination.File) {
|
||||||
|
return new sourceDestination.File({
|
||||||
|
path: selected,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return new sourceDestination.Http({ url: selected, auth });
|
||||||
|
}
|
||||||
|
|
||||||
|
public normalizeImagePath(imgPath: string) {
|
||||||
|
const decodedPath = decodeURIComponent(imgPath);
|
||||||
|
if (isJson(decodedPath)) {
|
||||||
|
return JSON.parse(decodedPath).url ?? decodedPath;
|
||||||
|
}
|
||||||
|
return decodedPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
private reselectSource() {
|
||||||
|
analytics.logEvent('Reselect image', {
|
||||||
|
previousImage: selectionState.getImage(),
|
||||||
|
});
|
||||||
|
|
||||||
|
selectionState.deselectImage();
|
||||||
|
}
|
||||||
|
|
||||||
|
private selectSource(
|
||||||
|
selected: string | DrivelistDrive,
|
||||||
|
SourceType: Source,
|
||||||
|
auth?: Authentication,
|
||||||
|
): { promise: Promise<void>; cancel: () => void } {
|
||||||
|
let cancelled = false;
|
||||||
|
return {
|
||||||
|
cancel: () => {
|
||||||
|
cancelled = true;
|
||||||
|
},
|
||||||
|
promise: (async () => {
|
||||||
|
const sourcePath = isString(selected) ? selected : selected.device;
|
||||||
|
let source;
|
||||||
|
let metadata: SourceMetadata | undefined;
|
||||||
|
if (isString(selected)) {
|
||||||
|
if (
|
||||||
|
SourceType === sourceDestination.Http &&
|
||||||
|
!isURL(this.normalizeImagePath(selected))
|
||||||
|
) {
|
||||||
|
this.handleError(
|
||||||
|
i18next.t('source.unsupportedProtocol'),
|
||||||
|
selected,
|
||||||
|
messages.error.unsupportedProtocol(),
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (supportedFormats.looksLikeWindowsImage(selected)) {
|
||||||
|
analytics.logEvent('Possibly Windows image', { image: selected });
|
||||||
|
this.setState({
|
||||||
|
warning: {
|
||||||
|
message: messages.warning.looksLikeWindowsImage(),
|
||||||
|
title: i18next.t('source.windowsImage'),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
source = await this.createSource(selected, SourceType, auth);
|
||||||
|
|
||||||
|
if (cancelled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const innerSource = await source.getInnerSource();
|
||||||
|
if (cancelled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
metadata = await this.getMetadata(innerSource, selected);
|
||||||
|
if (cancelled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
metadata.SourceType = SourceType;
|
||||||
|
|
||||||
|
if (!metadata.hasMBR && this.state.warning === null) {
|
||||||
|
analytics.logEvent('Missing partition table', { metadata });
|
||||||
|
this.setState({
|
||||||
|
warning: {
|
||||||
|
message: messages.warning.missingPartitionTable(),
|
||||||
|
title: i18next.t('source.partitionTable'),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
this.handleError(
|
||||||
|
i18next.t('source.errorOpen'),
|
||||||
|
sourcePath,
|
||||||
|
messages.error.openSource(sourcePath, error.message),
|
||||||
|
error,
|
||||||
|
);
|
||||||
|
} finally {
|
||||||
|
try {
|
||||||
|
await source.close();
|
||||||
|
} catch (error: any) {
|
||||||
|
// Noop
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (selected.partitionTableType === null) {
|
||||||
|
analytics.logEvent('Missing partition table', { selected });
|
||||||
|
this.setState({
|
||||||
|
warning: {
|
||||||
|
message: messages.warning.driveMissingPartitionTable(),
|
||||||
|
title: i18next.t('source.partitionTable'),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
metadata = {
|
||||||
|
path: selected.device,
|
||||||
|
displayName: selected.displayName,
|
||||||
|
description: selected.displayName,
|
||||||
|
size: selected.size as SourceMetadata['size'],
|
||||||
|
SourceType: sourceDestination.BlockDevice,
|
||||||
|
drive: selected,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (metadata !== undefined) {
|
||||||
|
metadata.auth = auth;
|
||||||
|
selectionState.selectSource(metadata);
|
||||||
|
analytics.logEvent('Select image', {
|
||||||
|
// An easy way so we can quickly identify if we're making use of
|
||||||
|
// certain features without printing pages of text to DevTools.
|
||||||
|
image: {
|
||||||
|
...metadata,
|
||||||
|
logo: Boolean(metadata.logo),
|
||||||
|
blockMap: Boolean(metadata.blockMap),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleError(
|
||||||
|
title: string,
|
||||||
|
sourcePath: string,
|
||||||
|
description: string,
|
||||||
|
error?: Error,
|
||||||
|
) {
|
||||||
|
const imageError = errors.createUserError({
|
||||||
|
title,
|
||||||
|
description,
|
||||||
|
});
|
||||||
|
osDialog.showError(imageError);
|
||||||
|
if (error) {
|
||||||
|
analytics.logException(error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
analytics.logEvent(title, { path: sourcePath });
|
||||||
|
}
|
||||||
|
|
||||||
|
private async getMetadata(
|
||||||
|
source: sourceDestination.SourceDestination,
|
||||||
|
selected: string | DrivelistDrive,
|
||||||
|
) {
|
||||||
|
const metadata = (await source.getMetadata()) as SourceMetadata;
|
||||||
|
const partitionTable = await source.getPartitionTable();
|
||||||
|
if (partitionTable) {
|
||||||
|
metadata.hasMBR = true;
|
||||||
|
metadata.partitions = partitionTable.partitions;
|
||||||
|
} else {
|
||||||
|
metadata.hasMBR = false;
|
||||||
|
}
|
||||||
|
if (isString(selected)) {
|
||||||
|
metadata.extension = path.extname(selected).slice(1);
|
||||||
|
metadata.path = selected;
|
||||||
|
}
|
||||||
|
return metadata;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async openImageSelector() {
|
||||||
|
analytics.logEvent('Open image selector');
|
||||||
|
this.setState({ imageSelectorOpen: true });
|
||||||
|
|
||||||
|
try {
|
||||||
|
const imagePath = await osDialog.selectImage();
|
||||||
|
// Avoid analytics and selection state changes
|
||||||
|
// if no file was resolved from the dialog.
|
||||||
|
if (!imagePath) {
|
||||||
|
analytics.logEvent('Image selector closed');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
await this.selectSource(imagePath, sourceDestination.File).promise;
|
||||||
|
} catch (error: any) {
|
||||||
|
exceptionReporter.report(error);
|
||||||
|
} finally {
|
||||||
|
this.setState({ imageSelectorOpen: false });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async onDrop(event: React.DragEvent<HTMLDivElement>) {
|
||||||
|
const [file] = event.dataTransfer.files;
|
||||||
|
if (file) {
|
||||||
|
await this.selectSource(file.path, sourceDestination.File).promise;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private openURLSelector() {
|
||||||
|
analytics.logEvent('Open image URL selector');
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
showURLSelector: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private openDriveSelector() {
|
||||||
|
analytics.logEvent('Open drive selector');
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
showDriveSelector: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private onDragOver(event: React.DragEvent<HTMLDivElement>) {
|
||||||
|
// Needed to get onDrop events on div elements
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
private onDragEnter(event: React.DragEvent<HTMLDivElement>) {
|
||||||
|
// Needed to get onDrop events on div elements
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
private showSelectedImageDetails() {
|
||||||
|
analytics.logEvent('Show selected image tooltip', {
|
||||||
|
imagePath: selectionState.getImage()?.path,
|
||||||
|
});
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
showImageDetails: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private setDefaultFlowActive(defaultFlowActive: boolean) {
|
||||||
|
this.setState({ defaultFlowActive });
|
||||||
|
}
|
||||||
|
|
||||||
|
private closeModal() {
|
||||||
|
this.setState({
|
||||||
|
showDriveSelector: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO add a visual change when dragging a file over the selector
|
||||||
|
public render() {
|
||||||
|
const { flashing } = this.props;
|
||||||
|
const {
|
||||||
|
showImageDetails,
|
||||||
|
showURLSelector,
|
||||||
|
showDriveSelector,
|
||||||
|
imageLoading,
|
||||||
|
} = this.state;
|
||||||
|
const selectionImage = selectionState.getImage();
|
||||||
|
let image: SourceMetadata | DrivelistDrive =
|
||||||
|
selectionImage !== undefined ? selectionImage : ({} as SourceMetadata);
|
||||||
|
|
||||||
|
image = image.drive ?? image;
|
||||||
|
|
||||||
|
let cancelURLSelection = () => {
|
||||||
|
// noop
|
||||||
|
};
|
||||||
|
image.name = image.description || image.name;
|
||||||
|
const imagePath = image.path || image.displayName || '';
|
||||||
|
const imageBasename = path.basename(imagePath);
|
||||||
|
const imageName = image.name || '';
|
||||||
|
const imageSize = image.size;
|
||||||
|
const imageLogo = image.logo || '';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Flex
|
||||||
|
flexDirection="column"
|
||||||
|
alignItems="center"
|
||||||
|
onDrop={(evt: React.DragEvent<HTMLDivElement>) => this.onDrop(evt)}
|
||||||
|
onDragEnter={(evt: React.DragEvent<HTMLDivElement>) =>
|
||||||
|
this.onDragEnter(evt)
|
||||||
|
}
|
||||||
|
onDragOver={(evt: React.DragEvent<HTMLDivElement>) =>
|
||||||
|
this.onDragOver(evt)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<SVGIcon
|
||||||
|
contents={imageLogo}
|
||||||
|
fallback={ImageSvg}
|
||||||
|
style={{
|
||||||
|
marginBottom: 30,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{selectionImage !== undefined || imageLoading ? (
|
||||||
|
<>
|
||||||
|
<StepNameButton
|
||||||
|
plain
|
||||||
|
onClick={() => this.showSelectedImageDetails()}
|
||||||
|
tooltip={imageName || imageBasename}
|
||||||
|
>
|
||||||
|
<Spinner show={imageLoading}>
|
||||||
|
{middleEllipsis(imageName || imageBasename, 20)}
|
||||||
|
</Spinner>
|
||||||
|
</StepNameButton>
|
||||||
|
{!flashing && !imageLoading && (
|
||||||
|
<ChangeButton
|
||||||
|
plain
|
||||||
|
mb={14}
|
||||||
|
onClick={() => this.reselectSource()}
|
||||||
|
>
|
||||||
|
{i18next.t('cancel')}
|
||||||
|
</ChangeButton>
|
||||||
|
)}
|
||||||
|
{!_.isNil(imageSize) && !imageLoading && (
|
||||||
|
<DetailsText>{prettyBytes(imageSize)}</DetailsText>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<FlowSelector
|
||||||
|
disabled={this.state.imageSelectorOpen}
|
||||||
|
primary={this.state.defaultFlowActive}
|
||||||
|
key="Flash from file"
|
||||||
|
flow={{
|
||||||
|
onClick: () => this.openImageSelector(),
|
||||||
|
label: i18next.t('source.fromFile'),
|
||||||
|
icon: <FileSvg height="1em" fill="currentColor" />,
|
||||||
|
}}
|
||||||
|
onMouseEnter={() => this.setDefaultFlowActive(false)}
|
||||||
|
onMouseLeave={() => this.setDefaultFlowActive(true)}
|
||||||
|
/>
|
||||||
|
<FlowSelector
|
||||||
|
key="Flash from URL"
|
||||||
|
flow={{
|
||||||
|
onClick: () => this.openURLSelector(),
|
||||||
|
label: i18next.t('source.fromURL'),
|
||||||
|
icon: <LinkSvg height="1em" fill="currentColor" />,
|
||||||
|
}}
|
||||||
|
onMouseEnter={() => this.setDefaultFlowActive(false)}
|
||||||
|
onMouseLeave={() => this.setDefaultFlowActive(true)}
|
||||||
|
/>
|
||||||
|
<FlowSelector
|
||||||
|
key="Clone drive"
|
||||||
|
flow={{
|
||||||
|
onClick: () => this.openDriveSelector(),
|
||||||
|
label: i18next.t('source.clone'),
|
||||||
|
icon: <CopySvg height="1em" fill="currentColor" />,
|
||||||
|
}}
|
||||||
|
onMouseEnter={() => this.setDefaultFlowActive(false)}
|
||||||
|
onMouseLeave={() => this.setDefaultFlowActive(true)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
|
||||||
|
{this.state.warning != null && (
|
||||||
|
<SmallModal
|
||||||
|
style={{
|
||||||
|
boxShadow: '0 3px 7px rgba(0, 0, 0, 0.3)',
|
||||||
|
}}
|
||||||
|
titleElement={
|
||||||
|
<span>
|
||||||
|
<ExclamationTriangleSvg fill="#fca321" height="1em" />{' '}
|
||||||
|
<span>{this.state.warning.title}</span>
|
||||||
|
</span>
|
||||||
|
}
|
||||||
|
action={i18next.t('continue')}
|
||||||
|
cancel={() => {
|
||||||
|
this.setState({ warning: null });
|
||||||
|
this.reselectSource();
|
||||||
|
}}
|
||||||
|
done={() => {
|
||||||
|
this.setState({ warning: null });
|
||||||
|
}}
|
||||||
|
primaryButtonProps={{ warning: true, primary: false }}
|
||||||
|
>
|
||||||
|
<ModalText
|
||||||
|
dangerouslySetInnerHTML={{ __html: this.state.warning.message }}
|
||||||
|
/>
|
||||||
|
</SmallModal>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{showImageDetails && (
|
||||||
|
<SmallModal
|
||||||
|
title={i18next.t('source.image')}
|
||||||
|
done={() => {
|
||||||
|
this.setState({ showImageDetails: false });
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Txt.p>
|
||||||
|
<Txt.span bold>{i18next.t('source.name')}</Txt.span>
|
||||||
|
<Txt.span>{imageName || imageBasename}</Txt.span>
|
||||||
|
</Txt.p>
|
||||||
|
<Txt.p>
|
||||||
|
<Txt.span bold>{i18next.t('source.path')}</Txt.span>
|
||||||
|
<Txt.span>{imagePath}</Txt.span>
|
||||||
|
</Txt.p>
|
||||||
|
</SmallModal>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{showURLSelector && (
|
||||||
|
<URLSelector
|
||||||
|
cancel={() => {
|
||||||
|
cancelURLSelection();
|
||||||
|
this.setState({
|
||||||
|
showURLSelector: false,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
done={async (imageURL: string, auth?: Authentication) => {
|
||||||
|
// Avoid analytics and selection state changes
|
||||||
|
// if no file was resolved from the dialog.
|
||||||
|
if (!imageURL) {
|
||||||
|
analytics.logEvent('URL selector closed');
|
||||||
|
} else {
|
||||||
|
let promise;
|
||||||
|
({ promise, cancel: cancelURLSelection } = this.selectSource(
|
||||||
|
imageURL,
|
||||||
|
sourceDestination.Http,
|
||||||
|
auth,
|
||||||
|
));
|
||||||
|
await promise;
|
||||||
|
}
|
||||||
|
this.setState({
|
||||||
|
showURLSelector: false,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{showDriveSelector && (
|
||||||
|
<DriveSelector
|
||||||
|
write={false}
|
||||||
|
multipleSelection={false}
|
||||||
|
titleLabel={i18next.t('source.selectSource')}
|
||||||
|
emptyListLabel={i18next.t('source.plugSource')}
|
||||||
|
emptyListIcon={<SrcSvg width="40px" />}
|
||||||
|
cancel={(originalList) => {
|
||||||
|
if (originalList.length) {
|
||||||
|
const originalSource = originalList[0];
|
||||||
|
if (selectionImage?.drive?.device !== originalSource.device) {
|
||||||
|
this.selectSource(
|
||||||
|
originalSource,
|
||||||
|
sourceDestination.BlockDevice,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
selectionState.deselectImage();
|
||||||
|
}
|
||||||
|
this.closeModal();
|
||||||
|
}}
|
||||||
|
done={() => this.closeModal()}
|
||||||
|
onSelect={(drive) => {
|
||||||
|
if (drive) {
|
||||||
|
if (
|
||||||
|
selectionState.getImage()?.drive?.device === drive?.device
|
||||||
|
) {
|
||||||
|
return selectionState.deselectImage();
|
||||||
|
}
|
||||||
|
this.selectSource(drive, sourceDestination.BlockDevice);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/* eslint-disable jsdoc/require-example */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.SVGIcon
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const react2angular = require('react2angular').react2angular
|
|
||||||
|
|
||||||
const MODULE_NAME = 'Etcher.Components.SVGIcon'
|
|
||||||
const angularSVGIcon = angular.module(MODULE_NAME, [])
|
|
||||||
|
|
||||||
angularSVGIcon.component('svgIcon', react2angular(require('./svg-icon.jsx')))
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
|
|
||||||
svg-icon {
|
|
||||||
display: inline-block;
|
|
||||||
|
|
||||||
img {
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,176 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2018 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.SVGIcon
|
|
||||||
*/
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
const react = require('react')
|
|
||||||
const propTypes = require('prop-types')
|
|
||||||
const path = require('path')
|
|
||||||
const fs = require('fs')
|
|
||||||
const analytics = require('../../modules/analytics')
|
|
||||||
const domParser = new window.DOMParser()
|
|
||||||
|
|
||||||
const DEFAULT_SIZE = '40px'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Try to parse SVG contents and return it data encoded
|
|
||||||
*
|
|
||||||
* @param {String} contents - SVG XML contents
|
|
||||||
* @returns {String|null}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* const encodedSVG = tryParseSVGContents('<svg><path></path></svg>')
|
|
||||||
*
|
|
||||||
* img.src = encodedSVG
|
|
||||||
*/
|
|
||||||
const tryParseSVGContents = (contents) => {
|
|
||||||
const doc = domParser.parseFromString(contents, 'image/svg+xml')
|
|
||||||
const parserError = doc.querySelector('parsererror')
|
|
||||||
const svg = doc.querySelector('svg')
|
|
||||||
|
|
||||||
if (!parserError && svg) {
|
|
||||||
return `data:image/svg+xml,${encodeURIComponent(svg.outerHTML)}`
|
|
||||||
}
|
|
||||||
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
/* eslint-disable jsdoc/require-example */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary SVG element that takes both filepaths and file contents
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
class SVGIcon extends react.Component {
|
|
||||||
/**
|
|
||||||
* @summary Render the SVG
|
|
||||||
* @returns {react.Element}
|
|
||||||
*/
|
|
||||||
render () {
|
|
||||||
// __dirname behaves strangely inside a Webpack bundle,
|
|
||||||
// so we need to provide different base directories
|
|
||||||
// depending on whether __dirname is absolute or not,
|
|
||||||
// which helps detecting a Webpack bundle.
|
|
||||||
// We use global.__dirname inside a Webpack bundle since
|
|
||||||
// that's the only way to get the "real" __dirname.
|
|
||||||
const baseDirectory = path.isAbsolute(__dirname)
|
|
||||||
? path.join(__dirname, '..')
|
|
||||||
// eslint-disable-next-line no-underscore-dangle
|
|
||||||
: global.__dirname
|
|
||||||
|
|
||||||
let svgData = ''
|
|
||||||
|
|
||||||
_.find(this.props.contents, (content) => {
|
|
||||||
const attempt = tryParseSVGContents(content)
|
|
||||||
|
|
||||||
if (attempt) {
|
|
||||||
svgData = attempt
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
})
|
|
||||||
|
|
||||||
if (!svgData) {
|
|
||||||
_.find(this.props.paths, (relativePath) => {
|
|
||||||
// This means the path to the icon should be
|
|
||||||
// relative to *this directory*.
|
|
||||||
// TODO: There might be a way to compute the path
|
|
||||||
// relatively to the `index.html`.
|
|
||||||
const imagePath = path.join(baseDirectory, 'assets', relativePath)
|
|
||||||
|
|
||||||
const contents = _.attempt(() => {
|
|
||||||
return fs.readFileSync(imagePath, {
|
|
||||||
encoding: 'utf8'
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
if (_.isError(contents)) {
|
|
||||||
analytics.logException(contents)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
const parsed = _.attempt(tryParseSVGContents, contents)
|
|
||||||
|
|
||||||
if (parsed) {
|
|
||||||
svgData = parsed
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const width = this.props.width || DEFAULT_SIZE
|
|
||||||
const height = this.props.height || DEFAULT_SIZE
|
|
||||||
|
|
||||||
return react.createElement('img', {
|
|
||||||
className: 'svg-icon',
|
|
||||||
style: {
|
|
||||||
width,
|
|
||||||
height
|
|
||||||
},
|
|
||||||
src: svgData,
|
|
||||||
disabled: this.props.disabled
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Cause a re-render due to changed element properties
|
|
||||||
* @param {Object} nextProps - the new properties
|
|
||||||
*/
|
|
||||||
componentWillReceiveProps (nextProps) {
|
|
||||||
// This will update the element if the properties change
|
|
||||||
this.setState(nextProps)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
SVGIcon.propTypes = {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Paths to SVG files to be tried in succession if any fails
|
|
||||||
*/
|
|
||||||
paths: propTypes.array,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary List of embedded SVG contents to be tried in succession if any fails
|
|
||||||
*/
|
|
||||||
contents: propTypes.array,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary SVG image width unit
|
|
||||||
*/
|
|
||||||
width: propTypes.string,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary SVG image height unit
|
|
||||||
*/
|
|
||||||
height: propTypes.string,
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Should the element visually appear grayed out and disabled?
|
|
||||||
*/
|
|
||||||
disabled: propTypes.bool
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = SVGIcon
|
|
||||||
74
lib/gui/app/components/svg-icon/svg-icon.tsx
Normal file
74
lib/gui/app/components/svg-icon/svg-icon.tsx
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2018 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
|
||||||
|
const domParser = new window.DOMParser();
|
||||||
|
|
||||||
|
const DEFAULT_SIZE = '40px';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary Try to parse SVG contents and return it data encoded
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function tryParseSVGContents(contents?: string): string | undefined {
|
||||||
|
if (contents === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const doc = domParser.parseFromString(contents, 'image/svg+xml');
|
||||||
|
const parserError = doc.querySelector('parsererror');
|
||||||
|
const svg = doc.querySelector('svg');
|
||||||
|
if (!parserError && svg) {
|
||||||
|
return `data:image/svg+xml,${encodeURIComponent(svg.outerHTML)}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SVGIconProps {
|
||||||
|
// Optional string representing the SVG contents to be tried
|
||||||
|
contents?: string;
|
||||||
|
// Fallback SVG element to show if `contents` is invalid/undefined
|
||||||
|
fallback: React.FunctionComponent<React.SVGProps<HTMLOrSVGElement>>;
|
||||||
|
// SVG image width unit
|
||||||
|
width?: string;
|
||||||
|
// SVG image height unit
|
||||||
|
height?: string;
|
||||||
|
// Should the element visually appear grayed out and disabled?
|
||||||
|
disabled?: boolean;
|
||||||
|
style?: React.CSSProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @summary SVG element that takes file contents
|
||||||
|
*/
|
||||||
|
export class SVGIcon extends React.PureComponent<SVGIconProps> {
|
||||||
|
public render() {
|
||||||
|
const svgData = tryParseSVGContents(this.props.contents);
|
||||||
|
const { width, height, style = {} } = this.props;
|
||||||
|
style.width = width || DEFAULT_SIZE;
|
||||||
|
style.height = height || DEFAULT_SIZE;
|
||||||
|
if (svgData !== undefined) {
|
||||||
|
return (
|
||||||
|
<img
|
||||||
|
className={this.props.disabled ? 'disabled' : ''}
|
||||||
|
style={style}
|
||||||
|
src={svgData}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const { fallback: FallbackSVG } = this.props;
|
||||||
|
return <FallbackSVG style={style} />;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,158 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2019 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ExclamationTriangleSvg from '@fortawesome/fontawesome-free/svgs/solid/exclamation-triangle.svg';
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex, FlexProps, Txt } from 'rendition';
|
||||||
|
|
||||||
|
import {
|
||||||
|
getDriveImageCompatibilityStatuses,
|
||||||
|
DriveStatus,
|
||||||
|
} from '../../../../shared/drive-constraints';
|
||||||
|
import { compatibility, warning } from '../../../../shared/messages';
|
||||||
|
import * as prettyBytes from 'pretty-bytes';
|
||||||
|
import { getImage, getSelectedDrives } from '../../models/selection-state';
|
||||||
|
import {
|
||||||
|
ChangeButton,
|
||||||
|
DetailsText,
|
||||||
|
StepButton,
|
||||||
|
StepNameButton,
|
||||||
|
} from '../../styled-components';
|
||||||
|
import { middleEllipsis } from '../../utils/middle-ellipsis';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
interface TargetSelectorProps {
|
||||||
|
targets: any[];
|
||||||
|
disabled: boolean;
|
||||||
|
openDriveSelector: () => void;
|
||||||
|
reselectDrive: () => void;
|
||||||
|
flashing: boolean;
|
||||||
|
show: boolean;
|
||||||
|
tooltip: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDriveWarning(status: DriveStatus) {
|
||||||
|
switch (status.message) {
|
||||||
|
case compatibility.containsImage():
|
||||||
|
return warning.sourceDrive();
|
||||||
|
case compatibility.largeDrive():
|
||||||
|
return warning.largeDriveSize();
|
||||||
|
case compatibility.system():
|
||||||
|
return warning.systemDrive();
|
||||||
|
default:
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const DriveCompatibilityWarning = ({
|
||||||
|
warnings,
|
||||||
|
...props
|
||||||
|
}: {
|
||||||
|
warnings: string[];
|
||||||
|
} & FlexProps) => {
|
||||||
|
const systemDrive = warnings.find(
|
||||||
|
(message) => message === warning.systemDrive(),
|
||||||
|
);
|
||||||
|
return (
|
||||||
|
<Flex tooltip={warnings.join(', ')} {...props}>
|
||||||
|
<ExclamationTriangleSvg
|
||||||
|
fill={systemDrive ? '#fca321' : '#8f9297'}
|
||||||
|
height="1em"
|
||||||
|
/>
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export function TargetSelectorButton(props: TargetSelectorProps) {
|
||||||
|
const targets = getSelectedDrives();
|
||||||
|
|
||||||
|
if (targets.length === 1) {
|
||||||
|
const target = targets[0];
|
||||||
|
const warnings = getDriveImageCompatibilityStatuses(
|
||||||
|
target,
|
||||||
|
getImage(),
|
||||||
|
true,
|
||||||
|
).map(getDriveWarning);
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<StepNameButton plain tooltip={props.tooltip}>
|
||||||
|
{warnings.length > 0 && (
|
||||||
|
<DriveCompatibilityWarning warnings={warnings} mr={2} />
|
||||||
|
)}
|
||||||
|
{middleEllipsis(target.description, 20)}
|
||||||
|
</StepNameButton>
|
||||||
|
{!props.flashing && (
|
||||||
|
<ChangeButton plain mb={14} onClick={props.reselectDrive}>
|
||||||
|
{i18next.t('target.change')}
|
||||||
|
</ChangeButton>
|
||||||
|
)}
|
||||||
|
{target.size != null && (
|
||||||
|
<DetailsText>{prettyBytes(target.size)}</DetailsText>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (targets.length > 1) {
|
||||||
|
const targetsTemplate = [];
|
||||||
|
for (const target of targets) {
|
||||||
|
const warnings = getDriveImageCompatibilityStatuses(
|
||||||
|
target,
|
||||||
|
getImage(),
|
||||||
|
true,
|
||||||
|
).map(getDriveWarning);
|
||||||
|
targetsTemplate.push(
|
||||||
|
<DetailsText
|
||||||
|
key={target.device}
|
||||||
|
tooltip={`${target.description} ${target.displayName} ${
|
||||||
|
target.size != null ? prettyBytes(target.size) : ''
|
||||||
|
}`}
|
||||||
|
px={21}
|
||||||
|
>
|
||||||
|
{warnings.length > 0 ? (
|
||||||
|
<DriveCompatibilityWarning warnings={warnings} mr={2} />
|
||||||
|
) : null}
|
||||||
|
<Txt mr={2}>{middleEllipsis(target.description, 14)}</Txt>
|
||||||
|
{target.size != null && <Txt>{prettyBytes(target.size)}</Txt>}
|
||||||
|
</DetailsText>,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<StepNameButton plain tooltip={props.tooltip}>
|
||||||
|
{targets.length} {i18next.t('target.targets')}
|
||||||
|
</StepNameButton>
|
||||||
|
{!props.flashing && (
|
||||||
|
<ChangeButton plain onClick={props.reselectDrive} mb={14}>
|
||||||
|
{i18next.t('target.change')}
|
||||||
|
</ChangeButton>
|
||||||
|
)}
|
||||||
|
{targetsTemplate}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<StepButton
|
||||||
|
primary
|
||||||
|
tabIndex={targets.length > 0 ? -1 : 2}
|
||||||
|
disabled={props.disabled}
|
||||||
|
onClick={props.openDriveSelector}
|
||||||
|
>
|
||||||
|
{i18next.t('target.selectTarget')}
|
||||||
|
</StepButton>
|
||||||
|
);
|
||||||
|
}
|
||||||
194
lib/gui/app/components/target-selector/target-selector.tsx
Normal file
194
lib/gui/app/components/target-selector/target-selector.tsx
Normal file
@@ -0,0 +1,194 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 balena.io
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as React from 'react';
|
||||||
|
import { Flex, Txt } from 'rendition';
|
||||||
|
|
||||||
|
import {
|
||||||
|
DriveSelector,
|
||||||
|
DriveSelectorProps,
|
||||||
|
} from '../drive-selector/drive-selector';
|
||||||
|
import {
|
||||||
|
isDriveSelected,
|
||||||
|
getImage,
|
||||||
|
getSelectedDrives,
|
||||||
|
deselectDrive,
|
||||||
|
selectDrive,
|
||||||
|
deselectAllDrives,
|
||||||
|
} from '../../models/selection-state';
|
||||||
|
import { observe } from '../../models/store';
|
||||||
|
import * as analytics from '../../modules/analytics';
|
||||||
|
import { TargetSelectorButton } from './target-selector-button';
|
||||||
|
|
||||||
|
import TgtSvg from '../../../assets/tgt.svg';
|
||||||
|
import DriveSvg from '../../../assets/drive.svg';
|
||||||
|
import { warning } from '../../../../shared/messages';
|
||||||
|
import { DrivelistDrive } from '../../../../shared/drive-constraints';
|
||||||
|
import * as i18next from 'i18next';
|
||||||
|
|
||||||
|
export const getDriveListLabel = () => {
|
||||||
|
return getSelectedDrives()
|
||||||
|
.map((drive: any) => {
|
||||||
|
return `${drive.description} (${drive.displayName})`;
|
||||||
|
})
|
||||||
|
.join('\n');
|
||||||
|
};
|
||||||
|
|
||||||
|
const getDriveSelectionStateSlice = () => ({
|
||||||
|
driveListLabel: getDriveListLabel(),
|
||||||
|
targets: getSelectedDrives(),
|
||||||
|
image: getImage(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const TargetSelectorModal = (
|
||||||
|
props: Omit<
|
||||||
|
DriveSelectorProps,
|
||||||
|
'titleLabel' | 'emptyListLabel' | 'multipleSelection' | 'emptyListIcon'
|
||||||
|
>,
|
||||||
|
) => (
|
||||||
|
<DriveSelector
|
||||||
|
multipleSelection={true}
|
||||||
|
titleLabel={i18next.t('target.selectTarget')}
|
||||||
|
emptyListLabel={i18next.t('target.plugTarget')}
|
||||||
|
emptyListIcon={<TgtSvg width="40px" />}
|
||||||
|
showWarnings={true}
|
||||||
|
selectedList={getSelectedDrives()}
|
||||||
|
updateSelectedList={getSelectedDrives}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
export const selectAllTargets = (modalTargets: DrivelistDrive[]) => {
|
||||||
|
const selectedDrivesFromState = getSelectedDrives();
|
||||||
|
const deselected = selectedDrivesFromState.filter(
|
||||||
|
(drive) =>
|
||||||
|
!modalTargets.find((modalTarget) => modalTarget.device === drive.device),
|
||||||
|
);
|
||||||
|
// deselect drives
|
||||||
|
deselected.forEach((drive) => {
|
||||||
|
analytics.logEvent('Toggle drive', {
|
||||||
|
drive,
|
||||||
|
previouslySelected: true,
|
||||||
|
});
|
||||||
|
deselectDrive(drive.device);
|
||||||
|
});
|
||||||
|
// select drives
|
||||||
|
modalTargets.forEach((drive) => {
|
||||||
|
// Don't send events for drives that were already selected
|
||||||
|
if (!isDriveSelected(drive.device)) {
|
||||||
|
analytics.logEvent('Toggle drive', {
|
||||||
|
drive,
|
||||||
|
previouslySelected: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
selectDrive(drive.device);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
interface TargetSelectorProps {
|
||||||
|
disabled: boolean;
|
||||||
|
hasDrive: boolean;
|
||||||
|
flashing: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const TargetSelector = ({
|
||||||
|
disabled,
|
||||||
|
hasDrive,
|
||||||
|
flashing,
|
||||||
|
}: TargetSelectorProps) => {
|
||||||
|
// TODO: inject these from redux-connector
|
||||||
|
const [{ driveListLabel, targets }, setStateSlice] = React.useState(
|
||||||
|
getDriveSelectionStateSlice(),
|
||||||
|
);
|
||||||
|
const [showTargetSelectorModal, setShowTargetSelectorModal] =
|
||||||
|
React.useState(false);
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
return observe(() => {
|
||||||
|
setStateSlice(getDriveSelectionStateSlice());
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const hasSystemDrives = targets.some((target) => target.isSystem);
|
||||||
|
return (
|
||||||
|
<Flex flexDirection="column" alignItems="center">
|
||||||
|
<DriveSvg
|
||||||
|
className={disabled ? 'disabled' : ''}
|
||||||
|
width="40px"
|
||||||
|
style={{
|
||||||
|
marginBottom: 30,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<TargetSelectorButton
|
||||||
|
disabled={disabled}
|
||||||
|
show={!hasDrive}
|
||||||
|
tooltip={driveListLabel}
|
||||||
|
openDriveSelector={() => {
|
||||||
|
setShowTargetSelectorModal(true);
|
||||||
|
}}
|
||||||
|
reselectDrive={() => {
|
||||||
|
analytics.logEvent('Reselect drive');
|
||||||
|
setShowTargetSelectorModal(true);
|
||||||
|
}}
|
||||||
|
flashing={flashing}
|
||||||
|
targets={targets}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{hasSystemDrives ? (
|
||||||
|
<Txt
|
||||||
|
color="#fca321"
|
||||||
|
style={{
|
||||||
|
position: 'absolute',
|
||||||
|
bottom: '25px',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Warning: {warning.systemDrive()}
|
||||||
|
</Txt>
|
||||||
|
) : null}
|
||||||
|
|
||||||
|
{showTargetSelectorModal && (
|
||||||
|
<TargetSelectorModal
|
||||||
|
write={true}
|
||||||
|
cancel={(originalList) => {
|
||||||
|
if (originalList.length) {
|
||||||
|
selectAllTargets(originalList);
|
||||||
|
} else {
|
||||||
|
deselectAllDrives();
|
||||||
|
}
|
||||||
|
setShowTargetSelectorModal(false);
|
||||||
|
}}
|
||||||
|
done={(modalTargets) => {
|
||||||
|
if (modalTargets.length === 0) {
|
||||||
|
deselectAllDrives();
|
||||||
|
}
|
||||||
|
setShowTargetSelectorModal(false);
|
||||||
|
}}
|
||||||
|
onSelect={(drive) => {
|
||||||
|
if (
|
||||||
|
getSelectedDrives().find(
|
||||||
|
(selectedDrive) => selectedDrive.device === drive.device,
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return deselectDrive(drive.device);
|
||||||
|
}
|
||||||
|
selectDrive(drive.device);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Flex>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
|
|
||||||
module.exports = function (ModalService) {
|
|
||||||
/**
|
|
||||||
* @summary Open the tooltip modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} options - tooltip options
|
|
||||||
* @param {String} options.title - tooltip title
|
|
||||||
* @param {String} options.message - tooltip message
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* TooltipModalService.show({
|
|
||||||
* title: 'Important tooltip',
|
|
||||||
* message: 'Tooltip contents'
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.show = (options) => {
|
|
||||||
return ModalService.open({
|
|
||||||
name: 'tooltip',
|
|
||||||
template: require('../templates/tooltip-modal.tpl.html'),
|
|
||||||
controller: 'TooltipModalController as modal',
|
|
||||||
size: 'tooltip-modal',
|
|
||||||
resolve: {
|
|
||||||
tooltipData: _.constant(options)
|
|
||||||
}
|
|
||||||
}).result
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,23 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
.modal-tooltip-modal .modal-body {
|
|
||||||
text-align: center;
|
|
||||||
margin: 15px;
|
|
||||||
color: $palette-theme-light-foreground;
|
|
||||||
background-color: darken($palette-theme-light-background, 5%);
|
|
||||||
word-wrap: break-word;
|
|
||||||
}
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
<div class="modal-header">
|
|
||||||
<h4 class="modal-title">{{ ::modal.data.title }}</h4>
|
|
||||||
<button class="close" ng-click="modal.closeModal()">×</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-body">{{ ::modal.data.message }}</div>
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.TooltipModal
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const MODULE_NAME = 'Etcher.Components.TooltipModal'
|
|
||||||
const TooltipModal = angular.module(MODULE_NAME, [
|
|
||||||
require('../modal/modal')
|
|
||||||
])
|
|
||||||
|
|
||||||
TooltipModal.controller('TooltipModalController', require('./controllers/tooltip-modal'))
|
|
||||||
TooltipModal.service('TooltipModalService', require('./services/tooltip-modal'))
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
module.exports = function ($uibModalInstance, options) {
|
|
||||||
/**
|
|
||||||
* @summary Modal options
|
|
||||||
* @type {Object}
|
|
||||||
* @public
|
|
||||||
*/
|
|
||||||
this.options = options
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Reject the warning prompt
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* WarningModalController.reject();
|
|
||||||
*/
|
|
||||||
this.reject = () => {
|
|
||||||
$uibModalInstance.close(false)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @summary Accept the warning prompt
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* WarningModalController.accept();
|
|
||||||
*/
|
|
||||||
this.accept = () => {
|
|
||||||
$uibModalInstance.close(true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
const _ = require('lodash')
|
|
||||||
|
|
||||||
module.exports = function ($sce, ModalService) {
|
|
||||||
/**
|
|
||||||
* @summary Display the warning modal
|
|
||||||
* @function
|
|
||||||
* @public
|
|
||||||
*
|
|
||||||
* @param {Object} options - options
|
|
||||||
* @param {String} options.description - danger message
|
|
||||||
* @param {String} options.confirmationLabel - confirmation button text
|
|
||||||
* @param {String} options.rejectionLabel - rejection button text
|
|
||||||
* @fulfil {Boolean} - whether the user accepted or rejected the warning
|
|
||||||
* @returns {Promise}
|
|
||||||
*
|
|
||||||
* @example
|
|
||||||
* WarningModalService.display({
|
|
||||||
* description: 'Don\'t do this!',
|
|
||||||
* confirmationLabel: 'Yes, continue!'
|
|
||||||
* });
|
|
||||||
*/
|
|
||||||
this.display = (options = {}) => {
|
|
||||||
options.description = $sce.trustAsHtml(options.description)
|
|
||||||
return ModalService.open({
|
|
||||||
name: 'warning',
|
|
||||||
template: require('../templates/warning-modal.tpl.html'),
|
|
||||||
controller: 'WarningModalController as modal',
|
|
||||||
size: 'warning-modal',
|
|
||||||
resolve: {
|
|
||||||
options: _.constant(options)
|
|
||||||
}
|
|
||||||
}).result
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
<div class="modal-header">
|
|
||||||
<h4 class="modal-title">
|
|
||||||
<span class="glyphicon glyphicon-exclamation-sign"></span>
|
|
||||||
<span>Attention</span>
|
|
||||||
</h4>
|
|
||||||
<button class="close"
|
|
||||||
tabindex="11"
|
|
||||||
ng-click="modal.reject()">×</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-body">
|
|
||||||
<p ng-bind-html="modal.options.description"></p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="modal-footer">
|
|
||||||
<div class="modal-menu">
|
|
||||||
<button class="button button-danger button-block"
|
|
||||||
tabindex="13"
|
|
||||||
ng-click="modal.accept()">{{ ::modal.options.confirmationLabel }}</button>
|
|
||||||
<button ng-if="modal.options.rejectionLabel" class="button button-block"
|
|
||||||
tabindex="12"
|
|
||||||
ng-click="modal.reject()">{{ ::modal.options.rejectionLabel }}</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 resin.io
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
'use strict'
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @module Etcher.Components.WarningModal
|
|
||||||
*/
|
|
||||||
|
|
||||||
const angular = require('angular')
|
|
||||||
const MODULE_NAME = 'Etcher.Components.WarningModal'
|
|
||||||
const WarningModal = angular.module(MODULE_NAME, [
|
|
||||||
require('../modal/modal')
|
|
||||||
])
|
|
||||||
|
|
||||||
WarningModal.controller('WarningModalController', require('./controllers/warning-modal'))
|
|
||||||
WarningModal.service('WarningModalService', require('./services/warning-modal'))
|
|
||||||
|
|
||||||
module.exports = MODULE_NAME
|
|
||||||
BIN
lib/gui/app/css/fonts/SourceSansPro-Regular.ttf
Normal file
BIN
lib/gui/app/css/fonts/SourceSansPro-Regular.ttf
Normal file
Binary file not shown.
BIN
lib/gui/app/css/fonts/SourceSansPro-SemiBold.ttf
Normal file
BIN
lib/gui/app/css/fonts/SourceSansPro-SemiBold.ttf
Normal file
Binary file not shown.
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright 2016 resin.io
|
* Copyright 2016 balena.io
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
@@ -14,40 +14,36 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* Prevent text selection */
|
@font-face {
|
||||||
body {
|
font-family: "SourceSansPro";
|
||||||
-webkit-user-select: none;
|
src: url("./fonts/SourceSansPro-Regular.ttf") format("truetype");
|
||||||
|
font-weight: 500;
|
||||||
|
font-style: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@font-face {
|
||||||
/* Allow window to be dragged from anywhere */
|
font-family: "SourceSansPro";
|
||||||
body > header {
|
src: url("./fonts/SourceSansPro-SemiBold.ttf") format("truetype");
|
||||||
-webkit-app-region: drag;
|
font-weight: 600;
|
||||||
|
font-style: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
.modal-body {
|
|
||||||
-webkit-app-region: no-drag;
|
|
||||||
}
|
|
||||||
|
|
||||||
button,
|
|
||||||
a,
|
|
||||||
input {
|
|
||||||
-webkit-app-region: no-drag;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Prevent WebView bounce effect in OS X */
|
|
||||||
html,
|
html,
|
||||||
body {
|
body {
|
||||||
|
margin: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
|
/* Prevent white flash when running application */
|
||||||
|
background-color: #4d5057;
|
||||||
|
|
||||||
|
/* Prevent WebView bounce effect in OS X */
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
html {
|
/* Prevent text selection */
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
body {
|
body {
|
||||||
overflow: hidden;
|
-webkit-user-select: none;
|
||||||
-webkit-overflow-scrolling: touch;
|
-webkit-overflow-scrolling: touch;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -55,11 +51,16 @@ body {
|
|||||||
a:focus,
|
a:focus,
|
||||||
input:focus,
|
input:focus,
|
||||||
button:focus,
|
button:focus,
|
||||||
[tabindex]:focus {
|
[tabindex]:focus,
|
||||||
|
input[type="checkbox"] + div {
|
||||||
outline: none !important;
|
outline: none !important;
|
||||||
|
box-shadow: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Titles don't have margins on desktop apps */
|
.disabled {
|
||||||
h1, h2, h3, h4, h5, h6 {
|
opacity: 0.4;
|
||||||
margin: 0;
|
}
|
||||||
|
|
||||||
|
#rendition-tooltip-root > div {
|
||||||
|
font-family: "SourceSansPro", sans-serif;
|
||||||
}
|
}
|
||||||
42
lib/gui/app/i18n.ts
Normal file
42
lib/gui/app/i18n.ts
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
import * as i18next from 'i18next';
|
||||||
|
import { initReactI18next } from 'react-i18next';
|
||||||
|
import zh_CN_translation from './i18n/zh-CN';
|
||||||
|
import zh_TW_translation from './i18n/zh-TW';
|
||||||
|
import en_translation from './i18n/en';
|
||||||
|
|
||||||
|
export function langParser() {
|
||||||
|
if (process.env.LANG !== undefined) {
|
||||||
|
// Bypass mocha, where lang-detect don't works
|
||||||
|
return 'en';
|
||||||
|
}
|
||||||
|
|
||||||
|
const lang = Intl.DateTimeFormat().resolvedOptions().locale;
|
||||||
|
|
||||||
|
switch (lang.substr(0, 2)) {
|
||||||
|
case 'zh':
|
||||||
|
if (lang === 'zh-CN' || lang === 'zh-SG') {
|
||||||
|
return 'zh-CN';
|
||||||
|
} // Simplified Chinese
|
||||||
|
else {
|
||||||
|
return 'zh-TW';
|
||||||
|
} // Traditional Chinese
|
||||||
|
default:
|
||||||
|
return lang.substr(0, 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
i18next.use(initReactI18next).init({
|
||||||
|
lng: langParser(),
|
||||||
|
fallbackLng: 'en',
|
||||||
|
nonExplicitSupportedLngs: true,
|
||||||
|
interpolation: {
|
||||||
|
escapeValue: false,
|
||||||
|
},
|
||||||
|
resources: {
|
||||||
|
'zh-CN': zh_CN_translation,
|
||||||
|
'zh-TW': zh_TW_translation,
|
||||||
|
en: en_translation,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export default i18next;
|
||||||
23
lib/gui/app/i18n/README.md
Normal file
23
lib/gui/app/i18n/README.md
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# i18n
|
||||||
|
|
||||||
|
## How it was done
|
||||||
|
|
||||||
|
Using the open-source lib [i18next](https://www.i18next.com/).
|
||||||
|
|
||||||
|
## How to add your own language
|
||||||
|
|
||||||
|
1. Go to `lib/gui/app/i18n` and add a file named `xx.ts` (use the codes mentioned
|
||||||
|
in [the link](https://www.science.co.il/language/Locale-codes.php), and we support styles as `fr`, `de`, `es-ES`
|
||||||
|
and `pt-BR`)
|
||||||
|
.
|
||||||
|
2. Copy the content from an existing translation and start to translate.
|
||||||
|
3. Once done, go to `lib/gui/app/i18n.ts` and add a line of `import xx_translation from './i18n/xx'` after the
|
||||||
|
already-added imports and add `xx: xx_translation` in the `resources` section of `i18next.init()` function.
|
||||||
|
4. Now go to `lib/shared/catalina-sudo/` and copy the `sudo-askpass.osascript-en.js`, change it to
|
||||||
|
be `sudo-askpass.osascript-xx.js` and edit
|
||||||
|
the `'balenaEtcher needs privileged access in order to flash disks.\n\nType your password to allow this.'` line and
|
||||||
|
those `Ok`s and `Cancel`s to your own language.
|
||||||
|
5. If, your language has several variations when they are used in several countries/regions, such as `zh-CN` and `zh-TW`
|
||||||
|
, or `pt-BR` and `pt-PT`, edit
|
||||||
|
the `langParser()` in the `lib/gui/app/i18n.ts` file to meet your need.
|
||||||
|
6. Make a commit, and then a pull request on GitHub.
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user