Compare commits
546 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
05988c1c49 | ||
|
|
d46b26e3bc | ||
|
|
236c172c6f | ||
|
|
59fcb56972 | ||
|
|
c07cd3a856 | ||
|
|
37766347a5 | ||
|
|
79da61782b | ||
|
|
8af87f1a8b | ||
|
|
494c954cbb | ||
|
|
71bc9eea28 | ||
|
|
e3b2bcfd06 | ||
|
|
142d974641 | ||
|
|
e56129111a | ||
|
|
0e1d6aa85c | ||
|
|
bcdb8cd770 | ||
|
|
7b2ca55089 | ||
|
|
f6ef0b684a | ||
|
|
02e1cdf210 | ||
|
|
b58950c574 | ||
|
|
833a60f29c | ||
|
|
f776d67c03 | ||
|
|
13e7cca1a4 | ||
|
|
0f3c477ff3 | ||
|
|
039cc30c07 | ||
|
|
25c8cd9246 | ||
|
|
c58841100a | ||
|
|
03e24cccd0 | ||
|
|
35f011758d | ||
|
|
2ebfaf76f2 | ||
|
|
0cf187dee7 | ||
|
|
bdeb325bad | ||
|
|
a1225b6d0d | ||
|
|
f0368b02c4 | ||
|
|
202de1436d | ||
|
|
7f8746fcd4 | ||
|
|
e05a25d701 | ||
|
|
6930570fa2 | ||
|
|
aba2c5b938 | ||
|
|
d82f86dcd9 | ||
|
|
159b4f9734 | ||
|
|
46a737c7a1 | ||
|
|
a731486ab7 | ||
|
|
c3e57f1fdd | ||
|
|
a9af484412 | ||
|
|
007646774e | ||
|
|
2d78e35e16 | ||
|
|
7524b5e349 | ||
|
|
2a04a48b89 | ||
|
|
3cbdaab81e | ||
|
|
8c858a5953 | ||
|
|
1812958106 | ||
|
|
4e5324916c | ||
|
|
1a77becc6a | ||
|
|
23ccaea2ff | ||
|
|
2a4b252a9d | ||
|
|
9ae4edfee5 | ||
|
|
bf48809b61 | ||
|
|
57a80a3c10 | ||
|
|
3f3e52d7ae | ||
|
|
5c69110658 | ||
|
|
be055d9dcb | ||
|
|
1e34a61911 | ||
|
|
97bd1da2a2 | ||
|
|
330ffb803f | ||
|
|
7b77f200be | ||
|
|
15a3c8408f | ||
|
|
bc1784ed2b | ||
|
|
55f0a82249 | ||
|
|
7aada3f328 | ||
|
|
dad885c051 | ||
|
|
f5c7bbfda8 | ||
|
|
f832743009 | ||
|
|
7551de6439 | ||
|
|
e03b4b7505 | ||
|
|
2d59fdd178 | ||
|
|
e61c8046f4 | ||
|
|
c0796ac3d6 | ||
|
|
68be24ffc6 | ||
|
|
9dcc87c705 | ||
|
|
d36c536107 | ||
|
|
affeeb39de | ||
|
|
f5d8a952f2 | ||
|
|
da07f99d3d | ||
|
|
eef66de68c | ||
|
|
4aa1180fce | ||
|
|
553d52a45e | ||
|
|
347b153884 | ||
|
|
1e7c176481 | ||
|
|
e390405d0c | ||
|
|
7378a84c96 | ||
|
|
b25013c4a2 | ||
|
|
6942916f13 | ||
|
|
f69f0b97f5 | ||
|
|
4361ea9686 | ||
|
|
be2ee33273 | ||
|
|
8c2ddb0255 | ||
|
|
466a5a932b | ||
|
|
8a3c6382e9 | ||
|
|
a2b45120c5 | ||
|
|
546ad52e11 | ||
|
|
1aefc5b540 | ||
|
|
1085ca4a2d | ||
|
|
9766322e99 | ||
|
|
cfb68e3bff | ||
|
|
a006963fb8 | ||
|
|
24c95c27c3 | ||
|
|
3c40c0be6b | ||
|
|
b1fc80b79a | ||
|
|
50d793e49b | ||
|
|
34c43b8349 | ||
|
|
7002a316fd | ||
|
|
1f37faad42 | ||
|
|
68cf24d100 | ||
|
|
86491da253 | ||
|
|
90249cdafa | ||
|
|
7c75111c41 | ||
|
|
7b53b6bfef | ||
|
|
fded5fd900 | ||
|
|
950965bd4a | ||
|
|
3a359319fa | ||
|
|
d3dd82c699 | ||
|
|
81f192bccb | ||
|
|
60a23febed | ||
|
|
d0e280cbac | ||
|
|
ecb62c8659 | ||
|
|
12669df92b | ||
|
|
44b2afeffa | ||
|
|
70f435e909 | ||
|
|
512d82071e | ||
|
|
3896230199 | ||
|
|
b902880a05 | ||
|
|
418526af16 | ||
|
|
45ad212459 | ||
|
|
0f49d424d3 | ||
|
|
01e42c8d6f | ||
|
|
26107bd6c3 | ||
|
|
7d3ecd2297 | ||
|
|
16056661dd | ||
|
|
059f50dad4 | ||
|
|
4c9975a7d9 | ||
|
|
9f9cc1ffb5 | ||
|
|
e768e1e277 | ||
|
|
acaf7b969a | ||
|
|
2b94975345 | ||
|
|
e6b4e12689 | ||
|
|
7eaac995bd | ||
|
|
a19cdb5e72 | ||
|
|
f54fbd057e | ||
|
|
19eceb4ecc | ||
|
|
dcff1ec25f | ||
|
|
567cda4cd3 | ||
|
|
900d8790b3 | ||
|
|
cad284519f | ||
|
|
0727acf458 | ||
|
|
d8813179be | ||
|
|
10d690c8fb | ||
|
|
52f71cdda0 | ||
|
|
2a9a348164 | ||
|
|
00346781bb | ||
|
|
4c6e92eea1 | ||
|
|
b63f469110 | ||
|
|
f6f176afc1 | ||
|
|
3de37a61c5 | ||
|
|
2d955dae48 | ||
|
|
46577fb128 | ||
|
|
37dba6ebfd | ||
|
|
66b949bed1 | ||
|
|
c9a05187fb | ||
|
|
cc956583fb | ||
|
|
14206efb09 | ||
|
|
5e6d7f5d16 | ||
|
|
7a33831d14 | ||
|
|
4f120e19fd | ||
|
|
37d064d836 | ||
|
|
824150f89b | ||
|
|
f7dc4cca2c | ||
|
|
ea39bb4334 | ||
|
|
5680d5a7be | ||
|
|
004246124b | ||
|
|
c41beae99a | ||
|
|
fe2cffb25b | ||
|
|
f71d5c429d | ||
|
|
dce5816b18 | ||
|
|
f99a7b2a8c | ||
|
|
ec36c69984 | ||
|
|
2458db03de | ||
|
|
7528b7bc1a | ||
|
|
8af33084ed | ||
|
|
f643175156 | ||
|
|
0321dda1d7 | ||
|
|
ff5d79e3ee | ||
|
|
4ee3ec09df | ||
|
|
cfe9d47fa0 | ||
|
|
607d6125fc | ||
|
|
6215259565 | ||
|
|
d034fecc89 | ||
|
|
f18d8229c0 | ||
|
|
e736626953 | ||
|
|
c2c438637a | ||
|
|
94638fe42c | ||
|
|
55ecfda39a | ||
|
|
d97a272aa5 | ||
|
|
80a1944b9d | ||
|
|
138cf943a9 | ||
|
|
c7e672e533 | ||
|
|
1b74a04efd | ||
|
|
290c7e6009 | ||
|
|
e8a56e0fea | ||
|
|
1ae7b646b3 | ||
|
|
42e2d73ce2 | ||
|
|
9e2a65a5ce | ||
|
|
fea20ea913 | ||
|
|
5b2480fff2 | ||
|
|
b0dca2a363 | ||
|
|
59bbe72798 | ||
|
|
f99a30a57e | ||
|
|
aa4cb29621 | ||
|
|
91ad4e396b | ||
|
|
351e17aacf | ||
|
|
6c8e09acdb | ||
|
|
1a7b341745 | ||
|
|
af592ea8c1 | ||
|
|
bb096a0357 | ||
|
|
3c226892c6 | ||
|
|
47f6fe069a | ||
|
|
aa3c1d930b | ||
|
|
99b0b4f5b8 | ||
|
|
bcd239ac2b | ||
|
|
2cc25b1e6e | ||
|
|
5fd3ed782f | ||
|
|
c34a24b633 | ||
|
|
775612ec5a | ||
|
|
fd43b16213 | ||
|
|
5a455ec4f7 | ||
|
|
1277c3d156 | ||
|
|
8033d1ca6d | ||
|
|
28df6881a7 | ||
|
|
e5fa5df7be | ||
|
|
f7dbf2bdd4 | ||
|
|
857c57daba | ||
|
|
5515da3c2d | ||
|
|
cfc111f855 | ||
|
|
3dd4043827 | ||
|
|
351ecfae0f | ||
|
|
b22393092b | ||
|
|
1485ee8027 | ||
|
|
60826c2d0c | ||
|
|
fb383458d7 | ||
|
|
196ee1aa8b | ||
|
|
2df97cd2f5 | ||
|
|
501b523680 | ||
|
|
6efa6691b1 | ||
|
|
c47f1ae236 | ||
|
|
aac240fe41 | ||
|
|
041debcd93 | ||
|
|
0632a2d3c8 | ||
|
|
9f40b3a873 | ||
|
|
8fad0af935 | ||
|
|
48ad744ebf | ||
|
|
556d5b0ca5 | ||
|
|
e30d70b6d4 | ||
|
|
a58f5a925a | ||
|
|
a3cc3c57fd | ||
|
|
0d0d3edeae | ||
|
|
dd0be7c522 | ||
|
|
9d2982fcd7 | ||
|
|
ebfd7d2153 | ||
|
|
818cd2454d | ||
|
|
b31d1c06f5 | ||
|
|
6cd884555c | ||
|
|
47ef74a1bb | ||
|
|
cc6d6ddd66 | ||
|
|
6a6cf015a6 | ||
|
|
ca79e81b39 | ||
|
|
a9e86cecf5 | ||
|
|
5773b1c3e5 | ||
|
|
b562b3410b | ||
|
|
f6440e9830 | ||
|
|
e43636e1e9 | ||
|
|
6783bf9903 | ||
|
|
807723c5b2 | ||
|
|
d3c4936116 | ||
|
|
bbb40aef51 | ||
|
|
485a3e29e7 | ||
|
|
1477f99c2c | ||
|
|
2e1f9d5fa9 | ||
|
|
9dea251862 | ||
|
|
17edfd6573 | ||
|
|
458e9d6cc7 | ||
|
|
485459b8b2 | ||
|
|
fcf377d26b | ||
|
|
3be1c9261f | ||
|
|
38600b3347 | ||
|
|
62f7f7a689 | ||
|
|
552f616305 | ||
|
|
a3164177f8 | ||
|
|
fa6bf21cd1 | ||
|
|
eecf76c1fb | ||
|
|
d1635cf24e | ||
|
|
b43e9ed7e7 | ||
|
|
12b2ab5da8 | ||
|
|
1c9085556c | ||
|
|
9122f8acee | ||
|
|
ef8c9f093c | ||
|
|
801dffd571 | ||
|
|
0b1c57b39f | ||
|
|
2febc268f7 | ||
|
|
58995bb3a2 | ||
|
|
8c944815bc | ||
|
|
f065a21542 | ||
|
|
27e032d10d | ||
|
|
ab3980cd38 | ||
|
|
1db648a525 | ||
|
|
ce3b5b683d | ||
|
|
9d23f1298d | ||
|
|
3f791b65b5 | ||
|
|
317d8703ca | ||
|
|
fda619f704 | ||
|
|
e4a0669da8 | ||
|
|
89725df3dc | ||
|
|
51799844c9 | ||
|
|
48de136e9d | ||
|
|
cb6f97a831 | ||
|
|
7e0cd0ab60 | ||
|
|
8521f04087 | ||
|
|
8ba45808be | ||
|
|
d876fd7f5b | ||
|
|
352e409a6e | ||
|
|
d6ec441c8e | ||
|
|
d197497349 | ||
|
|
d892ba6aa5 | ||
|
|
84b2583973 | ||
|
|
108648b427 | ||
|
|
71bf8b6b4d | ||
|
|
576067c1e5 | ||
|
|
e23bab0103 | ||
|
|
4e111c84f3 | ||
|
|
8cecce7570 | ||
|
|
0338fd42e1 | ||
|
|
b3788bc143 | ||
|
|
18d66ddded | ||
|
|
701b5ea561 | ||
|
|
86d0de4b0e | ||
|
|
a95958f9f6 | ||
|
|
69ab236f3f | ||
|
|
4cf3c6a616 | ||
|
|
da48bbf312 | ||
|
|
ac957db6d1 | ||
|
|
64464f23ae | ||
|
|
52cb239194 | ||
|
|
efd54b7523 | ||
|
|
2aca57cb82 | ||
|
|
d68baf08cb | ||
|
|
a7578aa709 | ||
|
|
a8261d376a | ||
|
|
fc346b4efd | ||
|
|
ad09e734da | ||
|
|
a674fea1c2 | ||
|
|
9e22b34fac | ||
|
|
fe24408620 | ||
|
|
c07ad0941c | ||
|
|
2f02b38b62 | ||
|
|
3ac766530d | ||
|
|
de77c71042 | ||
|
|
9c854a1757 | ||
|
|
f66fa1150e | ||
|
|
f820706e4f | ||
|
|
29e9e0f2cc | ||
|
|
2933093e17 | ||
|
|
71cd8918be | ||
|
|
c049ba59ff | ||
|
|
51c5f28443 | ||
|
|
bb1ed902a9 | ||
|
|
b016a60a75 | ||
|
|
890d485bb5 | ||
|
|
208bb2d72f | ||
|
|
267bf289c4 | ||
|
|
b3e083d866 | ||
|
|
a675c64c2d | ||
|
|
8b50c8515f | ||
|
|
1eaa377583 | ||
|
|
4345b1d930 | ||
|
|
06bf0c2622 | ||
|
|
3ac8de0a64 | ||
|
|
f237fd9847 | ||
|
|
5730280325 | ||
|
|
ab4df7e078 | ||
|
|
b52e6c99ab | ||
|
|
7dab548522 | ||
|
|
785c341822 | ||
|
|
7d2e1f63b5 | ||
|
|
e119459411 | ||
|
|
97ef2191fd | ||
|
|
e833ccf309 | ||
|
|
a4134d30fa | ||
|
|
6069fd02d3 | ||
|
|
bb15dc57a4 | ||
|
|
bdfe170c3b | ||
|
|
0fa2ba53ab | ||
|
|
4bb657debf | ||
|
|
dd12840e34 | ||
|
|
b027dcfec9 | ||
|
|
9e9b6f1542 | ||
|
|
7cd66e20d0 | ||
|
|
d93df15eff | ||
|
|
ddfd20d997 | ||
|
|
fd8af88493 | ||
|
|
bfa488f77d | ||
|
|
03be793930 | ||
|
|
37d88d5ff7 | ||
|
|
4616f889fd | ||
|
|
59cbf95c4f | ||
|
|
058711d3a8 | ||
|
|
2ddc61fa5c | ||
|
|
e04b7d0f01 | ||
|
|
2faa2ed1f4 | ||
|
|
5e2889e776 | ||
|
|
5bda36fb28 | ||
|
|
53fbb257b9 | ||
|
|
65a32d6e20 | ||
|
|
92450920d4 | ||
|
|
0099a9822e | ||
|
|
0cf86974dd | ||
|
|
716705aa15 | ||
|
|
757993064e | ||
|
|
3f738cf905 | ||
|
|
570715100b | ||
|
|
ad8750b40d | ||
|
|
757ea93393 | ||
|
|
dbd5a222d5 | ||
|
|
bba80bc80f | ||
|
|
094143bc28 | ||
|
|
24a335d304 | ||
|
|
c62b318b9e | ||
|
|
ea5c7c321a | ||
|
|
6d92775ab5 | ||
|
|
1a9360ca75 | ||
|
|
22b9bbe702 | ||
|
|
6fb44083ec | ||
|
|
ba02be08bb | ||
|
|
56fe3ede5b | ||
|
|
e48a000784 | ||
|
|
6d1c150ff5 | ||
|
|
21190a240f | ||
|
|
8a525bc131 | ||
|
|
734905d1f7 | ||
|
|
90edf2fc60 | ||
|
|
e3f37c14db | ||
|
|
c6c92184d9 | ||
|
|
c4fbc65354 | ||
|
|
54d250bde4 | ||
|
|
ef309bd8d0 | ||
|
|
6cdb6ec711 | ||
|
|
03891b66b6 | ||
|
|
42dd6326d5 | ||
|
|
5c4defdb8e | ||
|
|
f08d53b0c6 | ||
|
|
6859b85266 | ||
|
|
075adb4f03 | ||
|
|
5ce72a3461 | ||
|
|
8c2958b86d | ||
|
|
f15b7cebac | ||
|
|
f6d8df1e83 | ||
|
|
19ed5bf993 | ||
|
|
5567e2843d | ||
|
|
0a8e20fd60 | ||
|
|
558c4341e4 | ||
|
|
250860d92c | ||
|
|
64aecba7a0 | ||
|
|
3689b08237 | ||
|
|
30e567e8b6 | ||
|
|
ddd74549fe | ||
|
|
14620c32aa | ||
|
|
fb7068d415 | ||
|
|
8614ff40df | ||
|
|
aa10a9d899 | ||
|
|
a5b8feca93 | ||
|
|
486e47f985 | ||
|
|
bb5a1ad513 | ||
|
|
eac0a52f10 | ||
|
|
7ac00258cc | ||
|
|
e3a0ae8a4b | ||
|
|
2953159f8b | ||
|
|
9693363c76 | ||
|
|
a2533af116 | ||
|
|
b4aecb5b74 | ||
|
|
15aa2498b5 | ||
|
|
0372ff0c2c | ||
|
|
7a8d5a391a | ||
|
|
2a6c81a89d | ||
|
|
301871aec6 | ||
|
|
25359e5320 | ||
|
|
b6fff53b21 | ||
|
|
ae7b5fac74 | ||
|
|
26168a9520 | ||
|
|
698dfca319 | ||
|
|
3bcb98e644 | ||
|
|
2deb436ccd | ||
|
|
2b3405c4a9 | ||
|
|
677a465630 | ||
|
|
8ecb76fc0b | ||
|
|
0178013fc1 | ||
|
|
c273a8ee69 | ||
|
|
0ed56b706b | ||
|
|
4582b6cf76 | ||
|
|
05513bcd1e | ||
|
|
f5dd135ed8 | ||
|
|
9c8f85741c | ||
|
|
ca515f2eae | ||
|
|
80c1ebd768 | ||
|
|
b51fd7fc13 | ||
|
|
efe86c37b2 | ||
|
|
d20a4a8bfc | ||
|
|
9da2d11e80 | ||
|
|
5ef554aecf | ||
|
|
9a7fea0447 | ||
|
|
ae52ff93b2 | ||
|
|
80a567bf1e | ||
|
|
ce2a3361eb | ||
|
|
ca9ea109c6 | ||
|
|
2a33a746f0 | ||
|
|
e8c5246645 | ||
|
|
98295b85ab | ||
|
|
af1823db8c | ||
|
|
a2ab6b89f1 | ||
|
|
5de300fb35 | ||
|
|
62a4c82e95 | ||
|
|
d522c864d4 | ||
|
|
aa8ff7ace3 | ||
|
|
4e6a931de3 | ||
|
|
5e141e869d | ||
|
|
611555514c | ||
|
|
e1c78fcbd3 | ||
|
|
8640d6bb1e | ||
|
|
28d5bedcc7 | ||
|
|
373b890e1d | ||
|
|
aad0f90a9d | ||
|
|
5dc45c35e6 | ||
|
|
b8c87632e6 | ||
|
|
c85903383a | ||
|
|
4aededf038 | ||
|
|
4bc6501b8d | ||
|
|
a1b3b47573 | ||
|
|
c8cf4fe09c | ||
|
|
ca07d75405 | ||
|
|
c5001f3620 |
34
.travis.yml
Normal file
34
.travis.yml
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
dist: bionic
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
include:
|
||||||
|
- stage: build
|
||||||
|
# TODO: Shallowly clone again once the .git folder is no longer required for building
|
||||||
|
git:
|
||||||
|
depth: false
|
||||||
|
language: crystal
|
||||||
|
crystal: latest
|
||||||
|
before_install:
|
||||||
|
- shards update
|
||||||
|
- shards install
|
||||||
|
install:
|
||||||
|
- crystal build --warnings all --error-on-warnings src/invidious.cr
|
||||||
|
script:
|
||||||
|
- crystal tool format --check
|
||||||
|
- crystal spec
|
||||||
|
|
||||||
|
- stage: build_docker
|
||||||
|
# TODO: Shallowly clone again once the .git folder is no longer required for building
|
||||||
|
git:
|
||||||
|
depth: false
|
||||||
|
language: minimal
|
||||||
|
services:
|
||||||
|
- docker
|
||||||
|
install:
|
||||||
|
- docker-compose build
|
||||||
|
script:
|
||||||
|
- docker-compose up -d
|
||||||
|
- sleep 15 # Wait for cluster to become ready, TODO: do not sleep
|
||||||
|
- HEADERS="$(curl -I -s http://localhost:3000/)"
|
||||||
|
- STATUS="$(echo $HEADERS | head -n1)"
|
||||||
|
- if [[ "$STATUS" != *"200 OK"* ]]; then echo "$HEADERS"; exit 1; fi
|
||||||
316
CHANGELOG.md
316
CHANGELOG.md
@@ -1,3 +1,319 @@
|
|||||||
|
# 0.20.0 (2019-011-06)
|
||||||
|
|
||||||
|
# Version 0.20.0: Custom Playlists
|
||||||
|
|
||||||
|
It's been quite a while since the last release! There've been [198 commits](https://github.com/omarroth/invidious/compare/0.19.0..0.20.0) from 27 contributors.
|
||||||
|
|
||||||
|
A couple smaller features have since been added. Channel pages and playlists in particular have received a bit of a face-lift, with both now displaying their descriptions as expected, and playlists providing video count and published information. Channels will also now provide video descriptions in their RSS feed.
|
||||||
|
|
||||||
|
Turkish (tr), Chinese (zh-TW, in addition to zh-CN), and Japanese (jp) are all now supported languages. Thank you as always to the hard work done by translators that makes this possible.
|
||||||
|
|
||||||
|
The feed menu and default home page are both now configurable for registered and unregistered users, and is quite a bit of an improvement for users looking to reduce distractions for their daily use.
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
`feed_menu` and `default_home` are now configurable by the user, and have therefore been moved into `default_user_preferences`:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
feed_menu: ["Popular", "Top"]
|
||||||
|
default_home: Top
|
||||||
|
|
||||||
|
# becomes:
|
||||||
|
|
||||||
|
default_user_preferences:
|
||||||
|
feed_menu: ["Popular", "Top"]
|
||||||
|
default_home: Top
|
||||||
|
```
|
||||||
|
|
||||||
|
Several new options have also been added, including the ability to set a support email for the instance using `admin_email: EMAIL`, and forcing the use of a specific connection in the case of rate-limiting using `force_resolve` (see below).
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
Authenticated endpoints are now [properly documented](https://github.com/omarroth/invidious/wiki/Authenticated-Endpoints), as well how to generate and use API tokens. My hope is that this makes some of the more [interesting](https://github.com/omarroth/invidious/wiki/Authenticated-Endpoints#get-apiv1authnotifications) endpoints more accessible for developers to use in their own applications.
|
||||||
|
|
||||||
|
API endpoints for interacting with custom playlists have also been added with documentation available [here](https://github.com/omarroth/invidious/wiki/Authenticated-Endpoints#get-apiv1authplaylists).
|
||||||
|
|
||||||
|
## Custom playlists
|
||||||
|
|
||||||
|
This is probably the feature that has been the longest in the pipe and that I'm quite pleased is now implemented. It is now possible to create custom playlists, which can be played and edited through Invidious. API endpoints have also been added (documentation [here](https://github.com/omarroth/invidious/wiki/Authenticated-Endpoints#get-apiv1authplaylists)).
|
||||||
|
|
||||||
|
Overall I'm quite pleased with how smoothly it has been rolled out and with the experience so far, and I'm exctited for how it can be extended and improved in future.
|
||||||
|
|
||||||
|
## [instances.invidio.us](https://instances.invidio.us)
|
||||||
|
|
||||||
|
It is now possible to view a list of public instances (as provided in the [wiki](https://github.com/omarroth/invidious/wiki/Invidious-Instances)) through an API or a pretty new interface [here](https://instances.invidio.us). It combines uptime information, statistics from each instance and basic information already provided in the wiki. I expect it should be much more user-friendly than compiling the information yourself, and is already used by [Invidition](https://codeberg.org/Booteille/Invidition) to provide a list of instances for users to choose from.
|
||||||
|
|
||||||
|
The site itself is licensed under the AGPLv3 and the source is available [here](https://github.com/omarroth/instances.invidio.us).
|
||||||
|
|
||||||
|
## Video unavailable [#811](https://github.com/omarroth/invidious/issues/811)
|
||||||
|
|
||||||
|
Many users have likely noticed this error message if using Invidious directly or through another service, such as FreeTube. This issue is caused by rate-limiting by Google, and is not a new issuee for projects like Invidious (notably [youtube-dl](https://github.com/ytdl-org/youtube-dl#http-error-429-too-many-requests-or-402-payment-required)) and appears to be affecting smaller, private instances as well.
|
||||||
|
|
||||||
|
There is not a permanent fix for administrators currently, however there is some information available [here](https://github.com/omarroth/invidious/issues/811#issuecomment-540017772) that may provide a temporary solution. Unfortanately, in most cases the best option is to wait for the instance to be unbanned or to move the instance to a different IP. A more informative error message is also now provided, which should help an administrator more quickly diagnose the problem.
|
||||||
|
|
||||||
|
For those interested, I would recommend following [#811](https://github.com/omarroth/invidious/issues/811) for any future progress on the issue.
|
||||||
|
|
||||||
|
## BAT verified publisher
|
||||||
|
|
||||||
|
I'm quite late to this announcement, however I'm pleased to mention that Invidious is now a BAT verified publisher! I would recommend looking [here](https://basicattentiontoken.org/about/) or [here](https://www.reddit.com/r/BATProject/comments/7cr7yc/new_to_bat_read_this_introduction_to_basic/) for learning more about what it is and how it works. Overall I think it makes an interesting substitute for services like Liberapay, and a (hopefully) much less-intrusive alternative to direct advertising.
|
||||||
|
|
||||||
|
BAT is combined under other cryptocurrencies below. Currently there's a fairly significant delay in payout, which is the reason for the large fluctuation in crypto donations between September and October (and also the reason for the late announcement).
|
||||||
|
|
||||||
|
## Release schedule
|
||||||
|
|
||||||
|
Currently I'm quite pleased with the current state of the project. There's plenty of things I'd still like to add, however at this point I expect the rate of most new additions will slow down a bit, with more focus on stabililty and any long-standing bugs.
|
||||||
|
|
||||||
|
Because of this, I'm planning on releasing a new version quarterly, with any necessary hotfixes being pushed as a new patch release as necessary. As always it will be possible to run Invidious directly from [master](https://github.com/omarroth/invidious/wiki/Updating) if you'd still like to have the lastest version.
|
||||||
|
|
||||||
|
I'll plan on providing finances each release, with a similar monthly breakdown as below.
|
||||||
|
|
||||||
|
## Finances for September 2019
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$64.37
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$76.04
|
||||||
|
- Crypto : ~\$99.89 (converted from BAT, BCH, BTC)
|
||||||
|
- Total : \$240.30
|
||||||
|
|
||||||
|
### Expenses
|
||||||
|
|
||||||
|
- invidious-lb1 (nyc1) : \$10.00 (load balancer)
|
||||||
|
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
|
||||||
|
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node6 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node7 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node8 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node9 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node10 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node11 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node12 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node13 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node14 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node15 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node16 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$135.00
|
||||||
|
|
||||||
|
## Finances for October 2019
|
||||||
|
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$134.40
|
||||||
|
- Crypto : ~\$8.29 (converted from BAT, BCH, BTC)
|
||||||
|
- Total : \$142.69
|
||||||
|
|
||||||
|
### Expenses
|
||||||
|
|
||||||
|
- invidious-lb1 (nyc1) : \$5.00 (load balancer)
|
||||||
|
- invidious-lb2 (nyc1) : \$5.00 (load balancer)
|
||||||
|
- invidious-lb3 (nyc1) : \$5.00 (load balancer)
|
||||||
|
- invidious-lb4 (nyc1) : \$5.00 (load balancer)
|
||||||
|
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
|
||||||
|
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node6 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node7 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node8 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node9 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node10 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node11 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node12 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node13 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node14 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node15 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node16 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node17 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node18 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$155.00
|
||||||
|
|
||||||
|
# 0.19.0 (2019-07-13)
|
||||||
|
|
||||||
|
# Version 0.19.0: Communities
|
||||||
|
|
||||||
|
Hello again everyone! Focus this month has mainly been on improving playback performance, along with a couple new features I'd like to announce. There have been [109 commits](https://github.com/omarroth/invidious/compare/0.18.0...0.19.0) this past month from 10 contributors.
|
||||||
|
|
||||||
|
This past month has seen the addition of Chinese (`zh-CN`) and Icelandic (`is`) translations. I would like to give a huge thanks to their respective translators, and again an enormous thanks to everyone who helps translate the site.
|
||||||
|
|
||||||
|
I'm delighted to mention that [FreeTube 0.6.0](https://github.com/FreeTubeApp/FreeTube) now supports 1080p thanks to the Invidious API. I would very much recommend reading the [relevant post](https://freetube.writeas.com/freetube-release-0-6-0-beta-1080p-and-a-lot-of-qol) for some more information on how it works, along with several other major improvements. Folks that are interested in adding similar functionality for their own projects should feel free to get in touch.
|
||||||
|
|
||||||
|
This past month there has been quite a bit of work on improving memory usage and improving download and playback speeds. As mentioned in the previous release, some extra hardware has been allocated which should also help with this. I'm still looking for ways to improve performance and feedback is always appreciated.
|
||||||
|
|
||||||
|
Along with performance, a couple quality of life improvements have been added, including author thumbnails and banners, clickable titles for embedded videos, and better styling for captions, among some other enhancements.
|
||||||
|
|
||||||
|
## Communities
|
||||||
|
|
||||||
|
Support for YouTube's [communities tab](https://creatoracademy.youtube.com/page/lesson/community-tab) has been added. It's a very interesting but surprisingly unknown feature. Essentially, providing comments for a channel, rather than a video, where an author can post updates for their subscribers.
|
||||||
|
|
||||||
|
It's commonly used to promote interesting links and foster discussion. I hope this feature helps people find more interesting content that otherwise would have been overlooked.
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
For accessing channel communities, an `/api/v1/channels/comments/:ucid` endpoint has been added, with similar behavior and schema to `/api/v1/comments/:id`, with an extra `attachment` field for top-level comments. More info on usage and available data can be found in the [wiki](https://github.com/omarroth/invidious/wiki/API#get-apiv1channelscommentsucid-apiv1channelsucidcomments).
|
||||||
|
|
||||||
|
An `/api/v1/auth/feeds` endpoint has been added for programmatically accessing a user's subscription feed, with options for displaying notifications and filtering an existing feed.
|
||||||
|
|
||||||
|
An `/api/v1/search/suggestions` endpoint has been added for retrieving suggestions for a given query.
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
It is now possible to disable more resource intensive features, such as downloads and DASH functionality by adding `disable_proxy` to your config. See [#453](https://github.com/omarroth/invidious/issues/453) and the [Wiki](https://github.com/omarroth/invidious/wiki/Configuration) for more information and example usage. I expect this to be a big help for folks with limited bandwidth when hosting their own instances.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$38.39
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$84.85
|
||||||
|
- Crypto : ~\$0.00 (converted from BCH, BTC)
|
||||||
|
- Total : \$123.24
|
||||||
|
|
||||||
|
### Expenses
|
||||||
|
|
||||||
|
- invidious-load1 (nyc1) : \$10.00 (load balancer)
|
||||||
|
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
|
||||||
|
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node6 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node7 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node8 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node9 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node10 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$105.00
|
||||||
|
|
||||||
|
The goal on Patreon has been updated to reflect the above expenses. As mentioned above, the main reason for more hardware is to improve playback and download speeds, although I'm still looking into improving performance without allocating more hardware.
|
||||||
|
|
||||||
|
As always I'm grateful for everyone's support and feedback. I'll see you all next month.
|
||||||
|
|
||||||
|
# 0.18.0 (2019-06-06)
|
||||||
|
|
||||||
|
# Version 0.18.0: Native Notifications and Optimizations
|
||||||
|
|
||||||
|
Hope everyone has been doing well. This past month there have been [97 commits](https://github.com/omarroth/invidious/compare/0.17.0...0.18.0) from 10 contributors. For the most part changes this month have been on optimizing various parts of the site, mainly subscription feeds and support for serving images and other assets.
|
||||||
|
|
||||||
|
I'm quite happy to mention that support for Greek (`el`) has been added, which I hope will continue to make the site accessible for more users.
|
||||||
|
|
||||||
|
Subscription feeds will now only update when necessary, rather than periodically. This greatly lightens the load on DB as well as making the feeds generally more responsive when changing subscriptions, importing data, and when receiving new uploads.
|
||||||
|
|
||||||
|
Caching for images and other assets should be greatly improved with [#456](https://github.com/omarroth/invidious/issues/456). JavaScript has been pulled out into separate files where possible to take advantage of this, which should result in lighter pages and faster load times.
|
||||||
|
|
||||||
|
This past month several people have encountered issues with downloads and watching high quality video through the site, see [#532](https://github.com/omarroth/invidious/issues/532) and [#562](https://github.com/omarroth/invidious/issues/562). For this coming month I've allocated some more hardware which should help with this, and I'm also looking into optimizing how videos are currently served.
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
`viewCount` is now available for `/api/v1/popular` and all videos returned from `/api/v1/auth/notifications`. Both also now provide `"type"` for indicating available information for each object.
|
||||||
|
|
||||||
|
An `/authorize_token` page is now available for more easily creating new tokens for use in applications, see [this comment](https://github.com/omarroth/invidious/issues/473#issuecomment-496230812) in [#473](https://github.com/omarroth/invidious/issues/473) for more details.
|
||||||
|
|
||||||
|
A POST `/api/v1/auth/notifications` endpoint is also now available for correctly returning notifications for 150+ channels.
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
There are two new schema changes for administrators: `views` for adding view count to the popular page, and `feed_needs_update` for tracking feed changes.
|
||||||
|
|
||||||
|
As always the relevant migration scripts are provided which should run when following instructions for [updating](https://github.com/omarroth/invidious/wiki/Updating). Otherwise, adding `check_tables: true` to your config will automatically make the required changes.
|
||||||
|
|
||||||
|
## Native Notifications
|
||||||
|
|
||||||
|
[<img src="https://omar.yt/81c3ae1839831bd9300d75e273b6552a86dc2352/native_notification.png" height="160" width="472">](https://omar.yt/81c3ae1839831bd9300d75e273b6552a86dc2352/native_notification.png "Example of native notification, available in repository under screnshots/native_notification.png")
|
||||||
|
|
||||||
|
It is now possible to receive [Web notifications](https://developer.mozilla.org/en-US/docs/Web/API/Notifications_API) from subscribed channels.
|
||||||
|
|
||||||
|
You can enable notifications by clicking "Enable web notifications" in your preferences. Generally they appear within 20-60 seconds of a new video being uploaded, and I've found them to be an enormous quality of life improvement.
|
||||||
|
|
||||||
|
Although it has been fairly stable, please feel free to report any issues you find [here](https://github.com/omarroth/invidious/issues) or emailing me directly at omarroth@protonmail.com.
|
||||||
|
|
||||||
|
Important to note for administrators is that instances require [`use_pubsub_feeds`](https://github.com/omarroth/invidious/wiki/Configuration) and must be served over HTTPS in order to correctly send web notifications.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$49.73
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$100.57
|
||||||
|
- Crypto : ~\$11.12 (converted from BCH, BTC)
|
||||||
|
- Total : \$161.42
|
||||||
|
|
||||||
|
### Expenses
|
||||||
|
|
||||||
|
- invidious-load1 (nyc1) : \$10.00 (load balancer)
|
||||||
|
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
|
||||||
|
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node6 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$85.00
|
||||||
|
|
||||||
|
See you all next month!
|
||||||
|
|
||||||
|
# 0.17.0 (2019-05-06)
|
||||||
|
|
||||||
|
# Version 0.17.0: Player and Authentication API
|
||||||
|
|
||||||
|
Hello everyone! This past month there have been [130 commits](https://github.com/omarroth/invidious/compare/0.16.0..0.17.0) from 11 contributors. Large focus has been on improving the player as well as adding API access for other projects to make use of Invidious.
|
||||||
|
|
||||||
|
There have also been significant changes in preparation of native notifications (see [#195](https://github.com/omarroth/invidious/issues/195), [#469](https://github.com/omarroth/invidious/issues/469), [#473](https://github.com/omarroth/invidious/issues/473), and [#502](https://github.com/omarroth/invidious/issues/502)), and playlists. I expect to see both of these to be added in the next release.
|
||||||
|
|
||||||
|
I'm quite happy to mention that new translations have been added for Esperanto (`eo`) and Ukranian (`uk`). Support for pluralization has also been added, so it should now be possible to make a more native experience for speakers in other languages. The system currently in place is a bit cumbersome, so for any help using this feature please get in touch!
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
A `check_tables` option has been added to automatically migrate without the use of custom scripts. This method will likely prove to be much more robust, and is currently enabled for the official instance. To prevent any unintended changes to the DB, `check_tables` is disabled by default and will print commands before executing. Having this makes features that require schema changes much easier to implement, and also makes it easier to upgrade from older instances.
|
||||||
|
|
||||||
|
As part of [#303](https://github.com/omarroth/invidious/issues/303), a `cache_annotations` option has been added to speed up access from `/api/v1/annotations/:id`. This vastly improves the experience for videos with annotations. Currently, only videos that contain legacy annotations will be cached, which should help keep down the size of the cache. `cache_annotations` is disabled by default.
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
An authorization API has been added which allows other applications to read and modify user subscriptions and preferences (see [#473](https://github.com/omarroth/invidious/issues/473)). Support for accessing user feeds and notifications is also planned. I believe this feature is a large step forward in supporting syncing subscriptions and preferences with other services, and I'm excited to see what other developers do with this functionality.
|
||||||
|
|
||||||
|
Support for server-to-client push notifications is currently underway. This allows Invidious users, as well as applications using the Invidious API, to receive notifications about uploads in near real-time (see #469). An `/api/v1/auth/notifications` endpoint is currently available. I'm very excited for this to be integrated into the site, and to see how other developers use it in their own projects.
|
||||||
|
|
||||||
|
An `/api/v1/storyboards/:id` endpoint has been added for accessing storyboard URLs, which allows developers to add video previews to their players (see below).
|
||||||
|
|
||||||
|
## Player
|
||||||
|
|
||||||
|
Support for annotations has been merged into master with [#303](https://github.com/omarroth/invidious/issues/303), thanks @glmdgrielson! Annotations can be enabled by default or only for subscribed channels, and can also be toggled per video. I'm extremely proud of the progress made here, and I'm so thankful to everyone that has made this possible. I expect this to be the last update with regards to supporting annotations, but I do plan on continuing to improve the experience as much as possible.
|
||||||
|
|
||||||
|
The Invidious player now supports video previews and a corresponding API endpoint `/api/v1/storyboards/:id` has been added for developers looking to add similar functionality to their own players. Not much else to say here. Overall it's a very nice quality of life improvement and an attractive addition to the site.
|
||||||
|
|
||||||
|
It is now possible to select specific sources for videos provided using DASH (see [#34](https://github.com/omarroth/invidious/issues/34)). I would consider support largely feature complete, although there are still several issues to be fixed before I would consider it ready for larger rollout. You can watch videos in 1080p by setting `Default quality` to `dash` in your preferences, or by adding `&quality=dash` to the end of video URLs.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$49.73
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$63.03
|
||||||
|
- Crypto : ~\$0.00 (converted from BCH, BTC)
|
||||||
|
- Total : \$112.76
|
||||||
|
|
||||||
|
### Expenses
|
||||||
|
|
||||||
|
- invidious-load1 (nyc1) : \$10.00 (load balancer)
|
||||||
|
- invidious-update1 (s-1vcpu-1gb) : \$5.00 (updates feeds)
|
||||||
|
- invidious-node1 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node2 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node3 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node4 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-node5 (s-1vcpu-1gb) : \$5.00 (web server)
|
||||||
|
- invidious-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$80.00
|
||||||
|
|
||||||
|
That's all for now. Thanks!
|
||||||
|
|
||||||
# 0.16.0 (2019-04-06)
|
# 0.16.0 (2019-04-06)
|
||||||
|
|
||||||
# Version 0.16.0: API Improvements and Annotations
|
# Version 0.16.0: API Improvements and Annotations
|
||||||
|
|||||||
61
README.md
61
README.md
@@ -1,5 +1,7 @@
|
|||||||
# Invidious
|
# Invidious
|
||||||
|
|
||||||
|
[](https://travis-ci.org/omarroth/invidious)
|
||||||
|
|
||||||
## Invidious is an alternative front-end to YouTube
|
## Invidious is an alternative front-end to YouTube
|
||||||
|
|
||||||
- Audio-only mode (and no need to keep window open on mobile)
|
- Audio-only mode (and no need to keep window open on mobile)
|
||||||
@@ -23,16 +25,19 @@
|
|||||||
- Developer [API](https://github.com/omarroth/invidious/wiki/API)
|
- Developer [API](https://github.com/omarroth/invidious/wiki/API)
|
||||||
|
|
||||||
Liberapay: https://liberapay.com/omarroth
|
Liberapay: https://liberapay.com/omarroth
|
||||||
Patreon: https://patreon.com/omarroth
|
|
||||||
BTC: 356DpZyMXu6rYd55Yqzjs29n79kGKWcYrY
|
BTC: 356DpZyMXu6rYd55Yqzjs29n79kGKWcYrY
|
||||||
BCH: qq4ptclkzej5eza6a50et5ggc58hxsq5aylqut2npk
|
BCH: qq4ptclkzej5eza6a50et5ggc58hxsq5aylqut2npk
|
||||||
|
|
||||||
Onion links:
|
## Invidious Instances
|
||||||
|
|
||||||
- kgg2m7yk5aybusll.onion
|
See [Invidious Instances](https://github.com/omarroth/invidious/wiki/Invidious-Instances) for a full list of publicly available instances.
|
||||||
- axqzx4s6s54s32yentfqojs3x5i7faxza6xo3ehd4bzzsg2ii4fv2iid.onion
|
|
||||||
|
|
||||||
[Alternative Invidious instances](https://github.com/omarroth/invidious/wiki/Invidious-Instances)
|
### Official Instances
|
||||||
|
|
||||||
|
- [invidio.us](https://invidio.us) 🇺🇸
|
||||||
|
Issuer: Let's Encrypt, [SSLLabs Verification](https://www.ssllabs.com/ssltest/analyze.html?d=invidio.us)
|
||||||
|
- [kgg2m7yk5aybusll.onion](http://kgg2m7yk5aybusll.onion)
|
||||||
|
- [axqzx4s6s54s32yentfqojs3x5i7faxza6xo3ehd4bzzsg2ii4fv2iid.onion](http://axqzx4s6s54s32yentfqojs3x5i7faxza6xo3ehd4bzzsg2ii4fv2iid.onion)
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||
|
|
||||||
@@ -74,7 +79,7 @@ $ docker-compose build
|
|||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Arch Linux
|
# Arch Linux
|
||||||
$ sudo pacman -S shards crystal imagemagick librsvg postgresql
|
$ sudo pacman -S shards crystal librsvg postgresql
|
||||||
|
|
||||||
# Ubuntu or Debian
|
# Ubuntu or Debian
|
||||||
# First you have to add the repository to your APT configuration. For easy setup just run in your command line:
|
# First you have to add the repository to your APT configuration. For easy setup just run in your command line:
|
||||||
@@ -83,7 +88,7 @@ $ curl -sSL https://dist.crystal-lang.org/apt/setup.sh | sudo bash
|
|||||||
$ curl -sL "https://keybase.io/crystal/pgp_keys.asc" | sudo apt-key add -
|
$ curl -sL "https://keybase.io/crystal/pgp_keys.asc" | sudo apt-key add -
|
||||||
$ echo "deb https://dist.crystal-lang.org/apt crystal main" | sudo tee /etc/apt/sources.list.d/crystal.list
|
$ echo "deb https://dist.crystal-lang.org/apt crystal main" | sudo tee /etc/apt/sources.list.d/crystal.list
|
||||||
$ sudo apt-get update
|
$ sudo apt-get update
|
||||||
$ sudo apt install crystal libssl-dev libxml2-dev libyaml-dev libgmp-dev libreadline-dev librsvg2-dev postgresql imagemagick libsqlite3-dev
|
$ sudo apt install crystal libssl-dev libxml2-dev libyaml-dev libgmp-dev libreadline-dev postgresql librsvg2-bin libsqlite3-dev
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Add invidious user and clone repository
|
#### Add invidious user and clone repository
|
||||||
@@ -101,14 +106,15 @@ $ exit
|
|||||||
$ sudo systemctl enable postgresql
|
$ sudo systemctl enable postgresql
|
||||||
$ sudo systemctl start postgresql
|
$ sudo systemctl start postgresql
|
||||||
$ sudo -i -u postgres
|
$ sudo -i -u postgres
|
||||||
$ psql -c "CREATE USER kemal WITH PASSWORD 'kemal';"
|
$ psql -c "CREATE USER kemal WITH PASSWORD 'kemal';" # Change 'kemal' here to a stronger password, and update `password` in config/config.yml
|
||||||
$ createdb -O kemal invidious
|
$ createdb -O kemal invidious
|
||||||
$ psql invidious < /home/invidious/invidious/config/sql/channels.sql
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/channels.sql
|
||||||
$ psql invidious < /home/invidious/invidious/config/sql/videos.sql
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/videos.sql
|
||||||
$ psql invidious < /home/invidious/invidious/config/sql/channel_videos.sql
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/channel_videos.sql
|
||||||
$ psql invidious < /home/invidious/invidious/config/sql/users.sql
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/users.sql
|
||||||
$ psql invidious < /home/invidious/invidious/config/sql/session_ids.sql
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/session_ids.sql
|
||||||
$ psql invidious < /home/invidious/invidious/config/sql/nonces.sql
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/nonces.sql
|
||||||
|
$ psql invidious kemal < /home/invidious/invidious/config/sql/annotations.sql
|
||||||
$ exit
|
$ exit
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -137,20 +143,21 @@ $ sudo systemctl start invidious.service
|
|||||||
```bash
|
```bash
|
||||||
# Install dependencies
|
# Install dependencies
|
||||||
$ brew update
|
$ brew update
|
||||||
$ brew install shards crystal-lang postgres imagemagick librsvg
|
$ brew install shards crystal postgres imagemagick librsvg
|
||||||
|
|
||||||
# Clone repository and setup postgres database
|
# Clone repository and setup postgres database
|
||||||
$ git clone https://github.com/omarroth/invidious
|
$ git clone https://github.com/omarroth/invidious
|
||||||
$ cd invidious
|
$ cd invidious
|
||||||
$ brew services start postgresql
|
$ brew services start postgresql
|
||||||
$ psql -c "CREATE ROLE kemal WITH LOGIN PASSWORD 'kemal';"
|
$ psql -c "CREATE ROLE kemal WITH PASSWORD 'kemal';" # Change 'kemal' here to a stronger password, and update `password` in config/config.yml
|
||||||
$ createdb invidious -U kemal
|
$ createdb -O kemal invidious
|
||||||
$ psql invidious < config/sql/channels.sql
|
$ psql invidious kemal < config/sql/channels.sql
|
||||||
$ psql invidious < config/sql/videos.sql
|
$ psql invidious kemal < config/sql/videos.sql
|
||||||
$ psql invidious < config/sql/channel_videos.sql
|
$ psql invidious kemal < config/sql/channel_videos.sql
|
||||||
$ psql invidious < config/sql/users.sql
|
$ psql invidious kemal < config/sql/users.sql
|
||||||
$ psql invidious < config/sql/session_ids.sql
|
$ psql invidious kemal < config/sql/session_ids.sql
|
||||||
$ psql invidious < config/sql/nonces.sql
|
$ psql invidious kemal < config/sql/nonces.sql
|
||||||
|
$ psql invidious kemal < config/sql/annotations.sql
|
||||||
|
|
||||||
# Setup Invidious
|
# Setup Invidious
|
||||||
$ shards update && shards install
|
$ shards update && shards install
|
||||||
@@ -172,15 +179,12 @@ Usage: invidious [arguments]
|
|||||||
--ssl-key-file FILE SSL key file
|
--ssl-key-file FILE SSL key file
|
||||||
--ssl-cert-file FILE SSL certificate file
|
--ssl-cert-file FILE SSL certificate file
|
||||||
-h, --help Shows this help
|
-h, --help Shows this help
|
||||||
-t THREADS, --crawl-threads=THREADS
|
|
||||||
Number of threads for crawling YouTube (default: 0)
|
|
||||||
-c THREADS, --channel-threads=THREADS
|
-c THREADS, --channel-threads=THREADS
|
||||||
Number of threads for refreshing channels (default: 1)
|
Number of threads for refreshing channels (default: 1)
|
||||||
-f THREADS, --feed-threads=THREADS
|
-f THREADS, --feed-threads=THREADS
|
||||||
Number of threads for refreshing feeds (default: 1)
|
Number of threads for refreshing feeds (default: 1)
|
||||||
-v THREADS, --video-threads=THREADS
|
|
||||||
Number of threads for refreshing videos (default: 0)
|
|
||||||
-o OUTPUT, --output=OUTPUT Redirect output (default: STDOUT)
|
-o OUTPUT, --output=OUTPUT Redirect output (default: STDOUT)
|
||||||
|
-v, --version Print version
|
||||||
```
|
```
|
||||||
|
|
||||||
Or for development:
|
Or for development:
|
||||||
@@ -188,6 +192,7 @@ Or for development:
|
|||||||
```bash
|
```bash
|
||||||
$ curl -fsSLo- https://raw.githubusercontent.com/samueleaton/sentry/master/install.cr | crystal eval
|
$ curl -fsSLo- https://raw.githubusercontent.com/samueleaton/sentry/master/install.cr | crystal eval
|
||||||
$ ./sentry
|
$ ./sentry
|
||||||
|
🤖 Your SentryBot is vigilant. beep-boop...
|
||||||
```
|
```
|
||||||
|
|
||||||
## Documentation
|
## Documentation
|
||||||
@@ -201,7 +206,7 @@ $ ./sentry
|
|||||||
## Made with Invidious
|
## Made with Invidious
|
||||||
|
|
||||||
- [FreeTube](https://github.com/FreeTubeApp/FreeTube): An Open Source YouTube app for privacy.
|
- [FreeTube](https://github.com/FreeTubeApp/FreeTube): An Open Source YouTube app for privacy.
|
||||||
- [CloudTube](https://github.com/cloudrac3r/cadencegq): Website featuring pastebin, image host, and YouTube player
|
- [CloudTube](https://cadence.moe/cloudtube/subscriptions): A JS-rich alternate YouTube player
|
||||||
- [PeerTubeify](https://gitlab.com/Ealhad/peertubeify): On YouTube, displays a link to the same video on PeerTube, if it exists.
|
- [PeerTubeify](https://gitlab.com/Ealhad/peertubeify): On YouTube, displays a link to the same video on PeerTube, if it exists.
|
||||||
- [MusicPiped](https://github.com/deep-gaurav/MusicPiped): A materialistic music player that streams music from YouTube.
|
- [MusicPiped](https://github.com/deep-gaurav/MusicPiped): A materialistic music player that streams music from YouTube.
|
||||||
|
|
||||||
|
|||||||
@@ -21,10 +21,9 @@ body {
|
|||||||
color: #f0f0f0;
|
color: #f0f0f0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pure-form > fieldset > input,
|
input,
|
||||||
.pure-control-group > input,
|
select,
|
||||||
.pure-form > fieldset > select,
|
textarea {
|
||||||
.pure-control-group > select {
|
|
||||||
color: rgba(35, 35, 35, 1);
|
color: rgba(35, 35, 35, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,25 @@
|
|||||||
|
html,
|
||||||
|
body {
|
||||||
|
font-family: BlinkMacSystemFont, -apple-system, "Segoe UI", Roboto, Oxygen,
|
||||||
|
Ubuntu, Cantarell, "Fira Sans", "Droid Sans", "Helvetica Neue", Helvetica,
|
||||||
|
Arial, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
.deleted {
|
.deleted {
|
||||||
background-color: rgb(255, 0, 0, 0.5);
|
background-color: rgb(255, 0, 0, 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.channel-profile > * {
|
||||||
|
font-size: 1.17em;
|
||||||
|
font-weight: bold;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
|
||||||
|
.channel-profile > img {
|
||||||
|
width: 48px;
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
|
||||||
.channel-owner {
|
.channel-owner {
|
||||||
background-color: #008bec;
|
background-color: #008bec;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
@@ -92,6 +110,7 @@ img.thumbnail {
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: 0;
|
top: 0;
|
||||||
|
object-fit: cover;
|
||||||
}
|
}
|
||||||
|
|
||||||
.length {
|
.length {
|
||||||
@@ -102,9 +121,8 @@ img.thumbnail {
|
|||||||
border-radius: 2px;
|
border-radius: 2px;
|
||||||
padding: 2px;
|
padding: 2px;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
font-family: sans-serif;
|
right: 0.25em;
|
||||||
right: 0.5em;
|
bottom: -0.75em;
|
||||||
bottom: -0.5em;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.watched {
|
.watched {
|
||||||
@@ -115,7 +133,6 @@ img.thumbnail {
|
|||||||
border-radius: 2px;
|
border-radius: 2px;
|
||||||
padding: 4px 8px 4px 8px;
|
padding: 4px 8px 4px 8px;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
font-family: sans-serif;
|
|
||||||
left: 0.2em;
|
left: 0.2em;
|
||||||
top: -0.7em;
|
top: -0.7em;
|
||||||
}
|
}
|
||||||
@@ -145,9 +162,12 @@ img.thumbnail {
|
|||||||
|
|
||||||
.navbar .index-link {
|
.navbar .index-link {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
|
display: inline;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar > .searchbar .pure-form input[type="search"] {
|
.navbar > .searchbar .pure-form input[type="search"] {
|
||||||
|
margin-bottom: 1px;
|
||||||
|
|
||||||
border-top: 0;
|
border-top: 0;
|
||||||
border-left: 0;
|
border-left: 0;
|
||||||
border-right: 0;
|
border-right: 0;
|
||||||
@@ -158,7 +178,6 @@ img.thumbnail {
|
|||||||
|
|
||||||
box-shadow: none;
|
box-shadow: none;
|
||||||
|
|
||||||
transition: 0.1s border-bottom;
|
|
||||||
-webkit-appearance: none;
|
-webkit-appearance: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -177,6 +196,7 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
|
|
||||||
/* attract focus to the searchbar by adding a subtle transition */
|
/* attract focus to the searchbar by adding a subtle transition */
|
||||||
.navbar > .searchbar .pure-form input[type="search"]:focus {
|
.navbar > .searchbar .pure-form input[type="search"]:focus {
|
||||||
|
margin-bottom: 0px;
|
||||||
border-bottom: 2px solid #aaa;
|
border-bottom: 2px solid #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,6 +242,11 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
.navbar > .searchbar > form {
|
.navbar > .searchbar > form {
|
||||||
width: 60%;
|
width: 60%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size: 1.25em;
|
||||||
|
margin: 0.42em 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (max-width: 320px) {
|
@media screen and (max-width: 320px) {
|
||||||
@@ -258,13 +283,27 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Control Bar */
|
/* Control Bar */
|
||||||
@media screen and (max-width: 480px) {
|
@media screen and (max-width: 640px) {
|
||||||
.video-js .vjs-control-bar,
|
.video-js .vjs-control-bar,
|
||||||
.vjs-menu-button-popup .vjs-menu .vjs-menu-content {
|
.vjs-menu-button-popup .vjs-menu .vjs-menu-content {
|
||||||
overflow: -webkit-paged-x;
|
overflow-x: scroll;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ul.vjs-menu-content::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vjs-user-inactive {
|
||||||
|
cursor: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-js .vjs-text-track-display > div > div > div {
|
||||||
|
background-color: rgba(0, 0, 0, 0.75) !important;
|
||||||
|
border-radius: 9px !important;
|
||||||
|
padding: 5px !important;
|
||||||
|
}
|
||||||
|
|
||||||
.vjs-play-control,
|
.vjs-play-control,
|
||||||
.vjs-volume-panel,
|
.vjs-volume-panel,
|
||||||
.vjs-current-time,
|
.vjs-current-time,
|
||||||
@@ -279,7 +318,8 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
order: 2;
|
order: 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vjs-quality-selector {
|
.vjs-quality-selector,
|
||||||
|
.video-js .vjs-http-source-selector {
|
||||||
order: 3;
|
order: 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -295,9 +335,22 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
order: 6;
|
order: 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.vjs-playback-rate > .vjs-menu {
|
||||||
|
width: 50px;
|
||||||
|
}
|
||||||
|
|
||||||
.vjs-control-bar {
|
.vjs-control-bar {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
|
scrollbar-width: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vjs-control-bar::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-js .vjs-icon-cog {
|
||||||
|
font-size: 18px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.video-js .vjs-control-bar,
|
.video-js .vjs-control-bar,
|
||||||
@@ -322,6 +375,11 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
background-color: rgba(15, 15, 15, 0.5);
|
background-color: rgba(15, 15, 15, 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fieldset > select,
|
||||||
|
span > select {
|
||||||
|
color: rgba(49, 49, 51, 1);
|
||||||
|
}
|
||||||
|
|
||||||
.video-js .vjs-load-progress,
|
.video-js .vjs-load-progress,
|
||||||
.video-js .vjs-load-progress div {
|
.video-js .vjs-load-progress div {
|
||||||
background: rgba(87, 87, 88, 1);
|
background: rgba(87, 87, 88, 1);
|
||||||
@@ -336,9 +394,16 @@ input[type="search"]::-webkit-search-cancel-button {
|
|||||||
background-color: rgba(0, 182, 240, 1);
|
background-color: rgba(0, 182, 240, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Overlay */
|
||||||
|
.video-js .vjs-overlay {
|
||||||
|
background-color: rgba(35, 35, 35, 0.75);
|
||||||
|
color: rgba(255, 255, 255, 1);
|
||||||
|
}
|
||||||
|
|
||||||
/* ProgressBar marker */
|
/* ProgressBar marker */
|
||||||
.vjs-marker {
|
.vjs-marker {
|
||||||
background-color: rgba(255, 255, 255, 1);
|
background-color: rgba(255, 255, 255, 1);
|
||||||
|
z-index: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Big "Play" Button */
|
/* Big "Play" Button */
|
||||||
@@ -385,3 +450,22 @@ video.video-js {
|
|||||||
.pure-control-group label {
|
.pure-control-group label {
|
||||||
word-wrap: normal;
|
word-wrap: normal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.video-js.player-style-invidious {
|
||||||
|
/* This is already the default */
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-js.player-style-youtube .vjs-control-bar {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
}
|
||||||
|
.video-js.player-style-youtube .vjs-big-play-button {
|
||||||
|
/*
|
||||||
|
Styles copied from video-js.min.css, definition of
|
||||||
|
.vjs-big-play-centered .vjs-big-play-button
|
||||||
|
*/
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
margin-top: -0.81666em;
|
||||||
|
margin-left: -1.5em;
|
||||||
|
}
|
||||||
|
|||||||
4
assets/css/grids-responsive-min.css
vendored
4
assets/css/grids-responsive-min.css
vendored
File diff suppressed because one or more lines are too long
4
assets/css/ionicons.min.css
vendored
4
assets/css/ionicons.min.css
vendored
File diff suppressed because one or more lines are too long
@@ -10,7 +10,7 @@ a {
|
|||||||
|
|
||||||
/* All links that do not fit with the default color goes here */
|
/* All links that do not fit with the default color goes here */
|
||||||
a:not([data-id]) > .icon,
|
a:not([data-id]) > .icon,
|
||||||
.pure-u-md-1-5 > .h-box > a[href^="/watch?"],
|
.pure-u-lg-1-5 > .h-box > a[href^="/watch?"],
|
||||||
.playlist-restricted > ol > li > a {
|
.playlist-restricted > ol > li > a {
|
||||||
color: #303030;
|
color: #303030;
|
||||||
}
|
}
|
||||||
|
|||||||
6
assets/css/pure-min.css
vendored
6
assets/css/pure-min.css
vendored
File diff suppressed because one or more lines are too long
2
assets/css/video-js.min.css
vendored
2
assets/css/video-js.min.css
vendored
File diff suppressed because one or more lines are too long
7
assets/css/videojs-http-source-selector.css
Normal file
7
assets/css/videojs-http-source-selector.css
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
/**
|
||||||
|
* videojs-http-source-selector
|
||||||
|
* @version 1.1.6
|
||||||
|
* @copyright 2019 Justin Fujita <Justin@pivotshare.com>
|
||||||
|
* @license MIT
|
||||||
|
*/
|
||||||
|
.video-js.vjs-http-source-selector{display:block}
|
||||||
1
assets/css/videojs-overlay.css
Normal file
1
assets/css/videojs-overlay.css
Normal file
@@ -0,0 +1 @@
|
|||||||
|
.video-js .vjs-overlay{color:#fff;position:absolute;text-align:center}.video-js .vjs-overlay-no-background{max-width:33%}.video-js .vjs-overlay-background{background-color:#646464;background-color:rgba(255,255,255,0.4);border-radius:3px;padding:10px;width:33%}.video-js .vjs-overlay-top-left{top:5px;left:5px}.video-js .vjs-overlay-top{left:50%;margin-left:-16.5%;top:5px}.video-js .vjs-overlay-top-right{right:5px;top:5px}.video-js .vjs-overlay-right{right:5px;top:50%;transform:translateY(-50%)}.video-js .vjs-overlay-bottom-right{bottom:3.5em;right:5px}.video-js .vjs-overlay-bottom{bottom:3.5em;left:50%;margin-left:-16.5%}.video-js .vjs-overlay-bottom-left{bottom:3.5em;left:5px}.video-js .vjs-overlay-left{left:5px;top:50%;transform:translateY(-50%)}.video-js .vjs-overlay-center{left:50%;margin-left:-16.5%;top:50%;transform:translateY(-50%)}.video-js .vjs-no-flex .vjs-overlay-left,.video-js .vjs-no-flex .vjs-overlay-center,.video-js .vjs-no-flex .vjs-overlay-right{margin-top:-15px}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
/**
|
/**
|
||||||
* videojs-share
|
* videojs-share
|
||||||
* @version 3.0.0
|
* @version 3.2.1
|
||||||
* @copyright 2018 Mikhail Khazov <mkhazov.work@gmail.com>
|
* @copyright 2019 Mikhail Khazov <mkhazov.work@gmail.com>
|
||||||
* @license MIT
|
* @license MIT
|
||||||
*/
|
*/
|
||||||
.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-modal-dialog-content{display:flex;align-items:center;padding:0;background-image:linear-gradient(to bottom, rgba(0,0,0,0.77), rgba(0,0,0,0.75))}.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button{position:absolute;right:0;top:5px;width:30px;height:30px;color:#fff;cursor:pointer;opacity:0.9;transition:opacity 0.25s ease-out}.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button:before{content:'×';font-size:20px;line-height:15px}.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button:hover{opacity:1}.video-js .vjs-share{display:flex;flex-direction:column;justify-content:space-around;align-items:center;width:100%;height:100%;max-height:400px}.video-js .vjs-share__top,.video-js .vjs-share__middle,.video-js .vjs-share__bottom{display:flex}.video-js .vjs-share__top,.video-js .vjs-share__middle{flex-direction:column;justify-content:space-between}.video-js .vjs-share__middle{padding:0 25px}.video-js .vjs-share__title{align-self:center;font-size:22px;color:#fff}.video-js .vjs-share__subtitle{width:100%;margin:0 auto 12px;font-size:16px;color:#fff;opacity:0.7}.video-js .vjs-share__short-link-wrapper{position:relative;display:block;width:100%;height:40px;margin:0 auto;margin-bottom:15px;border:0;color:rgba(255,255,255,0.65);background-color:#363636;outline:none;overflow:hidden;flex-shrink:0}.video-js .vjs-share__short-link{display:block;width:100%;height:100%;padding:0 40px 0 15px;border:0;color:rgba(255,255,255,0.65);background-color:#363636;outline:none}.video-js .vjs-share__btn{position:absolute;right:0;bottom:0;height:40px;width:40px;display:flex;align-items:center;padding:0 11px;border:0;color:#fff;background-color:#2e2e2e;background-size:18px 19px;background-position:center;background-repeat:no-repeat;cursor:pointer;outline:none;transition:width 0.3s ease-out, padding 0.3s ease-out}.video-js .vjs-share__btn svg{flex-shrink:0}.video-js .vjs-share__btn span{position:relative;padding-left:10px;opacity:0;transition:opacity 0.3s ease-out}.video-js .vjs-share__btn:hover{justify-content:center;width:100%;padding:0 40px;background-image:none}.video-js .vjs-share__btn:hover span{opacity:1}.video-js .vjs-share__socials{display:flex;flex-wrap:wrap;justify-content:center;align-content:flex-start;transition:width 0.3s ease-out, height 0.3s ease-out}.video-js .vjs-share__social{display:flex;justify-content:center;align-items:center;flex-shrink:0;width:32px;height:32px;margin-right:6px;margin-bottom:6px;cursor:pointer;font-size:8px;transition:transform 0.3s ease-out, filter 0.2s ease-out;border:none;outline:none}.video-js .vjs-share__social:hover{filter:brightness(115%)}.video-js .vjs-share__social svg{overflow:visible;max-height:24px}.video-js .vjs-share__social_vk{background-color:#5d7294}.video-js .vjs-share__social_ok{background-color:#ed7c20}.video-js .vjs-share__social_mail{background-color:#134785}.video-js .vjs-share__social_tw{background-color:#76aaeb}.video-js .vjs-share__social_reddit{background-color:#ff4500}.video-js .vjs-share__social_fbFeed{background-color:#475995}.video-js .vjs-share__social_messenger{background-color:#0084ff}.video-js .vjs-share__social_gp{background-color:#d53f35}.video-js .vjs-share__social_linkedin{background-color:#0077b5}.video-js .vjs-share__social_viber{background-color:#766db5}.video-js .vjs-share__social_telegram{background-color:#4bb0e2}.video-js .vjs-share__social_whatsapp{background-color:#78c870}.video-js .vjs-share__bottom{justify-content:center}@media (max-height: 220px){.video-js .vjs-share .hidden-xs{display:none}}@media (max-height: 350px){.video-js .vjs-share .hidden-sm{display:none}}@media (min-height: 400px){.video-js .vjs-share__title{margin-bottom:15px}.video-js .vjs-share__short-link-wrapper{margin-bottom:30px}}@media (min-width: 320px){.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button{right:5px;top:10px}}@media (min-width: 660px){.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button{right:20px;top:20px}.video-js .vjs-share__social{width:40px;height:40px}}
|
.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-modal-dialog-content{display:flex;align-items:center;padding:0;background-image:linear-gradient(to bottom, rgba(0,0,0,0.77), rgba(0,0,0,0.75))}.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button{position:absolute;right:0;top:5px;width:30px;height:30px;color:#fff;cursor:pointer;opacity:0.9;transition:opacity 0.25s ease-out}.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button:before{content:'×';font-size:20px;line-height:15px}.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button:hover{opacity:1}.video-js .vjs-share{display:flex;flex-direction:column;justify-content:space-around;align-items:center;width:100%;height:100%;max-height:400px}.video-js .vjs-share__top,.video-js .vjs-share__middle,.video-js .vjs-share__bottom{display:flex}.video-js .vjs-share__top,.video-js .vjs-share__middle{flex-direction:column;justify-content:space-between}.video-js .vjs-share__middle{padding:0 25px}.video-js .vjs-share__title{align-self:center;font-size:22px;color:#fff}.video-js .vjs-share__subtitle{width:100%;margin:0 auto 12px;font-size:16px;color:#fff;opacity:0.7}.video-js .vjs-share__short-link-wrapper{position:relative;display:block;width:100%;height:40px;margin:0 auto;margin-bottom:15px;border:0;color:rgba(255,255,255,0.65);background-color:#363636;outline:none;overflow:hidden;flex-shrink:0}.video-js .vjs-share__short-link{display:block;width:100%;height:100%;padding:0 40px 0 15px;border:0;color:rgba(255,255,255,0.65);background-color:#363636;outline:none}.video-js .vjs-share__btn{position:absolute;right:0;bottom:0;height:40px;width:40px;display:flex;align-items:center;padding:0 11px;border:0;color:#fff;background-color:#2e2e2e;background-size:18px 19px;background-position:center;background-repeat:no-repeat;cursor:pointer;outline:none;transition:width 0.3s ease-out, padding 0.3s ease-out}.video-js .vjs-share__btn svg{flex-shrink:0}.video-js .vjs-share__btn span{position:relative;padding-left:10px;opacity:0;transition:opacity 0.3s ease-out}.video-js .vjs-share__btn:hover{justify-content:center;width:100%;padding:0 40px;background-image:none}.video-js .vjs-share__btn:hover span{opacity:1}.video-js .vjs-share__socials{display:flex;flex-wrap:wrap;justify-content:center;align-content:flex-start;transition:width 0.3s ease-out, height 0.3s ease-out}.video-js .vjs-share__social{display:flex;justify-content:center;align-items:center;flex-shrink:0;width:32px;height:32px;margin-right:6px;margin-bottom:6px;cursor:pointer;font-size:8px;transition:transform 0.3s ease-out, filter 0.2s ease-out;border:none;outline:none}.video-js .vjs-share__social:hover{filter:brightness(115%)}.video-js .vjs-share__social svg{overflow:visible;max-height:24px}.video-js .vjs-share__social_vk{background-color:#5d7294}.video-js .vjs-share__social_ok{background-color:#ed7c20}.video-js .vjs-share__social_mail,.video-js .vjs-share__social_email{background-color:#134785}.video-js .vjs-share__social_tw{background-color:#76aaeb}.video-js .vjs-share__social_reddit{background-color:#ff4500}.video-js .vjs-share__social_fbFeed{background-color:#475995}.video-js .vjs-share__social_messenger{background-color:#0084ff}.video-js .vjs-share__social_gp{background-color:#d53f35}.video-js .vjs-share__social_linkedin{background-color:#0077b5}.video-js .vjs-share__social_viber{background-color:#766db5}.video-js .vjs-share__social_telegram{background-color:#4bb0e2}.video-js .vjs-share__social_whatsapp{background-color:#78c870}.video-js .vjs-share__bottom{justify-content:center}@media (max-height: 220px){.video-js .vjs-share .hidden-xs{display:none}}@media (max-height: 350px){.video-js .vjs-share .hidden-sm{display:none}}@media (min-height: 400px){.video-js .vjs-share__title{margin-bottom:15px}.video-js .vjs-share__short-link-wrapper{margin-bottom:30px}}@media (min-width: 320px){.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button{right:5px;top:10px}}@media (min-width: 660px){.video-js.vjs-videojs-share_open .vjs-modal-dialog .vjs-close-button{right:20px;top:20px}.video-js .vjs-share__social{width:40px;height:40px}}
|
||||||
|
|||||||
7
assets/css/videojs-vtt-thumbnails.css
Normal file
7
assets/css/videojs-vtt-thumbnails.css
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
/**
|
||||||
|
* videojs-vtt-thumbnails
|
||||||
|
* @version 0.0.13
|
||||||
|
* @copyright 2019 Chris Boustead <chris@forgemotion.com>
|
||||||
|
* @license MIT
|
||||||
|
*/
|
||||||
|
.video-js.vjs-vtt-thumbnails{display:block}.video-js .vjs-vtt-thumbnail-display{position:absolute;bottom:85%;pointer-events:none;box-shadow:0 0 7px rgba(0,0,0,0.6)}
|
||||||
1
assets/css/videojs-youtube-annotations.min.css
vendored
Normal file
1
assets/css/videojs-youtube-annotations.min.css
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
.__cxt-ar-annotations-container__{--annotation-close-size: 20px;position:absolute;width:100%;height:100%;top:0;left:0;pointer-events:none;overflow:hidden}.__cxt-ar-annotation__{position:absolute;box-sizing:border-box;font-family:Arial,sans-serif;color:#fff;z-index:20;pointer-events:auto}.__cxt-ar-annotation__ span{position:absolute;left:0;top:0;overflow:hidden;word-wrap:break-word;white-space:pre-wrap;pointer-events:none;box-sizing:border-box;padding:2%;user-select:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none}.__cxt-ar-annotation-close__{display:none;position:absolute;width:var(--annotation-close-size);height:var(--annotation-close-size);cursor:pointer;right:calc(var(--annotation-close-size)/-1.8);top:calc(var(--annotation-close-size)/-1.8);z-index:1}.__cxt-ar-annotation__:hover:not([hidden]):not([data-ar-closed]) .__cxt-ar-annotation-close__{display:block}.__cxt-ar-annotation__[hidden]{display:none!important}.__cxt-ar-annotation__[data-ar-type=highlight]{border:1px solid rgba(255,255,255,.1);background-color:transparent}.__cxt-ar-annotation__[data-ar-type=highlight]:hover{border:1px solid rgba(255,255,255,.5);background-color:transparent}.__cxt-ar-annotation__ svg{pointer-events:all}
|
||||||
Binary file not shown.
@@ -1,13 +1,13 @@
|
|||||||
<?xml version="1.0" standalone="no"?>
|
<?xml version="1.0" standalone="no"?>
|
||||||
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" >
|
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" >
|
||||||
<!--
|
<!--
|
||||||
2018-6-14: Created with FontForge (http://fontforge.org)
|
2019-5-24: Created with FontForge (http://fontforge.org)
|
||||||
-->
|
-->
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1">
|
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1">
|
||||||
<metadata>
|
<metadata>
|
||||||
Created by FontForge 20160407 at Thu Jun 14 08:50:34 2018
|
Created by FontForge 20160407 at Fri May 24 15:45:40 2019
|
||||||
By Adam Bradley
|
By Adam Bradley
|
||||||
Copyright (c) 2018, Adam Bradley
|
Copyright (c) 2019, Adam Bradley
|
||||||
</metadata>
|
</metadata>
|
||||||
<defs>
|
<defs>
|
||||||
<font id="Ionicons" horiz-adv-x="416" >
|
<font id="Ionicons" horiz-adv-x="416" >
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 305 KiB After Width: | Height: | Size: 305 KiB |
Binary file not shown.
Binary file not shown.
Binary file not shown.
101
assets/js/community.js
Normal file
101
assets/js/community.js
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
String.prototype.supplant = function (o) {
|
||||||
|
return this.replace(/{([^{}]*)}/g, function (a, b) {
|
||||||
|
var r = o[b];
|
||||||
|
return typeof r === 'string' || typeof r === 'number' ? r : a;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function hide_youtube_replies(event) {
|
||||||
|
var target = event.target;
|
||||||
|
|
||||||
|
sub_text = target.getAttribute('data-inner-text');
|
||||||
|
inner_text = target.getAttribute('data-sub-text');
|
||||||
|
|
||||||
|
body = target.parentNode.parentNode.children[1];
|
||||||
|
body.style.display = 'none';
|
||||||
|
|
||||||
|
target.innerHTML = sub_text;
|
||||||
|
target.onclick = show_youtube_replies;
|
||||||
|
target.setAttribute('data-inner-text', inner_text);
|
||||||
|
target.setAttribute('data-sub-text', sub_text);
|
||||||
|
}
|
||||||
|
|
||||||
|
function show_youtube_replies(event) {
|
||||||
|
var target = event.target;
|
||||||
|
|
||||||
|
sub_text = target.getAttribute('data-inner-text');
|
||||||
|
inner_text = target.getAttribute('data-sub-text');
|
||||||
|
|
||||||
|
body = target.parentNode.parentNode.children[1];
|
||||||
|
body.style.display = '';
|
||||||
|
|
||||||
|
target.innerHTML = sub_text;
|
||||||
|
target.onclick = hide_youtube_replies;
|
||||||
|
target.setAttribute('data-inner-text', inner_text);
|
||||||
|
target.setAttribute('data-sub-text', sub_text);
|
||||||
|
}
|
||||||
|
|
||||||
|
function number_with_separator(val) {
|
||||||
|
while (/(\d+)(\d{3})/.test(val.toString())) {
|
||||||
|
val = val.toString().replace(/(\d+)(\d{3})/, '$1' + ',' + '$2');
|
||||||
|
}
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_youtube_replies(target, load_more) {
|
||||||
|
var continuation = target.getAttribute('data-continuation');
|
||||||
|
|
||||||
|
var body = target.parentNode.parentNode;
|
||||||
|
var fallback = body.innerHTML;
|
||||||
|
body.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3>';
|
||||||
|
|
||||||
|
var url = '/api/v1/channels/comments/' + community_data.ucid +
|
||||||
|
'?format=html' +
|
||||||
|
'&hl=' + community_data.preferences.locale +
|
||||||
|
'&thin_mode=' + community_data.preferences.thin_mode +
|
||||||
|
'&continuation=' + continuation;
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', url, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status == 200) {
|
||||||
|
if (load_more) {
|
||||||
|
body = body.parentNode.parentNode;
|
||||||
|
body.removeChild(body.lastElementChild);
|
||||||
|
body.innerHTML += xhr.response.contentHtml;
|
||||||
|
} else {
|
||||||
|
body.removeChild(body.lastElementChild);
|
||||||
|
|
||||||
|
var p = document.createElement('p');
|
||||||
|
var a = document.createElement('a');
|
||||||
|
p.appendChild(a);
|
||||||
|
|
||||||
|
a.href = 'javascript:void(0)';
|
||||||
|
a.onclick = hide_youtube_replies;
|
||||||
|
a.setAttribute('data-sub-text', community_data.hide_replies_text);
|
||||||
|
a.setAttribute('data-inner-text', community_data.show_replies_text);
|
||||||
|
a.innerText = community_data.hide_replies_text;
|
||||||
|
|
||||||
|
var div = document.createElement('div');
|
||||||
|
div.innerHTML = xhr.response.contentHtml;
|
||||||
|
|
||||||
|
body.appendChild(p);
|
||||||
|
body.appendChild(div);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
body.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Pulling comments failed.');
|
||||||
|
body.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
28
assets/js/dash.mediaplayer.min.js
vendored
28
assets/js/dash.mediaplayer.min.js
vendored
File diff suppressed because one or more lines are too long
102
assets/js/embed.js
Normal file
102
assets/js/embed.js
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
function get_playlist(plid, retries) {
|
||||||
|
if (retries == undefined) retries = 5;
|
||||||
|
|
||||||
|
if (retries <= 0) {
|
||||||
|
console.log('Failed to pull playlist');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (plid.startsWith('RD')) {
|
||||||
|
var plid_url = '/api/v1/mixes/' + plid +
|
||||||
|
'?continuation=' + video_data.id +
|
||||||
|
'&format=html&hl=' + video_data.preferences.locale;
|
||||||
|
} else {
|
||||||
|
var plid_url = '/api/v1/playlists/' + plid +
|
||||||
|
'?index=' + video_data.index +
|
||||||
|
'&continuation' + video_data.id +
|
||||||
|
'&format=html&hl=' + video_data.preferences.locale;
|
||||||
|
}
|
||||||
|
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', plid_url, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState === 4) {
|
||||||
|
if (xhr.status === 200) {
|
||||||
|
if (xhr.response.nextVideo) {
|
||||||
|
player.on('ended', function () {
|
||||||
|
var url = new URL('https://example.com/embed/' + xhr.response.nextVideo);
|
||||||
|
|
||||||
|
if (video_data.params.autoplay || video_data.params.continue_autoplay) {
|
||||||
|
url.searchParams.set('autoplay', '1');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.listen !== video_data.preferences.listen) {
|
||||||
|
url.searchParams.set('listen', video_data.params.listen);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.speed !== video_data.preferences.speed) {
|
||||||
|
url.searchParams.set('speed', video_data.params.speed);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.local !== video_data.preferences.local) {
|
||||||
|
url.searchParams.set('local', video_data.params.local);
|
||||||
|
}
|
||||||
|
|
||||||
|
url.searchParams.set('list', plid);
|
||||||
|
if (!plid.startsWith('RD')) {
|
||||||
|
url.searchParams.set('index', xhr.response.index);
|
||||||
|
}
|
||||||
|
location.assign(url.pathname + url.search);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
console.log('Pulling playlist failed... ' + retries + '/5');
|
||||||
|
setTimeout(function () { get_playlist(plid, retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Pulling playlist failed... ' + retries + '/5');
|
||||||
|
get_playlist(plid, retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('load', function (e) {
|
||||||
|
if (video_data.plid) {
|
||||||
|
get_playlist(video_data.plid);
|
||||||
|
} else if (video_data.video_series) {
|
||||||
|
player.on('ended', function () {
|
||||||
|
var url = new URL('https://example.com/embed/' + video_data.video_series.shift());
|
||||||
|
|
||||||
|
if (video_data.params.autoplay || video_data.params.continue_autoplay) {
|
||||||
|
url.searchParams.set('autoplay', '1');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.listen !== video_data.preferences.listen) {
|
||||||
|
url.searchParams.set('listen', video_data.params.listen);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.speed !== video_data.preferences.speed) {
|
||||||
|
url.searchParams.set('speed', video_data.params.speed);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.local !== video_data.preferences.local) {
|
||||||
|
url.searchParams.set('local', video_data.params.local);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.video_series.length !== 0) {
|
||||||
|
url.searchParams.set('playlist', video_data.video_series.join(','))
|
||||||
|
}
|
||||||
|
|
||||||
|
location.assign(url.pathname + url.search);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
141
assets/js/notifications.js
Normal file
141
assets/js/notifications.js
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
var notifications, delivered;
|
||||||
|
|
||||||
|
function get_subscriptions(callback, retries) {
|
||||||
|
if (retries == undefined) retries = 5;
|
||||||
|
|
||||||
|
if (retries <= 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', '/api/v1/auth/subscriptions?fields=authorId', true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState === 4) {
|
||||||
|
if (xhr.status === 200) {
|
||||||
|
subscriptions = xhr.response;
|
||||||
|
callback(subscriptions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
console.log('Pulling subscriptions failed... ' + retries + '/5');
|
||||||
|
setTimeout(function () { get_subscriptions(callback, retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Pulling subscriptions failed... ' + retries + '/5');
|
||||||
|
get_subscriptions(callback, retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
function create_notification_stream(subscriptions) {
|
||||||
|
notifications = new SSE(
|
||||||
|
'/api/v1/auth/notifications?fields=videoId,title,author,authorId,publishedText,published,authorThumbnails,liveNow', {
|
||||||
|
withCredentials: true,
|
||||||
|
payload: 'topics=' + subscriptions.map(function (subscription) { return subscription.authorId }).join(','),
|
||||||
|
headers: { 'Content-Type': 'application/x-www-form-urlencoded' }
|
||||||
|
});
|
||||||
|
delivered = [];
|
||||||
|
|
||||||
|
var start_time = Math.round(new Date() / 1000);
|
||||||
|
|
||||||
|
notifications.onmessage = function (event) {
|
||||||
|
if (!event.id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var notification = JSON.parse(event.data);
|
||||||
|
console.log('Got notification:', notification);
|
||||||
|
|
||||||
|
if (start_time < notification.published && !delivered.includes(notification.videoId)) {
|
||||||
|
if (Notification.permission === 'granted') {
|
||||||
|
var system_notification =
|
||||||
|
new Notification((notification.liveNow ? notification_data.live_now_text : notification_data.upload_text).replace('`x`', notification.author), {
|
||||||
|
body: notification.title,
|
||||||
|
icon: '/ggpht' + new URL(notification.authorThumbnails[2].url).pathname,
|
||||||
|
img: '/ggpht' + new URL(notification.authorThumbnails[4].url).pathname,
|
||||||
|
tag: notification.videoId
|
||||||
|
});
|
||||||
|
|
||||||
|
system_notification.onclick = function (event) {
|
||||||
|
window.open('/watch?v=' + event.currentTarget.tag, '_blank');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
delivered.push(notification.videoId);
|
||||||
|
localStorage.setItem('notification_count', parseInt(localStorage.getItem('notification_count') || '0') + 1);
|
||||||
|
var notification_ticker = document.getElementById('notification_ticker');
|
||||||
|
|
||||||
|
if (parseInt(localStorage.getItem('notification_count')) > 0) {
|
||||||
|
notification_ticker.innerHTML =
|
||||||
|
'<span id="notification_count">' + localStorage.getItem('notification_count') + '</span> <i class="icon ion-ios-notifications"></i>';
|
||||||
|
} else {
|
||||||
|
notification_ticker.innerHTML =
|
||||||
|
'<i class="icon ion-ios-notifications-outline"></i>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
notifications.addEventListener('error', handle_notification_error);
|
||||||
|
notifications.stream();
|
||||||
|
}
|
||||||
|
|
||||||
|
function handle_notification_error(event) {
|
||||||
|
console.log('Something went wrong with notifications, trying to reconnect...');
|
||||||
|
notifications = { close: function () { } };
|
||||||
|
setTimeout(function () { get_subscriptions(create_notification_stream) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('load', function (e) {
|
||||||
|
localStorage.setItem('notification_count', document.getElementById('notification_count') ? document.getElementById('notification_count').innerText : '0');
|
||||||
|
|
||||||
|
if (localStorage.getItem('stream')) {
|
||||||
|
localStorage.removeItem('stream');
|
||||||
|
} else {
|
||||||
|
setTimeout(function () {
|
||||||
|
if (!localStorage.getItem('stream')) {
|
||||||
|
notifications = { close: function () { } };
|
||||||
|
localStorage.setItem('stream', true);
|
||||||
|
get_subscriptions(create_notification_stream);
|
||||||
|
}
|
||||||
|
}, Math.random() * 1000 + 50);
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('storage', function (e) {
|
||||||
|
if (e.key === 'stream' && !e.newValue) {
|
||||||
|
if (notifications) {
|
||||||
|
localStorage.setItem('stream', true);
|
||||||
|
} else {
|
||||||
|
setTimeout(function () {
|
||||||
|
if (!localStorage.getItem('stream')) {
|
||||||
|
notifications = { close: function () { } };
|
||||||
|
localStorage.setItem('stream', true);
|
||||||
|
get_subscriptions(create_notification_stream);
|
||||||
|
}
|
||||||
|
}, Math.random() * 1000 + 50);
|
||||||
|
}
|
||||||
|
} else if (e.key === 'notification_count') {
|
||||||
|
var notification_ticker = document.getElementById('notification_ticker');
|
||||||
|
|
||||||
|
if (parseInt(e.newValue) > 0) {
|
||||||
|
notification_ticker.innerHTML =
|
||||||
|
'<span id="notification_count">' + e.newValue + '</span> <i class="icon ion-ios-notifications"></i>';
|
||||||
|
} else {
|
||||||
|
notification_ticker.innerHTML =
|
||||||
|
'<i class="icon ion-ios-notifications-outline"></i>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener('unload', function (e) {
|
||||||
|
if (notifications) {
|
||||||
|
localStorage.removeItem('stream');
|
||||||
|
}
|
||||||
|
});
|
||||||
471
assets/js/player.js
Normal file
471
assets/js/player.js
Normal file
@@ -0,0 +1,471 @@
|
|||||||
|
var options = {
|
||||||
|
preload: 'auto',
|
||||||
|
liveui: true,
|
||||||
|
playbackRates: [0.25, 0.5, 0.75, 1.0, 1.25, 1.5, 1.75, 2.0],
|
||||||
|
controlBar: {
|
||||||
|
children: [
|
||||||
|
'playToggle',
|
||||||
|
'volumePanel',
|
||||||
|
'currentTimeDisplay',
|
||||||
|
'timeDivider',
|
||||||
|
'durationDisplay',
|
||||||
|
'progressControl',
|
||||||
|
'remainingTimeDisplay',
|
||||||
|
'captionsButton',
|
||||||
|
'qualitySelector',
|
||||||
|
'playbackRateMenuButton',
|
||||||
|
'fullscreenToggle'
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (player_data.aspect_ratio) {
|
||||||
|
options.aspectRatio = player_data.aspect_ratio;
|
||||||
|
}
|
||||||
|
|
||||||
|
var embed_url = new URL(location);
|
||||||
|
embed_url.searchParams.delete('v');
|
||||||
|
short_url = location.origin + '/' + video_data.id + embed_url.search;
|
||||||
|
embed_url = location.origin + '/embed/' + video_data.id + embed_url.search;
|
||||||
|
|
||||||
|
var shareOptions = {
|
||||||
|
socials: ['fbFeed', 'tw', 'reddit', 'email'],
|
||||||
|
|
||||||
|
url: short_url,
|
||||||
|
title: player_data.title,
|
||||||
|
description: player_data.description,
|
||||||
|
image: player_data.thumbnail,
|
||||||
|
embedCode: "<iframe id='ivplayer' type='text/html' width='640' height='360' src='" + embed_url + "' frameborder='0'></iframe>"
|
||||||
|
}
|
||||||
|
|
||||||
|
var player = videojs('player', options);
|
||||||
|
|
||||||
|
if (location.pathname.startsWith('/embed/')) {
|
||||||
|
player.overlay({
|
||||||
|
overlays: [{
|
||||||
|
start: 'loadstart',
|
||||||
|
content: '<h1><a rel="noopener" target="_blank" href="' + location.origin + '/watch?v=' + video_data.id + '">' + player_data.title + '</a></h1>',
|
||||||
|
end: 'playing',
|
||||||
|
align: 'top'
|
||||||
|
}, {
|
||||||
|
start: 'pause',
|
||||||
|
content: '<h1><a rel="noopener" target="_blank" href="' + location.origin + '/watch?v=' + video_data.id + '">' + player_data.title + '</a></h1>',
|
||||||
|
end: 'playing',
|
||||||
|
align: 'top'
|
||||||
|
}]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
player.on('error', function (event) {
|
||||||
|
if (player.error().code === 2 || player.error().code === 4) {
|
||||||
|
setInterval(setTimeout(function (event) {
|
||||||
|
console.log('An error occured in the player, reloading...');
|
||||||
|
|
||||||
|
var currentTime = player.currentTime();
|
||||||
|
var playbackRate = player.playbackRate();
|
||||||
|
var paused = player.paused();
|
||||||
|
|
||||||
|
player.load();
|
||||||
|
|
||||||
|
if (currentTime > 0.5) {
|
||||||
|
currentTime -= 0.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
player.currentTime(currentTime);
|
||||||
|
player.playbackRate(playbackRate);
|
||||||
|
|
||||||
|
if (!paused) {
|
||||||
|
player.play();
|
||||||
|
}
|
||||||
|
}, 5000), 5000);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Add markers
|
||||||
|
if (video_data.params.video_start > 0 || video_data.params.video_end > 0) {
|
||||||
|
var markers = [{ time: video_data.params.video_start, text: 'Start' }];
|
||||||
|
|
||||||
|
if (video_data.params.video_end < 0) {
|
||||||
|
markers.push({ time: video_data.length_seconds - 0.5, text: 'End' });
|
||||||
|
} else {
|
||||||
|
markers.push({ time: video_data.params.video_end, text: 'End' });
|
||||||
|
}
|
||||||
|
|
||||||
|
player.markers({
|
||||||
|
onMarkerReached: function (marker) {
|
||||||
|
if (marker.text === 'End') {
|
||||||
|
if (player.loop()) {
|
||||||
|
player.markers.prev('Start');
|
||||||
|
} else {
|
||||||
|
player.pause();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
markers: markers
|
||||||
|
});
|
||||||
|
|
||||||
|
player.currentTime(video_data.params.video_start);
|
||||||
|
}
|
||||||
|
|
||||||
|
player.volume(video_data.params.volume / 100);
|
||||||
|
player.playbackRate(video_data.params.speed);
|
||||||
|
|
||||||
|
player.on('waiting', function () {
|
||||||
|
if (player.playbackRate() > 1 && player.liveTracker.isLive() && player.liveTracker.atLiveEdge()) {
|
||||||
|
console.log('Player has caught up to source, resetting playbackRate.')
|
||||||
|
player.playbackRate(1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (video_data.premiere_timestamp && Math.round(new Date() / 1000) < video_data.premiere_timestamp) {
|
||||||
|
player.getChild('bigPlayButton').hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.autoplay) {
|
||||||
|
var bpb = player.getChild('bigPlayButton');
|
||||||
|
bpb.hide();
|
||||||
|
|
||||||
|
player.ready(function () {
|
||||||
|
new Promise(function (resolve, reject) {
|
||||||
|
setTimeout(() => resolve(1), 1);
|
||||||
|
}).then(function (result) {
|
||||||
|
var promise = player.play();
|
||||||
|
|
||||||
|
if (promise !== undefined) {
|
||||||
|
promise.then(_ => {
|
||||||
|
}).catch(error => {
|
||||||
|
bpb.show();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!video_data.params.listen && video_data.params.quality === 'dash') {
|
||||||
|
player.httpSourceSelector();
|
||||||
|
}
|
||||||
|
|
||||||
|
player.vttThumbnails({
|
||||||
|
src: location.origin + '/api/v1/storyboards/' + video_data.id + '?height=90'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Enable annotations
|
||||||
|
if (!video_data.params.listen && video_data.params.annotations) {
|
||||||
|
window.addEventListener('load', function (e) {
|
||||||
|
var video_container = document.getElementById('player');
|
||||||
|
let xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'text';
|
||||||
|
xhr.timeout = 60000;
|
||||||
|
xhr.open('GET', '/api/v1/annotations/' + video_data.id, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState === 4) {
|
||||||
|
if (xhr.status === 200) {
|
||||||
|
videojs.registerPlugin('youtubeAnnotationsPlugin', youtubeAnnotationsPlugin);
|
||||||
|
if (!player.paused()) {
|
||||||
|
player.youtubeAnnotationsPlugin({ annotationXml: xhr.response, videoContainer: video_container });
|
||||||
|
} else {
|
||||||
|
player.one('play', function (event) {
|
||||||
|
player.youtubeAnnotationsPlugin({ annotationXml: xhr.response, videoContainer: video_container });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('__ar_annotation_click', e => {
|
||||||
|
const { url, target, seconds } = e.detail;
|
||||||
|
var path = new URL(url);
|
||||||
|
|
||||||
|
if (path.href.startsWith('https://www.youtube.com/watch?') && seconds) {
|
||||||
|
path.search += '&t=' + seconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
path = path.pathname + path.search;
|
||||||
|
|
||||||
|
if (target === 'current') {
|
||||||
|
window.location.href = path;
|
||||||
|
} else if (target === 'new') {
|
||||||
|
window.open(path, '_blank');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function increase_volume(delta) {
|
||||||
|
const curVolume = player.volume();
|
||||||
|
let newVolume = curVolume + delta;
|
||||||
|
if (newVolume > 1) {
|
||||||
|
newVolume = 1;
|
||||||
|
} else if (newVolume < 0) {
|
||||||
|
newVolume = 0;
|
||||||
|
}
|
||||||
|
player.volume(newVolume);
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggle_muted() {
|
||||||
|
const isMuted = player.muted();
|
||||||
|
player.muted(!isMuted);
|
||||||
|
}
|
||||||
|
|
||||||
|
function skip_seconds(delta) {
|
||||||
|
const duration = player.duration();
|
||||||
|
const curTime = player.currentTime();
|
||||||
|
let newTime = curTime + delta;
|
||||||
|
if (newTime > duration) {
|
||||||
|
newTime = duration;
|
||||||
|
} else if (newTime < 0) {
|
||||||
|
newTime = 0;
|
||||||
|
}
|
||||||
|
player.currentTime(newTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
function set_time_percent(percent) {
|
||||||
|
const duration = player.duration();
|
||||||
|
const newTime = duration * (percent / 100);
|
||||||
|
player.currentTime(newTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggle_play() {
|
||||||
|
if (player.paused()) {
|
||||||
|
player.play();
|
||||||
|
} else {
|
||||||
|
player.pause();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const toggle_captions = (function () {
|
||||||
|
let toggledTrack = null;
|
||||||
|
const onChange = function (e) {
|
||||||
|
toggledTrack = null;
|
||||||
|
};
|
||||||
|
const bindChange = function (onOrOff) {
|
||||||
|
player.textTracks()[onOrOff]('change', onChange);
|
||||||
|
};
|
||||||
|
// Wrapper function to ignore our own emitted events and only listen
|
||||||
|
// to events emitted by Video.js on click on the captions menu items.
|
||||||
|
const setMode = function (track, mode) {
|
||||||
|
bindChange('off');
|
||||||
|
track.mode = mode;
|
||||||
|
window.setTimeout(function () {
|
||||||
|
bindChange('on');
|
||||||
|
}, 0);
|
||||||
|
};
|
||||||
|
bindChange('on');
|
||||||
|
return function () {
|
||||||
|
if (toggledTrack !== null) {
|
||||||
|
if (toggledTrack.mode !== 'showing') {
|
||||||
|
setMode(toggledTrack, 'showing');
|
||||||
|
} else {
|
||||||
|
setMode(toggledTrack, 'disabled');
|
||||||
|
}
|
||||||
|
toggledTrack = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Used as a fallback if no captions are currently active.
|
||||||
|
// TODO: Make this more intelligent by e.g. relying on browser language.
|
||||||
|
let fallbackCaptionsTrack = null;
|
||||||
|
|
||||||
|
const tracks = player.textTracks();
|
||||||
|
for (let i = 0; i < tracks.length; i++) {
|
||||||
|
const track = tracks[i];
|
||||||
|
if (track.kind !== 'captions') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fallbackCaptionsTrack === null) {
|
||||||
|
fallbackCaptionsTrack = track;
|
||||||
|
}
|
||||||
|
if (track.mode === 'showing') {
|
||||||
|
setMode(track, 'disabled');
|
||||||
|
toggledTrack = track;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback if no captions are currently active.
|
||||||
|
if (fallbackCaptionsTrack !== null) {
|
||||||
|
setMode(fallbackCaptionsTrack, 'showing');
|
||||||
|
toggledTrack = fallbackCaptionsTrack;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
})();
|
||||||
|
|
||||||
|
function toggle_fullscreen() {
|
||||||
|
if (player.isFullscreen()) {
|
||||||
|
player.exitFullscreen();
|
||||||
|
} else {
|
||||||
|
player.requestFullscreen();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function increase_playback_rate(steps) {
|
||||||
|
const maxIndex = options.playbackRates.length - 1;
|
||||||
|
const curIndex = options.playbackRates.indexOf(player.playbackRate());
|
||||||
|
let newIndex = curIndex + steps;
|
||||||
|
if (newIndex > maxIndex) {
|
||||||
|
newIndex = maxIndex;
|
||||||
|
} else if (newIndex < 0) {
|
||||||
|
newIndex = 0;
|
||||||
|
}
|
||||||
|
player.playbackRate(options.playbackRates[newIndex]);
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('keydown', e => {
|
||||||
|
if (e.target.tagName.toLowerCase() === 'input') {
|
||||||
|
// Ignore input when focus is on certain elements, e.g. form fields.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// See https://github.com/ctd1500/videojs-hotkeys/blob/bb4a158b2e214ccab87c2e7b95f42bc45c6bfd87/videojs.hotkeys.js#L310-L313
|
||||||
|
const isPlayerFocused = false
|
||||||
|
|| e.target === document.querySelector('.video-js')
|
||||||
|
|| e.target === document.querySelector('.vjs-tech')
|
||||||
|
|| e.target === document.querySelector('.iframeblocker')
|
||||||
|
|| e.target === document.querySelector('.vjs-control-bar')
|
||||||
|
;
|
||||||
|
let action = null;
|
||||||
|
|
||||||
|
const code = e.keyCode;
|
||||||
|
const decoratedKey =
|
||||||
|
e.key
|
||||||
|
+ (e.altKey ? '+alt' : '')
|
||||||
|
+ (e.ctrlKey ? '+ctrl' : '')
|
||||||
|
+ (e.metaKey ? '+meta' : '')
|
||||||
|
;
|
||||||
|
switch (decoratedKey) {
|
||||||
|
case ' ':
|
||||||
|
case 'k':
|
||||||
|
action = toggle_play;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'ArrowUp':
|
||||||
|
if (isPlayerFocused) {
|
||||||
|
action = increase_volume.bind(this, 0.1);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'ArrowDown':
|
||||||
|
if (isPlayerFocused) {
|
||||||
|
action = increase_volume.bind(this, -0.1);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'm':
|
||||||
|
action = toggle_muted;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'ArrowRight':
|
||||||
|
action = skip_seconds.bind(this, 5);
|
||||||
|
break;
|
||||||
|
case 'ArrowLeft':
|
||||||
|
action = skip_seconds.bind(this, -5);
|
||||||
|
break;
|
||||||
|
case 'l':
|
||||||
|
action = skip_seconds.bind(this, 10);
|
||||||
|
break;
|
||||||
|
case 'j':
|
||||||
|
action = skip_seconds.bind(this, -10);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '0':
|
||||||
|
case '1':
|
||||||
|
case '2':
|
||||||
|
case '3':
|
||||||
|
case '4':
|
||||||
|
case '5':
|
||||||
|
case '6':
|
||||||
|
case '7':
|
||||||
|
case '8':
|
||||||
|
case '9':
|
||||||
|
const percent = (code - 48) * 10;
|
||||||
|
action = set_time_percent.bind(this, percent);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'c':
|
||||||
|
action = toggle_captions;
|
||||||
|
break;
|
||||||
|
case 'f':
|
||||||
|
action = toggle_fullscreen;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'N':
|
||||||
|
action = next_video;
|
||||||
|
break;
|
||||||
|
case 'P':
|
||||||
|
// TODO: Add support to play back previous video.
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '.':
|
||||||
|
// TODO: Add support for next-frame-stepping.
|
||||||
|
break;
|
||||||
|
case ',':
|
||||||
|
// TODO: Add support for previous-frame-stepping.
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '>':
|
||||||
|
action = increase_playback_rate.bind(this, 1);
|
||||||
|
break;
|
||||||
|
case '<':
|
||||||
|
action = increase_playback_rate.bind(this, -1);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
console.info('Unhandled key down event: %s:', decoratedKey, e);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (action) {
|
||||||
|
e.preventDefault();
|
||||||
|
action();
|
||||||
|
}
|
||||||
|
}, false);
|
||||||
|
|
||||||
|
// Add support for controlling the player volume by scrolling over it. Adapted from
|
||||||
|
// https://github.com/ctd1500/videojs-hotkeys/blob/bb4a158b2e214ccab87c2e7b95f42bc45c6bfd87/videojs.hotkeys.js#L292-L328
|
||||||
|
(function () {
|
||||||
|
const volumeStep = 0.05;
|
||||||
|
const enableVolumeScroll = true;
|
||||||
|
const enableHoverScroll = true;
|
||||||
|
const doc = document;
|
||||||
|
const pEl = document.getElementById('player');
|
||||||
|
|
||||||
|
var volumeHover = false;
|
||||||
|
var volumeSelector = pEl.querySelector('.vjs-volume-menu-button') || pEl.querySelector('.vjs-volume-panel');
|
||||||
|
if (volumeSelector != null) {
|
||||||
|
volumeSelector.onmouseover = function () { volumeHover = true; };
|
||||||
|
volumeSelector.onmouseout = function () { volumeHover = false; };
|
||||||
|
}
|
||||||
|
|
||||||
|
var mouseScroll = function mouseScroll(event) {
|
||||||
|
var activeEl = doc.activeElement;
|
||||||
|
if (enableHoverScroll) {
|
||||||
|
// If we leave this undefined then it can match non-existent elements below
|
||||||
|
activeEl = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// When controls are disabled, hotkeys will be disabled as well
|
||||||
|
if (player.controls()) {
|
||||||
|
if (volumeHover) {
|
||||||
|
if (enableVolumeScroll) {
|
||||||
|
event = window.event || event;
|
||||||
|
var delta = Math.max(-1, Math.min(1, (event.wheelDelta || -event.detail)));
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
if (delta == 1) {
|
||||||
|
increase_volume(volumeStep);
|
||||||
|
} else if (delta == -1) {
|
||||||
|
increase_volume(-volumeStep);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
player.on('mousewheel', mouseScroll);
|
||||||
|
player.on("DOMMouseScroll", mouseScroll);
|
||||||
|
}());
|
||||||
|
|
||||||
|
// Since videojs-share can sometimes be blocked, we defer it until last
|
||||||
|
player.share(shareOptions);
|
||||||
47
assets/js/playlist_widget.js
Normal file
47
assets/js/playlist_widget.js
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
function add_playlist_item(target) {
|
||||||
|
var tile = target.parentNode.parentNode.parentNode.parentNode.parentNode;
|
||||||
|
tile.style.display = 'none';
|
||||||
|
|
||||||
|
var url = '/playlist_ajax?action_add_video=1&redirect=false' +
|
||||||
|
'&video_id=' + target.getAttribute('data-id') +
|
||||||
|
'&playlist_id=' + target.getAttribute('data-plid');
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('POST', url, true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status != 200) {
|
||||||
|
tile.style.display = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send('csrf_token=' + playlist_data.csrf_token);
|
||||||
|
}
|
||||||
|
|
||||||
|
function remove_playlist_item(target) {
|
||||||
|
var tile = target.parentNode.parentNode.parentNode.parentNode.parentNode;
|
||||||
|
tile.style.display = 'none';
|
||||||
|
|
||||||
|
var url = '/playlist_ajax?action_remove_video=1&redirect=false' +
|
||||||
|
'&set_video_id=' + target.getAttribute('data-index') +
|
||||||
|
'&playlist_id=' + target.getAttribute('data-plid');
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('POST', url, true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status != 200) {
|
||||||
|
tile.style.display = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send('csrf_token=' + playlist_data.csrf_token);
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because one or more lines are too long
200
assets/js/sse.js
Normal file
200
assets/js/sse.js
Normal file
@@ -0,0 +1,200 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (C) 2016 Maxime Petazzoni <maxime.petazzoni@bulix.org>.
|
||||||
|
* All rights reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
var SSE = function (url, options) {
|
||||||
|
if (!(this instanceof SSE)) {
|
||||||
|
return new SSE(url, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.INITIALIZING = -1;
|
||||||
|
this.CONNECTING = 0;
|
||||||
|
this.OPEN = 1;
|
||||||
|
this.CLOSED = 2;
|
||||||
|
|
||||||
|
this.url = url;
|
||||||
|
|
||||||
|
options = options || {};
|
||||||
|
this.headers = options.headers || {};
|
||||||
|
this.payload = options.payload !== undefined ? options.payload : '';
|
||||||
|
this.method = options.method || (this.payload && 'POST' || 'GET');
|
||||||
|
|
||||||
|
this.FIELD_SEPARATOR = ':';
|
||||||
|
this.listeners = {};
|
||||||
|
|
||||||
|
this.xhr = null;
|
||||||
|
this.readyState = this.INITIALIZING;
|
||||||
|
this.progress = 0;
|
||||||
|
this.chunk = '';
|
||||||
|
|
||||||
|
this.addEventListener = function(type, listener) {
|
||||||
|
if (this.listeners[type] === undefined) {
|
||||||
|
this.listeners[type] = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.listeners[type].indexOf(listener) === -1) {
|
||||||
|
this.listeners[type].push(listener);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
this.removeEventListener = function(type, listener) {
|
||||||
|
if (this.listeners[type] === undefined) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var filtered = [];
|
||||||
|
this.listeners[type].forEach(function(element) {
|
||||||
|
if (element !== listener) {
|
||||||
|
filtered.push(element);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (filtered.length === 0) {
|
||||||
|
delete this.listeners[type];
|
||||||
|
} else {
|
||||||
|
this.listeners[type] = filtered;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
this.dispatchEvent = function(e) {
|
||||||
|
if (!e) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
e.source = this;
|
||||||
|
|
||||||
|
var onHandler = 'on' + e.type;
|
||||||
|
if (this.hasOwnProperty(onHandler)) {
|
||||||
|
this[onHandler].call(this, e);
|
||||||
|
if (e.defaultPrevented) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.listeners[e.type]) {
|
||||||
|
return this.listeners[e.type].every(function(callback) {
|
||||||
|
callback(e);
|
||||||
|
return !e.defaultPrevented;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
this._setReadyState = function (state) {
|
||||||
|
var event = new CustomEvent('readystatechange');
|
||||||
|
event.readyState = state;
|
||||||
|
this.readyState = state;
|
||||||
|
this.dispatchEvent(event);
|
||||||
|
};
|
||||||
|
|
||||||
|
this._onStreamFailure = function(e) {
|
||||||
|
this.dispatchEvent(new CustomEvent('error'));
|
||||||
|
this.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
this._onStreamProgress = function(e) {
|
||||||
|
if (this.xhr.status !== 200 && this.readyState !== this.CLOSED) {
|
||||||
|
this._onStreamFailure(e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.readyState == this.CONNECTING) {
|
||||||
|
this.dispatchEvent(new CustomEvent('open'));
|
||||||
|
this._setReadyState(this.OPEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
var data = this.xhr.responseText.substring(this.progress);
|
||||||
|
this.progress += data.length;
|
||||||
|
data.split(/(\r\n|\r|\n){2}/g).forEach(function(part) {
|
||||||
|
if (part.trim().length === 0) {
|
||||||
|
this.dispatchEvent(this._parseEventChunk(this.chunk.trim()));
|
||||||
|
this.chunk = '';
|
||||||
|
} else {
|
||||||
|
this.chunk += part;
|
||||||
|
}
|
||||||
|
}.bind(this));
|
||||||
|
};
|
||||||
|
|
||||||
|
this._onStreamLoaded = function(e) {
|
||||||
|
this._onStreamProgress(e);
|
||||||
|
|
||||||
|
// Parse the last chunk.
|
||||||
|
this.dispatchEvent(this._parseEventChunk(this.chunk));
|
||||||
|
this.chunk = '';
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parse a received SSE event chunk into a constructed event object.
|
||||||
|
*/
|
||||||
|
this._parseEventChunk = function(chunk) {
|
||||||
|
if (!chunk || chunk.length === 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var e = {'id': null, 'retry': null, 'data': '', 'event': 'message'};
|
||||||
|
chunk.split(/\n|\r\n|\r/).forEach(function(line) {
|
||||||
|
line = line.trimRight();
|
||||||
|
var index = line.indexOf(this.FIELD_SEPARATOR);
|
||||||
|
if (index <= 0) {
|
||||||
|
// Line was either empty, or started with a separator and is a comment.
|
||||||
|
// Either way, ignore.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var field = line.substring(0, index);
|
||||||
|
if (!(field in e)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var value = line.substring(index + 1).trimLeft();
|
||||||
|
if (field === 'data') {
|
||||||
|
e[field] += value;
|
||||||
|
} else {
|
||||||
|
e[field] = value;
|
||||||
|
}
|
||||||
|
}.bind(this));
|
||||||
|
|
||||||
|
var event = new CustomEvent(e.event);
|
||||||
|
event.data = e.data;
|
||||||
|
event.id = e.id;
|
||||||
|
return event;
|
||||||
|
};
|
||||||
|
|
||||||
|
this._checkStreamClosed = function() {
|
||||||
|
if (this.xhr.readyState === XMLHttpRequest.DONE) {
|
||||||
|
this._setReadyState(this.CLOSED);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
this.stream = function() {
|
||||||
|
this._setReadyState(this.CONNECTING);
|
||||||
|
|
||||||
|
this.xhr = new XMLHttpRequest();
|
||||||
|
this.xhr.addEventListener('progress', this._onStreamProgress.bind(this));
|
||||||
|
this.xhr.addEventListener('load', this._onStreamLoaded.bind(this));
|
||||||
|
this.xhr.addEventListener('readystatechange', this._checkStreamClosed.bind(this));
|
||||||
|
this.xhr.addEventListener('error', this._onStreamFailure.bind(this));
|
||||||
|
this.xhr.addEventListener('abort', this._onStreamFailure.bind(this));
|
||||||
|
this.xhr.open(this.method, this.url);
|
||||||
|
for (var header in this.headers) {
|
||||||
|
this.xhr.setRequestHeader(header, this.headers[header]);
|
||||||
|
}
|
||||||
|
this.xhr.send(this.payload);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.close = function() {
|
||||||
|
if (this.readyState === this.CLOSED) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.xhr.abort();
|
||||||
|
this.xhr = null;
|
||||||
|
this._setReadyState(this.CLOSED);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// Export our SSE module for npm.js
|
||||||
|
if (typeof exports !== 'undefined') {
|
||||||
|
exports.SSE = SSE;
|
||||||
|
}
|
||||||
88
assets/js/subscribe_widget.js
Normal file
88
assets/js/subscribe_widget.js
Normal file
@@ -0,0 +1,88 @@
|
|||||||
|
var subscribe_button = document.getElementById('subscribe');
|
||||||
|
subscribe_button.parentNode['action'] = 'javascript:void(0)';
|
||||||
|
|
||||||
|
if (subscribe_button.getAttribute('data-type') === 'subscribe') {
|
||||||
|
subscribe_button.onclick = subscribe;
|
||||||
|
} else {
|
||||||
|
subscribe_button.onclick = unsubscribe;
|
||||||
|
}
|
||||||
|
|
||||||
|
function subscribe(retries = 5) {
|
||||||
|
if (retries <= 0) {
|
||||||
|
console.log('Failed to subscribe.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var url = '/subscription_ajax?action_create_subscription_to_channel=1&redirect=false' +
|
||||||
|
'&c=' + subscribe_data.ucid;
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('POST', url, true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
|
||||||
|
|
||||||
|
var fallback = subscribe_button.innerHTML;
|
||||||
|
subscribe_button.onclick = unsubscribe;
|
||||||
|
subscribe_button.innerHTML = '<b>' + subscribe_data.unsubscribe_text + ' | ' + subscribe_data.sub_count_text + '</b>';
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status != 200) {
|
||||||
|
subscribe_button.onclick = subscribe;
|
||||||
|
subscribe_button.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
console.log('Subscribing failed... ' + retries + '/5');
|
||||||
|
setTimeout(function () { subscribe(retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Subscribing failed... ' + retries + '/5');
|
||||||
|
subscribe(retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send('csrf_token=' + subscribe_data.csrf_token);
|
||||||
|
}
|
||||||
|
|
||||||
|
function unsubscribe(retries = 5) {
|
||||||
|
if (retries <= 0) {
|
||||||
|
console.log('Failed to subscribe');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var url = '/subscription_ajax?action_remove_subscriptions=1&redirect=false' +
|
||||||
|
'&c=' + subscribe_data.ucid;
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('POST', url, true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
|
||||||
|
|
||||||
|
var fallback = subscribe_button.innerHTML;
|
||||||
|
subscribe_button.onclick = subscribe;
|
||||||
|
subscribe_button.innerHTML = '<b>' + subscribe_data.subscribe_text + ' | ' + subscribe_data.sub_count_text + '</b>';
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status != 200) {
|
||||||
|
subscribe_button.onclick = unsubscribe;
|
||||||
|
subscribe_button.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
console.log('Unsubscribing failed... ' + retries + '/5');
|
||||||
|
setTimeout(function () { unsubscribe(retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Unsubscribing failed... ' + retries + '/5');
|
||||||
|
unsubscribe(retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send('csrf_token=' + subscribe_data.csrf_token);
|
||||||
|
}
|
||||||
58
assets/js/themes.js
Normal file
58
assets/js/themes.js
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
var toggle_theme = document.getElementById('toggle_theme');
|
||||||
|
toggle_theme.href = 'javascript:void(0);';
|
||||||
|
|
||||||
|
toggle_theme.addEventListener('click', function () {
|
||||||
|
var dark_mode = document.getElementById('dark_theme').media === 'none';
|
||||||
|
|
||||||
|
var url = '/toggle_theme?redirect=false';
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', url, true);
|
||||||
|
|
||||||
|
set_mode(dark_mode);
|
||||||
|
window.localStorage.setItem('dark_mode', dark_mode ? 'dark' : 'light');
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener('storage', function (e) {
|
||||||
|
if (e.key === 'dark_mode') {
|
||||||
|
update_mode(e.newValue);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener('load', function () {
|
||||||
|
window.localStorage.setItem('dark_mode', document.getElementById('dark_mode_pref').textContent);
|
||||||
|
// Update localStorage if dark mode preference changed on preferences page
|
||||||
|
update_mode(window.localStorage.dark_mode);
|
||||||
|
});
|
||||||
|
|
||||||
|
function set_mode (bool) {
|
||||||
|
document.getElementById('dark_theme').media = !bool ? 'none' : '';
|
||||||
|
document.getElementById('light_theme').media = bool ? 'none' : '';
|
||||||
|
|
||||||
|
if (bool) {
|
||||||
|
toggle_theme.children[0].setAttribute('class', 'icon ion-ios-sunny');
|
||||||
|
} else {
|
||||||
|
toggle_theme.children[0].setAttribute('class', 'icon ion-ios-moon');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function update_mode (mode) {
|
||||||
|
if (mode === 'true' /* for backwards compatibility */ || mode === 'dark') {
|
||||||
|
// If preference for dark mode indicated
|
||||||
|
set_mode(true);
|
||||||
|
}
|
||||||
|
else if (mode === 'false' /* for backwards compaibility */ || mode === 'light') {
|
||||||
|
// If preference for light mode indicated
|
||||||
|
set_mode(false);
|
||||||
|
}
|
||||||
|
else if (document.getElementById('dark_mode_pref').textContent === '' && window.matchMedia('(prefers-color-scheme: dark)').matches) {
|
||||||
|
// If no preference indicated here and no preference indicated on the preferences page (backend), but the browser tells us that the operating system has a dark theme
|
||||||
|
set_mode(true);
|
||||||
|
}
|
||||||
|
// else do nothing, falling back to the mode defined by the `dark_mode` preference on the preferences page (backend)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
12
assets/js/video.min.js
vendored
12
assets/js/video.min.js
vendored
File diff suppressed because one or more lines are too long
@@ -1,2 +1,2 @@
|
|||||||
/*! @name videojs-contrib-quality-levels @version 2.0.7 @license Apache-2.0 */
|
/*! @name videojs-contrib-quality-levels @version 2.0.9 @license Apache-2.0 */
|
||||||
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t(require("video.js"),require("global/document")):"function"==typeof define&&define.amd?define(["video.js","global/document"],t):e.videojsContribQualityLevels=t(e.videojs,e.document)}(this,function(e,t){"use strict";e=e&&e.hasOwnProperty("default")?e.default:e,t=t&&t.hasOwnProperty("default")?t.default:t;var n=function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")},r=function(e,t){if(!e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return!t||"object"!=typeof t&&"function"!=typeof t?e:t},i=function(i){function o(){n(this,o);var l=r(this,i.call(this)),s=l;if(e.browser.IS_IE8)for(var u in s=t.createElement("custom"),o.prototype)"constructor"!==u&&(s[u]=o.prototype[u]);return s.levels_=[],s.selectedIndex_=-1,Object.defineProperty(s,"selectedIndex",{get:function(){return s.selectedIndex_}}),Object.defineProperty(s,"length",{get:function(){return s.levels_.length}}),r(l,s)}return function(e,t){if("function"!=typeof t&&null!==t)throw new TypeError("Super expression must either be null or a function, not "+typeof t);e.prototype=Object.create(t&&t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}}),t&&(Object.setPrototypeOf?Object.setPrototypeOf(e,t):e.__proto__=t)}(o,i),o.prototype.addQualityLevel=function(r){var i=this.getQualityLevelById(r.id);if(i)return i;var o=this.levels_.length;return i=new function r(i){n(this,r);var o=this;if(e.browser.IS_IE8)for(var l in o=t.createElement("custom"),r.prototype)"constructor"!==l&&(o[l]=r.prototype[l]);return o.id=i.id,o.label=o.id,o.width=i.width,o.height=i.height,o.bitrate=i.bandwidth,o.enabled_=i.enabled,Object.defineProperty(o,"enabled",{get:function(){return o.enabled_()},set:function(e){o.enabled_(e)}}),o}(r),""+o in this||Object.defineProperty(this,o,{get:function(){return this.levels_[o]}}),this.levels_.push(i),this.trigger({qualityLevel:i,type:"addqualitylevel"}),i},o.prototype.removeQualityLevel=function(e){for(var t=null,n=0,r=this.length;n<r;n++)if(this[n]===e){t=this.levels_.splice(n,1)[0],this.selectedIndex_===n?this.selectedIndex_=-1:this.selectedIndex_>n&&this.selectedIndex_--;break}return t&&this.trigger({qualityLevel:e,type:"removequalitylevel"}),t},o.prototype.getQualityLevelById=function(e){for(var t=0,n=this.length;t<n;t++){var r=this[t];if(r.id===e)return r}return null},o.prototype.dispose=function(){this.selectedIndex_=-1,this.levels_.length=0},o}(e.EventTarget);for(var o in i.prototype.allowedEvents_={change:"change",addqualitylevel:"addqualitylevel",removequalitylevel:"removequalitylevel"},i.prototype.allowedEvents_)i.prototype["on"+o]=null;var l=function(t){return n=this,e.mergeOptions({},t),r=n.qualityLevels,o=new i,n.on("dispose",function e(){o.dispose(),n.qualityLevels=r,n.off("dispose",e)}),n.qualityLevels=function(){return o},n.qualityLevels.VERSION="__VERSION__",o;var n,r,o};return(e.registerPlugin||e.plugin)("qualityLevels",l),l.VERSION="__VERSION__",l});
|
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t(require("video.js"),require("global/document")):"function"==typeof define&&define.amd?define(["video.js","global/document"],t):e.videojsContribQualityLevels=t(e.videojs,e.document)}(this,function(e,t){"use strict";function n(e){if(void 0===e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return e}e=e&&e.hasOwnProperty("default")?e.default:e,t=t&&t.hasOwnProperty("default")?t.default:t;var r=function(r){var i,l;function o(){var i,l=n(n(i=r.call(this)||this));if(e.browser.IS_IE8)for(var s in l=t.createElement("custom"),o.prototype)"constructor"!==s&&(l[s]=o.prototype[s]);return l.levels_=[],l.selectedIndex_=-1,Object.defineProperty(l,"selectedIndex",{get:function(){return l.selectedIndex_}}),Object.defineProperty(l,"length",{get:function(){return l.levels_.length}}),l||n(i)}l=r,(i=o).prototype=Object.create(l.prototype),i.prototype.constructor=i,i.__proto__=l;var s=o.prototype;return s.addQualityLevel=function(n){var r=this.getQualityLevelById(n.id);if(r)return r;var i=this.levels_.length;return r=new function n(r){var i=this;if(e.browser.IS_IE8)for(var l in i=t.createElement("custom"),n.prototype)"constructor"!==l&&(i[l]=n.prototype[l]);return i.id=r.id,i.label=i.id,i.width=r.width,i.height=r.height,i.bitrate=r.bandwidth,i.enabled_=r.enabled,Object.defineProperty(i,"enabled",{get:function(){return i.enabled_()},set:function(e){i.enabled_(e)}}),i}(n),""+i in this||Object.defineProperty(this,i,{get:function(){return this.levels_[i]}}),this.levels_.push(r),this.trigger({qualityLevel:r,type:"addqualitylevel"}),r},s.removeQualityLevel=function(e){for(var t=null,n=0,r=this.length;n<r;n++)if(this[n]===e){t=this.levels_.splice(n,1)[0],this.selectedIndex_===n?this.selectedIndex_=-1:this.selectedIndex_>n&&this.selectedIndex_--;break}return t&&this.trigger({qualityLevel:e,type:"removequalitylevel"}),t},s.getQualityLevelById=function(e){for(var t=0,n=this.length;t<n;t++){var r=this[t];if(r.id===e)return r}return null},s.dispose=function(){this.selectedIndex_=-1,this.levels_.length=0},o}(e.EventTarget);for(var i in r.prototype.allowedEvents_={change:"change",addqualitylevel:"addqualitylevel",removequalitylevel:"removequalitylevel"},r.prototype.allowedEvents_)r.prototype["on"+i]=null;var l=function(t){return n=this,e.mergeOptions({},t),i=n.qualityLevels,l=new r,n.on("dispose",function e(){l.dispose(),n.qualityLevels=i,n.off("dispose",e)}),n.qualityLevels=function(){return l},n.qualityLevels.VERSION="2.0.9",l;var n,i,l};return(e.registerPlugin||e.plugin)("qualityLevels",l),l.VERSION="2.0.9",l});
|
||||||
|
|||||||
3
assets/js/videojs-dash.min.js
vendored
3
assets/js/videojs-dash.min.js
vendored
File diff suppressed because one or more lines are too long
7
assets/js/videojs-http-source-selector.min.js
vendored
Normal file
7
assets/js/videojs-http-source-selector.min.js
vendored
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
/**
|
||||||
|
* videojs-http-source-selector
|
||||||
|
* @version 1.1.6
|
||||||
|
* @copyright 2019 Justin Fujita <Justin@pivotshare.com>
|
||||||
|
* @license MIT
|
||||||
|
*/
|
||||||
|
!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t(require("video.js")):"function"==typeof define&&define.amd?define(["video.js"],t):(e=e||self)["videojs-http-source-selector"]=t(e.videojs)}(this,function(r){"use strict";function o(e,t){e.prototype=Object.create(t.prototype),(e.prototype.constructor=e).__proto__=t}function s(e){if(void 0===e)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return e}var e=(r=r&&r.hasOwnProperty("default")?r.default:r).getComponent("MenuItem"),t=r.getComponent("Component"),a=function(n){function e(e,t){return t.selectable=!0,t.multiSelectable=!1,n.call(this,e,t)||this}o(e,n);var t=e.prototype;return t.handleClick=function(){var e=this.options_;console.log("Changing quality to:",e.label),n.prototype.handleClick.call(this);for(var t=this.player().qualityLevels(),o=0;o<t.length;o++)e.index==t.length?t[o].enabled=!0:e.index==o?t[o].enabled=!0:t[o].enabled=!1},t.update=function(){var e=this.player().qualityLevels().selectedIndex;this.selected(this.options_.index==e)},e}(e);t.registerComponent("SourceMenuItem",a);var u=r.getComponent("MenuButton"),n=function(i){function e(e,t){var o;o=i.call(this,e,t)||this,u.apply(s(o),arguments);var n=o.player().qualityLevels();if(t&&t.default)if("low"==t.default)for(var l=0;l<n.length;l++)n[l].enabled=0==l;else if(t.default="high")for(l=0;l<n.length;l++)n[l].enabled=l==n.length-1;return o.player().qualityLevels().on(["change","addqualitylevel"],r.bind(s(o),o.update)),o}o(e,i);var t=e.prototype;return t.createEl=function(){return r.dom.createEl("div",{className:"vjs-http-source-selector vjs-menu-button vjs-menu-button-popup vjs-control vjs-button"})},t.buildCSSClass=function(){return u.prototype.buildCSSClass.call(this)+" vjs-icon-cog"},t.update=function(){return u.prototype.update.call(this)},t.createItems=function(){for(var e=[],t=this.player().qualityLevels(),o=[],n=0;n<t.length;n++){var l=t.length-(n+1),i=l===t.selectedIndex,r=""+l,s=l;t[l].height?(r=t[l].height+"p",s=parseInt(t[l].height,10)):t[l].bitrate&&(r=Math.floor(t[l].bitrate/1e3)+" kbps",s=parseInt(t[l].bitrate,10)),0<=o.indexOf(r)||(o.push(r),e.push(new a(this.player_,{label:r,index:l,selected:i,sortVal:s})))}return 1<t.length&&e.push(new a(this.player_,{label:"Auto",index:t.length,selected:!1,sortVal:99999})),e.sort(function(e,t){return e.options_.sortVal<t.options_.sortVal?1:e.options_.sortVal>t.options_.sortVal?-1:0}),e},e}(u),l={},i=r.registerPlugin||r.plugin,c=function(e){var t=this;this.ready(function(){!function(n,e){if(n.addClass("vjs-http-source-selector"),console.log("videojs-http-source-selector initialized!"),console.log("player.techName_:"+n.techName_),"Html5"!=n.techName_)return;n.on(["loadedmetadata"],function(e){if(n.qualityLevels(),r.log("loadmetadata event"),"undefined"==n.videojs_http_source_selector_initialized||1==n.videojs_http_source_selector_initialized)console.log("player.videojs_http_source_selector_initialized == true");else{console.log("player.videojs_http_source_selector_initialized == false"),n.videojs_http_source_selector_initialized=!0;var t=n.controlBar,o=t.getChild("fullscreenToggle").el();t.el().insertBefore(t.addChild("SourceMenuButton").el(),o)}})}(t,r.mergeOptions(l,e))}),r.registerComponent("SourceMenuButton",n),r.registerComponent("SourceMenuItem",a)};return i("httpSourceSelector",c),c.VERSION="1.1.6",c});
|
||||||
14
assets/js/videojs-http-streaming.min.js
vendored
14
assets/js/videojs-http-streaming.min.js
vendored
File diff suppressed because one or more lines are too long
2
assets/js/videojs-overlay.min.js
vendored
Normal file
2
assets/js/videojs-overlay.min.js
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
/*! @name videojs-overlay @version 2.1.4 @license Apache-2.0 */
|
||||||
|
!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e(require("video.js"),require("global/window")):"function"==typeof define&&define.amd?define(["video.js","global/window"],e):t.videojsOverlay=e(t.videojs,t.window)}(this,function(t,e){"use strict";function n(t){if(void 0===t)throw new ReferenceError("this hasn't been initialised - super() hasn't been called");return t}t=t&&t.hasOwnProperty("default")?t.default:t,e=e&&e.hasOwnProperty("default")?e.default:e;var r={align:"top-left",class:"",content:"This overlay will show up while the video is playing",debug:!1,showBackground:!0,attachToControlBar:!1,overlays:[{start:"playing",end:"paused"}]},i=t.getComponent("Component"),o=t.dom||t,s=t.registerPlugin||t.plugin,a=function(t){return"number"==typeof t&&t==t},h=function(t){return"string"==typeof t&&/^\S+$/.test(t)},d=function(r){var i,s;function d(t,e){var i;return i=r.call(this,t,e)||this,["start","end"].forEach(function(t){var e=i.options_[t];if(a(e))i[t+"Event_"]="timeupdate";else if(h(e))i[t+"Event_"]=e;else if("start"===t)throw new Error('invalid "start" option; expected number or string')}),["endListener_","rewindListener_","startListener_"].forEach(function(t){i[t]=function(e){return d.prototype[t].call(n(n(i)),e)}}),"timeupdate"===i.startEvent_&&i.on(t,"timeupdate",i.rewindListener_),i.debug('created, listening to "'+i.startEvent_+'" for "start" and "'+(i.endEvent_||"nothing")+'" for "end"'),i.hide(),i}s=r,(i=d).prototype=Object.create(s.prototype),i.prototype.constructor=i,i.__proto__=s;var l=d.prototype;return l.createEl=function(){var t=this.options_,n=t.content,r=t.showBackground?"vjs-overlay-background":"vjs-overlay-no-background",i=o.createEl("div",{className:"\n vjs-overlay\n vjs-overlay-"+t.align+"\n "+t.class+"\n "+r+"\n vjs-hidden\n "});return"string"==typeof n?i.innerHTML=n:n instanceof e.DocumentFragment?i.appendChild(n):o.appendContent(i,n),i},l.debug=function(){if(this.options_.debug){for(var e=t.log,n=e,r=arguments.length,i=new Array(r),o=0;o<r;o++)i[o]=arguments[o];e.hasOwnProperty(i[0])&&"function"==typeof e[i[0]]&&(n=e[i.shift()]),n.apply(void 0,["overlay#"+this.id()+": "].concat(i))}},l.hide=function(){return r.prototype.hide.call(this),this.debug("hidden"),this.debug('bound `startListener_` to "'+this.startEvent_+'"'),this.endEvent_&&(this.debug('unbound `endListener_` from "'+this.endEvent_+'"'),this.off(this.player(),this.endEvent_,this.endListener_)),this.on(this.player(),this.startEvent_,this.startListener_),this},l.shouldHide_=function(t,e){var n=this.options_.end;return a(n)?t>=n:n===e},l.show=function(){return r.prototype.show.call(this),this.off(this.player(),this.startEvent_,this.startListener_),this.debug("shown"),this.debug('unbound `startListener_` from "'+this.startEvent_+'"'),this.endEvent_&&(this.debug('bound `endListener_` to "'+this.endEvent_+'"'),this.on(this.player(),this.endEvent_,this.endListener_)),this},l.shouldShow_=function(t,e){var n=this.options_.start,r=this.options_.end;return a(n)?a(r)?t>=n&&t<r:this.hasShownSinceSeek_?Math.floor(t)===n:(this.hasShownSinceSeek_=!0,t>=n):n===e},l.startListener_=function(t){var e=this.player().currentTime();this.shouldShow_(e,t.type)&&this.show()},l.endListener_=function(t){var e=this.player().currentTime();this.shouldHide_(e,t.type)&&this.hide()},l.rewindListener_=function(t){var e=this.player().currentTime(),n=this.previousTime_,r=this.options_.start,i=this.options_.end;e<n&&(this.debug("rewind detected"),a(i)&&!this.shouldShow_(e)?(this.debug("hiding; "+i+" is an integer and overlay should not show at this time"),this.hasShownSinceSeek_=!1,this.hide()):h(i)&&e<r&&(this.debug("hiding; show point ("+r+") is before now ("+e+") and end point ("+i+") is an event"),this.hasShownSinceSeek_=!1,this.hide())),this.previousTime_=e},d}(i);t.registerComponent("Overlay",d);var l=function(e){var n=this,i=t.mergeOptions(r,e);Array.isArray(this.overlays_)&&this.overlays_.forEach(function(t){n.removeChild(t),n.controlBar&&n.controlBar.removeChild(t),t.dispose()});var o=i.overlays;delete i.overlays,this.overlays_=o.map(function(e){var r=t.mergeOptions(i,e),o="string"==typeof r.attachToControlBar||!0===r.attachToControlBar;if(!n.controls()||!n.controlBar)return n.addChild("overlay",r);if(o&&-1!==r.align.indexOf("bottom")){var s=n.controlBar.children()[0];if(void 0!==n.controlBar.getChild(r.attachToControlBar)&&(s=n.controlBar.getChild(r.attachToControlBar)),s){var a=n.controlBar.addChild("overlay",r);return n.controlBar.el().insertBefore(a.el(),s.el()),a}}var h=n.addChild("overlay",r);return n.el().insertBefore(h.el(),n.controlBar.el()),h})};return l.VERSION="2.1.4",s("overlay",l),l});
|
||||||
6
assets/js/videojs-share.min.js
vendored
6
assets/js/videojs-share.min.js
vendored
File diff suppressed because one or more lines are too long
7
assets/js/videojs-vtt-thumbnails.min.js
vendored
Normal file
7
assets/js/videojs-vtt-thumbnails.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
1
assets/js/videojs-youtube-annotations.min.js
vendored
Normal file
1
assets/js/videojs-youtube-annotations.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
3
assets/js/videojs.hotkeys.min.js
vendored
3
assets/js/videojs.hotkeys.min.js
vendored
@@ -1,3 +0,0 @@
|
|||||||
/* videojs-hotkeys v0.2.25 - https://github.com/ctd1500/videojs-hotkeys */
|
|
||||||
!function(e,n){"undefined"!=typeof window&&window.videojs?n(window.videojs):"function"==typeof define&&define.amd?define("videojs-hotkeys",["video.js"],function(e){return n(e.default||e)}):"undefined"!=typeof module&&module.exports&&(module.exports=n(require("video.js")))}(0,function(e){"use strict";"undefined"!=typeof window&&(window.videojs_hotkeys={version:"0.2.25"});(e.registerPlugin||e.plugin)("hotkeys",function(n){function t(e){return"function"==typeof s?s(e):s}function r(e){null!=e&&"function"==typeof e.then&&e.then(null,function(e){})}var o=this,u=o.el(),l=document,i={volumeStep:.1,seekStep:5,enableMute:!0,enableVolumeScroll:!0,enableHoverScroll:!1,enableFullscreen:!0,enableNumbers:!0,enableJogStyle:!1,alwaysCaptureHotkeys:!1,enableModifiersForNumbers:!0,enableInactiveFocus:!0,skipInitialFocus:!1,playPauseKey:function(e){return 32===e.which||179===e.which},rewindKey:function(e){return 37===e.which||177===e.which},forwardKey:function(e){return 39===e.which||176===e.which},volumeUpKey:function(e){return 38===e.which},volumeDownKey:function(e){return 40===e.which},muteKey:function(e){return 77===e.which},fullscreenKey:function(e){return 70===e.which},customKeys:{}},c=e.mergeOptions||e.util.mergeOptions,a=(n=c(i,n||{})).volumeStep,s=n.seekStep,m=n.enableMute,f=n.enableVolumeScroll,y=n.enableHoverScroll,v=n.enableFullscreen,d=n.enableNumbers,p=n.enableJogStyle,b=n.alwaysCaptureHotkeys,h=n.enableModifiersForNumbers,w=n.enableInactiveFocus,k=n.skipInitialFocus,S=e.VERSION;u.hasAttribute("tabIndex")||u.setAttribute("tabIndex","-1"),u.style.outline="none",!b&&o.autoplay()||k||o.one("play",function(){u.focus()}),w&&o.on("userinactive",function(){var e=function(){clearTimeout(n)},n=setTimeout(function(){o.off("useractive",e);var n=l.activeElement,t=u.querySelector(".vjs-control-bar");n&&n.parentElement==t&&u.focus()},10);o.one("useractive",e)}),o.on("play",function(){var e=u.querySelector(".iframeblocker");e&&""===e.style.display&&(e.style.display="block",e.style.bottom="39px")});var K=!1,q=u.querySelector(".vjs-volume-menu-button")||u.querySelector(".vjs-volume-panel");null!=q&&(q.onmouseover=function(){K=!0},q.onmouseout=function(){K=!1});var j=function(e){if(y)n=0;else var n=l.activeElement;if(o.controls()&&(b||n==u||n==u.querySelector(".vjs-tech")||n==u.querySelector(".iframeblocker")||n==u.querySelector(".vjs-control-bar")||K)&&f){e=window.event||e;var t=Math.max(-1,Math.min(1,e.wheelDelta||-e.detail));e.preventDefault(),1==t?o.volume(o.volume()+a):-1==t&&o.volume(o.volume()-a)}},F=function(e,t){return n.playPauseKey(e,t)?1:n.rewindKey(e,t)?2:n.forwardKey(e,t)?3:n.volumeUpKey(e,t)?4:n.volumeDownKey(e,t)?5:n.muteKey(e,t)?6:n.fullscreenKey(e,t)?7:void 0};return o.on("keydown",function(e){var i,c,s=e.which,f=e.preventDefault,y=o.duration();if(o.controls()){var w=l.activeElement;if(b||w==u||w==u.querySelector(".vjs-tech")||w==u.querySelector(".vjs-control-bar")||w==u.querySelector(".iframeblocker"))switch(F(e,o)){case 1:f(),b&&e.stopPropagation(),o.paused()?r(o.play()):o.pause();break;case 2:i=!o.paused(),f(),i&&o.pause(),(c=o.currentTime()-t(e))<=0&&(c=0),o.currentTime(c),i&&r(o.play());break;case 3:i=!o.paused(),f(),i&&o.pause(),(c=o.currentTime()+t(e))>=y&&(c=i?y-.001:y),o.currentTime(c),i&&r(o.play());break;case 5:f(),p?(c=o.currentTime()-1,o.currentTime()<=1&&(c=0),o.currentTime(c)):o.volume(o.volume()-a);break;case 4:f(),p?((c=o.currentTime()+1)>=y&&(c=y),o.currentTime(c)):o.volume(o.volume()+a);break;case 6:m&&o.muted(!o.muted());break;case 7:v&&(o.isFullscreen()?o.exitFullscreen():o.requestFullscreen());break;default:if((s>47&&s<59||s>95&&s<106)&&(h||!(e.metaKey||e.ctrlKey||e.altKey))&&d){var k=48;s>95&&(k=96);var S=s-k;f(),o.currentTime(o.duration()*S*.1)}for(var K in n.customKeys){var q=n.customKeys[K];q&&q.key&&q.handler&&q.key(e)&&(f(),q.handler(o,n,e))}}}}),o.on("dblclick",function(e){if(null!=S&&S<="7.1.0"&&o.controls()){var n=e.relatedTarget||e.toElement||l.activeElement;n!=u&&n!=u.querySelector(".vjs-tech")&&n!=u.querySelector(".iframeblocker")||v&&(o.isFullscreen()?o.exitFullscreen():o.requestFullscreen())}}),o.on("mousewheel",j),o.on("DOMMouseScroll",j),this})});
|
|
||||||
//# sourceMappingURL=videojs.hotkeys.min.js.map
|
|
||||||
@@ -1,52 +1,459 @@
|
|||||||
|
String.prototype.supplant = function (o) {
|
||||||
|
return this.replace(/{([^{}]*)}/g, function (a, b) {
|
||||||
|
var r = o[b];
|
||||||
|
return typeof r === 'string' || typeof r === 'number' ? r : a;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function toggle_parent(target) {
|
function toggle_parent(target) {
|
||||||
body = target.parentNode.parentNode.children[1];
|
body = target.parentNode.parentNode.children[1];
|
||||||
if (body.style.display === null || body.style.display === "") {
|
if (body.style.display === null || body.style.display === '') {
|
||||||
target.innerHTML = "[ + ]";
|
target.innerHTML = '[ + ]';
|
||||||
body.style.display = "none";
|
body.style.display = 'none';
|
||||||
} else {
|
} else {
|
||||||
target.innerHTML = "[ - ]";
|
target.innerHTML = '[ - ]';
|
||||||
body.style.display = "";
|
body.style.display = '';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function toggle_comments(target) {
|
function toggle_comments(event) {
|
||||||
body = target.parentNode.parentNode.parentNode.children[1];
|
var target = event.target;
|
||||||
if (body.style.display === null || body.style.display === "") {
|
body = target.parentNode.parentNode.parentNode.children[1];
|
||||||
target.innerHTML = "[ + ]";
|
if (body.style.display === null || body.style.display === '') {
|
||||||
body.style.display = "none";
|
target.innerHTML = '[ + ]';
|
||||||
} else {
|
body.style.display = 'none';
|
||||||
target.innerHTML = "[ - ]";
|
} else {
|
||||||
body.style.display = "";
|
target.innerHTML = '[ - ]';
|
||||||
}
|
body.style.display = '';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function swap_comments(source) {
|
function swap_comments(event) {
|
||||||
if (source == "youtube") {
|
var source = event.target.getAttribute('data-comments');
|
||||||
get_youtube_comments();
|
|
||||||
} else if (source == "reddit") {
|
if (source === 'youtube') {
|
||||||
get_reddit_comments();
|
get_youtube_comments();
|
||||||
}
|
} else if (source === 'reddit') {
|
||||||
|
get_reddit_comments();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
String.prototype.supplant = function(o) {
|
function hide_youtube_replies(event) {
|
||||||
return this.replace(/{([^{}]*)}/g, function(a, b) {
|
var target = event.target;
|
||||||
var r = o[b];
|
|
||||||
return typeof r === "string" || typeof r === "number" ? r : a;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
function show_youtube_replies(target, inner_text, sub_text) {
|
sub_text = target.getAttribute('data-inner-text');
|
||||||
body = target.parentNode.parentNode.children[1];
|
inner_text = target.getAttribute('data-sub-text');
|
||||||
body.style.display = "";
|
|
||||||
|
|
||||||
target.innerHTML = inner_text;
|
body = target.parentNode.parentNode.children[1];
|
||||||
target.setAttribute("onclick", "hide_youtube_replies(this, \'" + inner_text + "\', \'" + sub_text + "\')");
|
body.style.display = 'none';
|
||||||
|
|
||||||
|
target.innerHTML = sub_text;
|
||||||
|
target.onclick = show_youtube_replies;
|
||||||
|
target.setAttribute('data-inner-text', inner_text);
|
||||||
|
target.setAttribute('data-sub-text', sub_text);
|
||||||
}
|
}
|
||||||
|
|
||||||
function hide_youtube_replies(target, inner_text, sub_text) {
|
function show_youtube_replies(event) {
|
||||||
body = target.parentNode.parentNode.children[1];
|
var target = event.target;
|
||||||
body.style.display = "none";
|
|
||||||
|
|
||||||
target.innerHTML = sub_text;
|
sub_text = target.getAttribute('data-inner-text');
|
||||||
target.setAttribute("onclick", "show_youtube_replies(this, \'" + inner_text + "\', \'" + sub_text + "\')");
|
inner_text = target.getAttribute('data-sub-text');
|
||||||
|
|
||||||
|
body = target.parentNode.parentNode.children[1];
|
||||||
|
body.style.display = '';
|
||||||
|
|
||||||
|
target.innerHTML = sub_text;
|
||||||
|
target.onclick = hide_youtube_replies;
|
||||||
|
target.setAttribute('data-inner-text', inner_text);
|
||||||
|
target.setAttribute('data-sub-text', sub_text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var continue_button = document.getElementById('continue');
|
||||||
|
if (continue_button) {
|
||||||
|
continue_button.onclick = continue_autoplay;
|
||||||
|
}
|
||||||
|
|
||||||
|
function next_video() {
|
||||||
|
var url = new URL('https://example.com/watch?v=' + video_data.next_video);
|
||||||
|
|
||||||
|
if (video_data.params.autoplay || video_data.params.continue_autoplay) {
|
||||||
|
url.searchParams.set('autoplay', '1');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.listen !== video_data.preferences.listen) {
|
||||||
|
url.searchParams.set('listen', video_data.params.listen);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.speed !== video_data.preferences.speed) {
|
||||||
|
url.searchParams.set('speed', video_data.params.speed);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.local !== video_data.preferences.local) {
|
||||||
|
url.searchParams.set('local', video_data.params.local);
|
||||||
|
}
|
||||||
|
|
||||||
|
url.searchParams.set('continue', '1');
|
||||||
|
location.assign(url.pathname + url.search);
|
||||||
|
}
|
||||||
|
|
||||||
|
function continue_autoplay(event) {
|
||||||
|
if (event.target.checked) {
|
||||||
|
player.on('ended', function () {
|
||||||
|
next_video();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
player.off('ended');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function number_with_separator(val) {
|
||||||
|
while (/(\d+)(\d{3})/.test(val.toString())) {
|
||||||
|
val = val.toString().replace(/(\d+)(\d{3})/, '$1' + ',' + '$2');
|
||||||
|
}
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_playlist(plid, retries) {
|
||||||
|
if (retries == undefined) retries = 5;
|
||||||
|
playlist = document.getElementById('playlist');
|
||||||
|
|
||||||
|
if (retries <= 0) {
|
||||||
|
console.log('Failed to pull playlist');
|
||||||
|
playlist.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
playlist.innerHTML = ' \
|
||||||
|
<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3> \
|
||||||
|
<hr>'
|
||||||
|
|
||||||
|
if (plid.startsWith('RD')) {
|
||||||
|
var plid_url = '/api/v1/mixes/' + plid +
|
||||||
|
'?continuation=' + video_data.id +
|
||||||
|
'&format=html&hl=' + video_data.preferences.locale;
|
||||||
|
} else {
|
||||||
|
var plid_url = '/api/v1/playlists/' + plid +
|
||||||
|
'?index=' + video_data.index +
|
||||||
|
'&continuation=' + video_data.id +
|
||||||
|
'&format=html&hl=' + video_data.preferences.locale;
|
||||||
|
}
|
||||||
|
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', plid_url, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status == 200) {
|
||||||
|
playlist.innerHTML = xhr.response.playlistHtml;
|
||||||
|
|
||||||
|
if (xhr.response.nextVideo) {
|
||||||
|
player.on('ended', function () {
|
||||||
|
var url = new URL('https://example.com/watch?v=' + xhr.response.nextVideo);
|
||||||
|
|
||||||
|
if (video_data.params.autoplay || video_data.params.continue_autoplay) {
|
||||||
|
url.searchParams.set('autoplay', '1');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.listen !== video_data.preferences.listen) {
|
||||||
|
url.searchParams.set('listen', video_data.params.listen);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.speed !== video_data.preferences.speed) {
|
||||||
|
url.searchParams.set('speed', video_data.params.speed);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.local !== video_data.preferences.local) {
|
||||||
|
url.searchParams.set('local', video_data.params.local);
|
||||||
|
}
|
||||||
|
|
||||||
|
url.searchParams.set('list', plid);
|
||||||
|
if (!plid.startsWith('RD')) {
|
||||||
|
url.searchParams.set('index', xhr.response.index);
|
||||||
|
}
|
||||||
|
location.assign(url.pathname + url.search);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
playlist.innerHTML = '';
|
||||||
|
document.getElementById('continue').style.display = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
playlist = document.getElementById('playlist');
|
||||||
|
playlist.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3><hr>';
|
||||||
|
|
||||||
|
console.log('Pulling playlist timed out... ' + retries + '/5');
|
||||||
|
setTimeout(function () { get_playlist(plid, retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
playlist = document.getElementById('playlist');
|
||||||
|
playlist.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3><hr>';
|
||||||
|
|
||||||
|
console.log('Pulling playlist timed out... ' + retries + '/5');
|
||||||
|
get_playlist(plid, retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_reddit_comments(retries) {
|
||||||
|
if (retries == undefined) retries = 5;
|
||||||
|
comments = document.getElementById('comments');
|
||||||
|
|
||||||
|
if (retries <= 0) {
|
||||||
|
console.log('Failed to pull comments');
|
||||||
|
comments.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var fallback = comments.innerHTML;
|
||||||
|
comments.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3>';
|
||||||
|
|
||||||
|
var url = '/api/v1/comments/' + video_data.id +
|
||||||
|
'?source=reddit&format=html' +
|
||||||
|
'&hl=' + video_data.preferences.locale;
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', url, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status == 200) {
|
||||||
|
comments.innerHTML = ' \
|
||||||
|
<div> \
|
||||||
|
<h3> \
|
||||||
|
<a href="javascript:void(0)">[ - ]</a> \
|
||||||
|
{title} \
|
||||||
|
</h3> \
|
||||||
|
<p> \
|
||||||
|
<b> \
|
||||||
|
<a href="javascript:void(0)" data-comments="youtube"> \
|
||||||
|
{youtubeCommentsText} \
|
||||||
|
</a> \
|
||||||
|
</b> \
|
||||||
|
</p> \
|
||||||
|
<b> \
|
||||||
|
<a rel="noopener" target="_blank" href="https://reddit.com{permalink}">{redditPermalinkText}</a> \
|
||||||
|
</b> \
|
||||||
|
</div> \
|
||||||
|
<div>{contentHtml}</div> \
|
||||||
|
<hr>'.supplant({
|
||||||
|
title: xhr.response.title,
|
||||||
|
youtubeCommentsText: video_data.youtube_comments_text,
|
||||||
|
redditPermalinkText: video_data.reddit_permalink_text,
|
||||||
|
permalink: xhr.response.permalink,
|
||||||
|
contentHtml: xhr.response.contentHtml
|
||||||
|
});
|
||||||
|
|
||||||
|
comments.children[0].children[0].children[0].onclick = toggle_comments;
|
||||||
|
comments.children[0].children[1].children[0].onclick = swap_comments;
|
||||||
|
} else {
|
||||||
|
if (video_data.params.comments[1] === 'youtube') {
|
||||||
|
console.log('Pulling comments failed... ' + retries + '/5');
|
||||||
|
setTimeout(function () { get_youtube_comments(retries - 1) }, 1000);
|
||||||
|
} else {
|
||||||
|
comments.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
console.log('Pulling comments failed... ' + retries + '/5');
|
||||||
|
setInterval(function () { get_reddit_comments(retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Pulling comments failed... ' + retries + '/5');
|
||||||
|
get_reddit_comments(retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_youtube_comments(retries) {
|
||||||
|
if (retries == undefined) retries = 5;
|
||||||
|
comments = document.getElementById('comments');
|
||||||
|
|
||||||
|
if (retries <= 0) {
|
||||||
|
console.log('Failed to pull comments');
|
||||||
|
comments.innerHTML = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var fallback = comments.innerHTML;
|
||||||
|
comments.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3>';
|
||||||
|
|
||||||
|
var url = '/api/v1/comments/' + video_data.id +
|
||||||
|
'?format=html' +
|
||||||
|
'&hl=' + video_data.preferences.locale +
|
||||||
|
'&thin_mode=' + video_data.preferences.thin_mode;
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', url, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status == 200) {
|
||||||
|
comments.innerHTML = ' \
|
||||||
|
<div> \
|
||||||
|
<h3> \
|
||||||
|
<a href="javascript:void(0)">[ - ]</a> \
|
||||||
|
{commentsText} \
|
||||||
|
</h3> \
|
||||||
|
<b> \
|
||||||
|
<a href="javascript:void(0)" data-comments="reddit"> \
|
||||||
|
{redditComments} \
|
||||||
|
</a> \
|
||||||
|
</b> \
|
||||||
|
</div> \
|
||||||
|
<div>{contentHtml}</div> \
|
||||||
|
<hr>'.supplant({
|
||||||
|
contentHtml: xhr.response.contentHtml,
|
||||||
|
redditComments: video_data.reddit_comments_text,
|
||||||
|
commentsText: video_data.comments_text.supplant(
|
||||||
|
{ commentCount: number_with_separator(xhr.response.commentCount) }
|
||||||
|
)
|
||||||
|
});
|
||||||
|
|
||||||
|
comments.children[0].children[0].children[0].onclick = toggle_comments;
|
||||||
|
comments.children[0].children[1].children[0].onclick = swap_comments;
|
||||||
|
} else {
|
||||||
|
if (video_data.params.comments[1] === 'youtube') {
|
||||||
|
setTimeout(function () { get_youtube_comments(retries - 1) }, 1000);
|
||||||
|
} else {
|
||||||
|
comments.innerHTML = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.onerror = function () {
|
||||||
|
comments.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3>';
|
||||||
|
console.log('Pulling comments failed... ' + retries + '/5');
|
||||||
|
setInterval(function () { get_youtube_comments(retries - 1) }, 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
comments.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3>';
|
||||||
|
console.log('Pulling comments failed... ' + retries + '/5');
|
||||||
|
get_youtube_comments(retries - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
function get_youtube_replies(target, load_more) {
|
||||||
|
var continuation = target.getAttribute('data-continuation');
|
||||||
|
|
||||||
|
var body = target.parentNode.parentNode;
|
||||||
|
var fallback = body.innerHTML;
|
||||||
|
body.innerHTML =
|
||||||
|
'<h3 style="text-align:center"><div class="loading"><i class="icon ion-ios-refresh"></i></div></h3>';
|
||||||
|
|
||||||
|
var url = '/api/v1/comments/' + video_data.id +
|
||||||
|
'?format=html' +
|
||||||
|
'&hl=' + video_data.preferences.locale +
|
||||||
|
'&thin_mode=' + video_data.preferences.thin_mode +
|
||||||
|
'&continuation=' + continuation;
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('GET', url, true);
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status == 200) {
|
||||||
|
if (load_more) {
|
||||||
|
body = body.parentNode.parentNode;
|
||||||
|
body.removeChild(body.lastElementChild);
|
||||||
|
body.innerHTML += xhr.response.contentHtml;
|
||||||
|
} else {
|
||||||
|
body.removeChild(body.lastElementChild);
|
||||||
|
|
||||||
|
var p = document.createElement('p');
|
||||||
|
var a = document.createElement('a');
|
||||||
|
p.appendChild(a);
|
||||||
|
|
||||||
|
a.href = 'javascript:void(0)';
|
||||||
|
a.onclick = hide_youtube_replies;
|
||||||
|
a.setAttribute('data-sub-text', video_data.hide_replies_text);
|
||||||
|
a.setAttribute('data-inner-text', video_data.show_replies_text);
|
||||||
|
a.innerText = video_data.hide_replies_text;
|
||||||
|
|
||||||
|
var div = document.createElement('div');
|
||||||
|
div.innerHTML = xhr.response.contentHtml;
|
||||||
|
|
||||||
|
body.appendChild(p);
|
||||||
|
body.appendChild(div);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
body.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.ontimeout = function () {
|
||||||
|
console.log('Pulling comments failed.');
|
||||||
|
body.innerHTML = fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.play_next) {
|
||||||
|
player.on('ended', function () {
|
||||||
|
var url = new URL('https://example.com/watch?v=' + video_data.next_video);
|
||||||
|
|
||||||
|
if (video_data.params.autoplay || video_data.params.continue_autoplay) {
|
||||||
|
url.searchParams.set('autoplay', '1');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.listen !== video_data.preferences.listen) {
|
||||||
|
url.searchParams.set('listen', video_data.params.listen);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.speed !== video_data.preferences.speed) {
|
||||||
|
url.searchParams.set('speed', video_data.params.speed);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.local !== video_data.preferences.local) {
|
||||||
|
url.searchParams.set('local', video_data.params.local);
|
||||||
|
}
|
||||||
|
|
||||||
|
url.searchParams.set('continue', '1');
|
||||||
|
location.assign(url.pathname + url.search);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener('load', function (e) {
|
||||||
|
if (video_data.plid) {
|
||||||
|
get_playlist(video_data.plid);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (video_data.params.comments[0] === 'youtube') {
|
||||||
|
get_youtube_comments();
|
||||||
|
} else if (video_data.params.comments[0] === 'reddit') {
|
||||||
|
get_reddit_comments();
|
||||||
|
} else if (video_data.params.comments[1] === 'youtube') {
|
||||||
|
get_youtube_comments();
|
||||||
|
} else if (video_data.params.comments[1] === 'reddit') {
|
||||||
|
get_reddit_comments();
|
||||||
|
} else {
|
||||||
|
comments = document.getElementById('comments');
|
||||||
|
comments.innerHTML = '';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|||||||
48
assets/js/watched_widget.js
Normal file
48
assets/js/watched_widget.js
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
function mark_watched(target) {
|
||||||
|
var tile = target.parentNode.parentNode.parentNode.parentNode.parentNode;
|
||||||
|
tile.style.display = 'none';
|
||||||
|
|
||||||
|
var url = '/watch_ajax?action_mark_watched=1&redirect=false' +
|
||||||
|
'&id=' + target.getAttribute('data-id');
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('POST', url, true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status != 200) {
|
||||||
|
tile.style.display = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send('csrf_token=' + watched_data.csrf_token);
|
||||||
|
}
|
||||||
|
|
||||||
|
function mark_unwatched(target) {
|
||||||
|
var tile = target.parentNode.parentNode.parentNode.parentNode.parentNode;
|
||||||
|
tile.style.display = 'none';
|
||||||
|
var count = document.getElementById('count')
|
||||||
|
count.innerText = count.innerText - 1;
|
||||||
|
|
||||||
|
var url = '/watch_ajax?action_mark_unwatched=1&redirect=false' +
|
||||||
|
'&id=' + target.getAttribute('data-id');
|
||||||
|
var xhr = new XMLHttpRequest();
|
||||||
|
xhr.responseType = 'json';
|
||||||
|
xhr.timeout = 10000;
|
||||||
|
xhr.open('POST', url, true);
|
||||||
|
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
|
||||||
|
|
||||||
|
xhr.onreadystatechange = function () {
|
||||||
|
if (xhr.readyState == 4) {
|
||||||
|
if (xhr.status != 200) {
|
||||||
|
count.innerText = count.innerText - 1 + 2;
|
||||||
|
tile.style.display = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xhr.send('csrf_token=' + watched_data.csrf_token);
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channels ADD COLUMN subscribed bool;"
|
psql invidious kemal -c "ALTER TABLE channels ADD COLUMN subscribed bool;"
|
||||||
psql invidious -c "UPDATE channels SET subscribed = false;"
|
psql invidious kemal -c "UPDATE channels SET subscribed = false;"
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channel_videos DROP COLUMN live_now CASCADE"
|
psql invidious kemal -c "ALTER TABLE channel_videos DROP COLUMN live_now CASCADE"
|
||||||
psql invidious -c "ALTER TABLE channel_videos DROP COLUMN premiere_timestamp CASCADE"
|
psql invidious kemal -c "ALTER TABLE channel_videos DROP COLUMN premiere_timestamp CASCADE"
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channel_videos ADD COLUMN live_now bool"
|
psql invidious kemal -c "ALTER TABLE channel_videos ADD COLUMN live_now bool"
|
||||||
psql invidious -c "ALTER TABLE channel_videos ADD COLUMN premiere_timestamp timestamptz"
|
psql invidious kemal -c "ALTER TABLE channel_videos ADD COLUMN premiere_timestamp timestamptz"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channels ADD COLUMN deleted bool;"
|
psql invidious kemal -c "ALTER TABLE channels ADD COLUMN deleted bool;"
|
||||||
psql invidious -c "UPDATE channels SET deleted = false;"
|
psql invidious kemal -c "UPDATE channels SET deleted = false;"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious < config/sql/session_ids.sql
|
psql invidious kemal < config/sql/session_ids.sql
|
||||||
psql invidious -c "INSERT INTO session_ids (SELECT unnest(id), email, CURRENT_TIMESTAMP FROM users) ON CONFLICT (id) DO NOTHING"
|
psql invidious kemal -c "INSERT INTO session_ids (SELECT unnest(id), email, CURRENT_TIMESTAMP FROM users) ON CONFLICT (id) DO NOTHING"
|
||||||
psql invidious -c "ALTER TABLE users DROP COLUMN id"
|
psql invidious kemal -c "ALTER TABLE users DROP COLUMN id"
|
||||||
|
|||||||
3
config/migrate-scripts/migrate-db-3bcb98e.sh
Executable file
3
config/migrate-scripts/migrate-db-3bcb98e.sh
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
psql invidious kemal < config/sql/annotations.sql
|
||||||
3
config/migrate-scripts/migrate-db-52cb239.sh
Executable file
3
config/migrate-scripts/migrate-db-52cb239.sh
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
psql invidious kemal -c "ALTER TABLE channel_videos ADD COLUMN views bigint;"
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channel_videos ADD COLUMN live_now bool;"
|
psql invidious kemal -c "ALTER TABLE channel_videos ADD COLUMN live_now bool;"
|
||||||
psql invidious -c "UPDATE channel_videos SET live_now = false;"
|
psql invidious kemal -c "UPDATE channel_videos SET live_now = false;"
|
||||||
|
|||||||
3
config/migrate-scripts/migrate-db-701b5ea.sh
Executable file
3
config/migrate-scripts/migrate-db-701b5ea.sh
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
psql invidious kemal -c "ALTER TABLE users ADD COLUMN feed_needs_update boolean"
|
||||||
@@ -1,3 +1,3 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channel_videos ADD COLUMN premiere_timestamp timestamptz;"
|
psql invidious kemal -c "ALTER TABLE channel_videos ADD COLUMN premiere_timestamp timestamptz;"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
psql invidious -c "ALTER TABLE channels DROP COLUMN subscribed"
|
psql invidious kemal -c "ALTER TABLE channels DROP COLUMN subscribed"
|
||||||
psql invidious -c "ALTER TABLE channels ADD COLUMN subscribed timestamptz"
|
psql invidious kemal -c "ALTER TABLE channels ADD COLUMN subscribed timestamptz"
|
||||||
psql invidious -c "UPDATE channels SET subscribed = '2019-01-01 00:00:00+00'"
|
psql invidious kemal -c "UPDATE channels SET subscribed = '2019-01-01 00:00:00+00'"
|
||||||
|
|||||||
12
config/sql/annotations.sql
Normal file
12
config/sql/annotations.sql
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
-- Table: public.annotations
|
||||||
|
|
||||||
|
-- DROP TABLE public.annotations;
|
||||||
|
|
||||||
|
CREATE TABLE public.annotations
|
||||||
|
(
|
||||||
|
id text NOT NULL,
|
||||||
|
annotations xml,
|
||||||
|
CONSTRAINT annotations_id_key UNIQUE (id)
|
||||||
|
);
|
||||||
|
|
||||||
|
GRANT ALL ON TABLE public.annotations TO kemal;
|
||||||
@@ -13,20 +13,12 @@ CREATE TABLE public.channel_videos
|
|||||||
length_seconds integer,
|
length_seconds integer,
|
||||||
live_now boolean,
|
live_now boolean,
|
||||||
premiere_timestamp timestamp with time zone,
|
premiere_timestamp timestamp with time zone,
|
||||||
|
views bigint,
|
||||||
CONSTRAINT channel_videos_id_key UNIQUE (id)
|
CONSTRAINT channel_videos_id_key UNIQUE (id)
|
||||||
);
|
);
|
||||||
|
|
||||||
GRANT ALL ON TABLE public.channel_videos TO kemal;
|
GRANT ALL ON TABLE public.channel_videos TO kemal;
|
||||||
|
|
||||||
-- Index: public.channel_videos_published_idx
|
|
||||||
|
|
||||||
-- DROP INDEX public.channel_videos_published_idx;
|
|
||||||
|
|
||||||
CREATE INDEX channel_videos_published_idx
|
|
||||||
ON public.channel_videos
|
|
||||||
USING btree
|
|
||||||
(published);
|
|
||||||
|
|
||||||
-- Index: public.channel_videos_ucid_idx
|
-- Index: public.channel_videos_ucid_idx
|
||||||
|
|
||||||
-- DROP INDEX public.channel_videos_ucid_idx;
|
-- DROP INDEX public.channel_videos_ucid_idx;
|
||||||
|
|||||||
19
config/sql/playlist_videos.sql
Normal file
19
config/sql/playlist_videos.sql
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
-- Table: public.playlist_videos
|
||||||
|
|
||||||
|
-- DROP TABLE public.playlist_videos;
|
||||||
|
|
||||||
|
CREATE TABLE playlist_videos
|
||||||
|
(
|
||||||
|
title text,
|
||||||
|
id text,
|
||||||
|
author text,
|
||||||
|
ucid text,
|
||||||
|
length_seconds integer,
|
||||||
|
published timestamptz,
|
||||||
|
plid text references playlists(id),
|
||||||
|
index int8,
|
||||||
|
live_now boolean,
|
||||||
|
PRIMARY KEY (index,plid)
|
||||||
|
);
|
||||||
|
|
||||||
|
GRANT ALL ON TABLE public.playlist_videos TO kemal;
|
||||||
18
config/sql/playlists.sql
Normal file
18
config/sql/playlists.sql
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
-- Table: public.playlists
|
||||||
|
|
||||||
|
-- DROP TABLE public.playlists;
|
||||||
|
|
||||||
|
CREATE TABLE public.playlists
|
||||||
|
(
|
||||||
|
title text,
|
||||||
|
id text primary key,
|
||||||
|
author text,
|
||||||
|
description text,
|
||||||
|
video_count integer,
|
||||||
|
created timestamptz,
|
||||||
|
updated timestamptz,
|
||||||
|
privacy privacy,
|
||||||
|
index int8[]
|
||||||
|
);
|
||||||
|
|
||||||
|
GRANT ALL ON public.playlists TO kemal;
|
||||||
10
config/sql/privacy.sql
Normal file
10
config/sql/privacy.sql
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
-- Type: public.privacy
|
||||||
|
|
||||||
|
-- DROP TYPE public.privacy;
|
||||||
|
|
||||||
|
CREATE TYPE public.privacy AS ENUM
|
||||||
|
(
|
||||||
|
'Public',
|
||||||
|
'Unlisted',
|
||||||
|
'Private'
|
||||||
|
);
|
||||||
@@ -12,6 +12,7 @@ CREATE TABLE public.users
|
|||||||
password text,
|
password text,
|
||||||
token text,
|
token text,
|
||||||
watched text[],
|
watched text[],
|
||||||
|
feed_needs_update boolean,
|
||||||
CONSTRAINT users_email_key UNIQUE (email)
|
CONSTRAINT users_email_key UNIQUE (email)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@@ -7,13 +7,15 @@ services:
|
|||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
volumes:
|
volumes:
|
||||||
- postgresdata:/var/lib/postgresql/data
|
- postgresdata:/var/lib/postgresql/data
|
||||||
|
healthcheck:
|
||||||
|
test: ["CMD", "pg_isready", "-U", "postgres"]
|
||||||
invidious:
|
invidious:
|
||||||
build:
|
build:
|
||||||
context: .
|
context: .
|
||||||
dockerfile: docker/Dockerfile
|
dockerfile: docker/Dockerfile
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
ports:
|
ports:
|
||||||
- "3000:3000"
|
- "127.0.0.1:3000:3000"
|
||||||
depends_on:
|
depends_on:
|
||||||
- postgres
|
- postgres
|
||||||
|
|
||||||
|
|||||||
@@ -1,15 +1,34 @@
|
|||||||
FROM archlinux/base
|
FROM alpine:edge
|
||||||
|
RUN apk add --no-cache crystal shards libc-dev \
|
||||||
RUN pacman -Sy --noconfirm shards crystal imagemagick librsvg \
|
yaml-dev libxml2-dev sqlite-dev zlib-dev curl && \
|
||||||
which pkgconf gcc ttf-liberation
|
curl -Lo /etc/apk/keys/omarroth.rsa.pub https://github.com/omarroth/boringssl-alpine/releases/download/1.1.0-r0/omarroth.rsa.pub && \
|
||||||
# base-devel contains many other basic packages, that are normally assumed to already exist on a clean arch system
|
curl -Lo boringssl-dev.apk https://github.com/omarroth/boringssl-alpine/releases/download/1.1.0-r0/boringssl-dev-1.1.0-r0.apk && \
|
||||||
|
curl -Lo lsquic.apk https://github.com/omarroth/lsquic-alpine/releases/download/2.6.3-r0/lsquic-2.6.3-r0.apk && \
|
||||||
ADD . /invidious
|
apk update && \
|
||||||
|
apk add boringssl-dev.apk lsquic.apk && \
|
||||||
|
rm -rf /var/cache/apk/* boringssl-dev.apk lsquic.apk
|
||||||
WORKDIR /invidious
|
WORKDIR /invidious
|
||||||
|
COPY ./shard.yml ./shard.yml
|
||||||
|
RUN shards update && shards install
|
||||||
|
RUN cp /usr/lib/libcrypto.a ./lib/lsquic/src/lsquic/ext/libcrypto.a && \
|
||||||
|
cp /usr/lib/libssl.a ./lib/lsquic/src/lsquic/ext/libssl.a && \
|
||||||
|
cp /usr/lib/liblsquic.a ./lib/lsquic/src/lsquic/ext/liblsquic.a
|
||||||
|
COPY ./src/ ./src/
|
||||||
|
# TODO: .git folder is required for building – this is destructive.
|
||||||
|
# See definition of CURRENT_BRANCH, CURRENT_COMMIT and CURRENT_VERSION.
|
||||||
|
COPY ./.git/ ./.git/
|
||||||
|
RUN crystal build --release --warnings all --error-on-warnings \
|
||||||
|
# TODO: Remove next line, see https://github.com/crystal-lang/crystal/issues/7946
|
||||||
|
-Dmusl \
|
||||||
|
./src/invidious.cr
|
||||||
|
|
||||||
RUN sed -i 's/host: localhost/host: postgres/' config/config.yml && \
|
RUN apk add --no-cache librsvg ttf-opensans
|
||||||
shards update && shards install && \
|
RUN addgroup -g 1000 -S invidious && \
|
||||||
crystal build src/invidious.cr
|
adduser -u 1000 -S invidious -G invidious
|
||||||
|
COPY ./assets/ ./assets/
|
||||||
|
COPY ./config/config.yml ./config/config.yml
|
||||||
|
COPY ./config/sql/ ./config/sql/
|
||||||
|
COPY ./locales/ ./locales/
|
||||||
|
RUN sed -i 's/host: \(127.0.0.1\|localhost\)/host: postgres/' config/config.yml
|
||||||
|
USER invidious
|
||||||
CMD [ "/invidious/invidious" ]
|
CMD [ "/invidious/invidious" ]
|
||||||
|
|||||||
@@ -12,12 +12,16 @@ if [ ! -f /var/lib/postgresql/data/setupFinished ]; then
|
|||||||
>&2 echo "### importing table schemas"
|
>&2 echo "### importing table schemas"
|
||||||
su postgres -c 'createdb invidious'
|
su postgres -c 'createdb invidious'
|
||||||
su postgres -c 'psql -c "CREATE USER kemal WITH PASSWORD '"'kemal'"'"'
|
su postgres -c 'psql -c "CREATE USER kemal WITH PASSWORD '"'kemal'"'"'
|
||||||
su postgres -c 'psql invidious < config/sql/channels.sql'
|
su postgres -c 'psql invidious kemal < config/sql/channels.sql'
|
||||||
su postgres -c 'psql invidious < config/sql/videos.sql'
|
su postgres -c 'psql invidious kemal < config/sql/videos.sql'
|
||||||
su postgres -c 'psql invidious < config/sql/channel_videos.sql'
|
su postgres -c 'psql invidious kemal < config/sql/channel_videos.sql'
|
||||||
su postgres -c 'psql invidious < config/sql/users.sql'
|
su postgres -c 'psql invidious kemal < config/sql/users.sql'
|
||||||
su postgres -c 'psql invidious < config/sql/session_ids.sql'
|
su postgres -c 'psql invidious kemal < config/sql/session_ids.sql'
|
||||||
su postgres -c 'psql invidious < config/sql/nonces.sql'
|
su postgres -c 'psql invidious kemal < config/sql/nonces.sql'
|
||||||
|
su postgres -c 'psql invidious kemal < config/sql/annotations.sql'
|
||||||
|
su postgres -c 'psql invidious kemal < config/sql/playlists.sql'
|
||||||
|
su postgres -c 'psql invidious kemal < config/sql/playlist_videos.sql'
|
||||||
|
su postgres -c 'psql invidious kemal < config/sql/privacy.sql'
|
||||||
touch /var/lib/postgresql/data/setupFinished
|
touch /var/lib/postgresql/data/setupFinished
|
||||||
echo "### invidious database setup finished"
|
echo "### invidious database setup finished"
|
||||||
exit
|
exit
|
||||||
|
|||||||
629
locales/ar.json
629
locales/ar.json
@@ -1,297 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` المشتركين",
|
"`x` subscribers": "`x` المشتركين",
|
||||||
"`x` videos": "`x` الفيديوهات",
|
"`x` videos": "`x` الفيديوهات",
|
||||||
"LIVE": "مباشر",
|
"`x` playlists": "`x` قوائم التشغيل",
|
||||||
"Shared `x` ago": "تم رفع الفيديو منذ `x`",
|
"LIVE": "مباشر",
|
||||||
"Unsubscribe": "إلغاء الإشتراك",
|
"Shared `x` ago": "تم رفع الفيديو منذ `x`",
|
||||||
"Subscribe": "إشتراك",
|
"Unsubscribe": "إلغاء الإشتراك",
|
||||||
"Login to subscribe to `x`": "سجل الدخول للإشتراك فى `x`",
|
"Subscribe": "إشتراك",
|
||||||
"View channel on YouTube": "زيارة القناة على موقع يوتيوب",
|
"View channel on YouTube": "زيارة القناة على موقع يوتيوب",
|
||||||
"newest": "الأجدد",
|
"View playlist on YouTube": "عرض قائمة التشغيل على اليوتيوب",
|
||||||
"oldest": "الأقدم",
|
"newest": "الأجدد",
|
||||||
"popular": "الاكثر شعبية",
|
"oldest": "الأقدم",
|
||||||
"last": "اخر الفيديوهات المعدلة",
|
"popular": "الأكثر شعبية",
|
||||||
"Next page": "الصفحة الثانية",
|
"last": "اخر قوائم التشغيل المعدلة",
|
||||||
"Previous page": "الصفحة السابقة",
|
"Next page": "الصفحة الثانية",
|
||||||
"Clear watch history?": "مسح السجل ؟",
|
"Previous page": "الصفحة السابقة",
|
||||||
"Yes": "نعم",
|
"Clear watch history?": "مسح السجل ؟",
|
||||||
"No": "لا",
|
"New password": "الرقم السرى الجديد",
|
||||||
"Import and Export Data": "استخراج و إضافة البيانات",
|
"New passwords must match": "الأرقام السرية يجب ان تكون متطابقة",
|
||||||
"Import": "إضافة",
|
"Cannot change password for Google accounts": "لا يستطيع تغيير الرقم السرى لحساب جوجل",
|
||||||
"Import Invidious data": "إضافة بيانات Invidious",
|
"Authorize token?": "رمز الإذن ؟",
|
||||||
"Import YouTube subscriptions": "إضافةالإشتراكات من موقع يوتيوب",
|
"Authorize token for `x`?": "تصريح الرمز لـ `x` ؟",
|
||||||
"Import FreeTube subscriptions (.db)": "إضافةالمشتركين من FreeTube (.db)",
|
"Yes": "نعم",
|
||||||
"Import NewPipe subscriptions (.json)": "إضافة المشتركين من NewPipe (.json)",
|
"No": "لا",
|
||||||
"Import NewPipe data (.zip)": "إضافة بيانات NewPipe (.zip)",
|
"Import and Export Data": "استخراج و إضافة البيانات",
|
||||||
"Export": "استخراج",
|
"Import": "إضافة",
|
||||||
"Export subscriptions as OPML": "استخراج المشتركين كـ OPML",
|
"Import Invidious data": "إضافة بيانات Invidious",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "استخراج المشتركين كـ OPML (لـ NewPipe و FreeTube)",
|
"Import YouTube subscriptions": "إضافةالإشتراكات من موقع يوتيوب",
|
||||||
"Export data as JSON": "استخراج البيانات كـ JSON",
|
"Import FreeTube subscriptions (.db)": "إضافةالمشتركين من FreeTube (.db)",
|
||||||
"Delete account?": "حذف الحساب ؟",
|
"Import NewPipe subscriptions (.json)": "إضافة المشتركين من NewPipe (.json)",
|
||||||
"History": "السجل",
|
"Import NewPipe data (.zip)": "إضافة بيانات NewPipe (.zip)",
|
||||||
"An alternative front-end to YouTube": "البديل الكامل لموقع يوتيوب",
|
"Export": "استخراج",
|
||||||
"JavaScript license information": "معلومات ترخيص JavaScript",
|
"Export subscriptions as OPML": "استخراج المشتركين كـ OPML",
|
||||||
"source": "المصدر",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "استخراج المشتركين كـ OPML (لـ NewPipe و FreeTube)",
|
||||||
"Login": "تسجيل الدخول",
|
"Export data as JSON": "استخراج البيانات كـ JSON",
|
||||||
"Login/Register": "تسجيل الدخول\\إنشاء حساب",
|
"Delete account?": "حذف الحساب ؟",
|
||||||
"Login to Google": "تسجيل الدخول بإستخدام جوجل",
|
"History": "السجل",
|
||||||
"User ID:": "إسم المستخدم:",
|
"An alternative front-end to YouTube": "البديل الكامل لموقع يوتيوب",
|
||||||
"Password:": "الرقم السرى:",
|
"JavaScript license information": "معلومات ترخيص JavaScript",
|
||||||
"Time (h:mm:ss):": "(يجب ان يكتب مثل هذا التنسيق) الوقت (h(ساعات):mm(دقائق):ss(ثوانى)):",
|
"source": "المصدر",
|
||||||
"Text CAPTCHA": "CAPTCHA كلامية",
|
"Log in": "تسجيل الدخول",
|
||||||
"Image CAPTCHA": "CAPTCHA صورية",
|
"Log in/register": "تسجيل الدخول\\إنشاء حساب",
|
||||||
"Sign In": "تسجيل الدخول",
|
"Log in with Google": "تسجيل الدخول بإستخدام جوجل",
|
||||||
"Register": "انشاء الحساب",
|
"User ID": "إسم المستخدم",
|
||||||
"Email:": "الإيميل:",
|
"Password": "الرقم السرى",
|
||||||
"Google verification code:": "رمز تحقق جوجل:",
|
"Time (h:mm:ss):": "(يجب ان يكتب مثل هذا التنسيق) الوقت (h(ساعات):mm(دقائق):ss(ثوانى)):",
|
||||||
"Preferences": "التفضيلات",
|
"Text CAPTCHA": "CAPTCHA كلامية",
|
||||||
"Player preferences": "التفضيلات المشغل",
|
"Image CAPTCHA": "CAPTCHA صورية",
|
||||||
"Always loop: ": "كرر الفيديو دائما: ",
|
"Sign In": "تسجيل الدخول",
|
||||||
"Autoplay: ": "تشغيل تلقائى: ",
|
"Register": "انشاء الحساب",
|
||||||
"Autoplay next video: ": "شغل الفيديو التالى تلقائى: ",
|
"E-mail": "الإيميل",
|
||||||
"Listen by default: ": "تشغيل النسخة السمعية تلقائى: ",
|
"Google verification code": "رمز تحقق جوجل",
|
||||||
"Proxy videos? ": "عرض الفيديوهات عن طريق الوكيل(proxy) ؟",
|
"Preferences": "التفضيلات",
|
||||||
"Default speed: ": "السرعة الإفتراضية: ",
|
"Player preferences": "التفضيلات المشغل",
|
||||||
"Preferred video quality: ": "الجودة المفضلة للفيديوهات: ",
|
"Always loop: ": "كرر الفيديو دائما: ",
|
||||||
"Player volume: ": "صوت المشغل: ",
|
"Autoplay: ": "تشغيل تلقائى: ",
|
||||||
"Default comments: ": "إضهار التعليقات الإفتراضية لـ: ",
|
"Play next by default: ": "شغل الفيديو التالي تلقائيا: ",
|
||||||
"youtube": "يوتيوب",
|
"Autoplay next video: ": "شغل الفيديو التالي تلقائيا (في قوائم التشغيل) ",
|
||||||
"reddit": "Reddit",
|
"Listen by default: ": "تشغيل النسخة السمعية تلقائى: ",
|
||||||
"Default captions: ": "الترجمات الإفتراضية: ",
|
"Proxy videos: ": "عرض الفيديوهات عن طريق البروكسي؟ ",
|
||||||
"Fallback captions: ": "الترجمات المصاحبة: ",
|
"Default speed: ": "السرعة الإفتراضية: ",
|
||||||
"Show related videos? ": "عرض مقاطع الفيديو ذات الصلة؟",
|
"Preferred video quality: ": "الجودة المفضلة للفيديوهات: ",
|
||||||
"Visual preferences": "التفضيلات المرئية",
|
"Player volume: ": "صوت المشغل: ",
|
||||||
"Dark mode: ": "الوضع الليلى: ",
|
"Default comments: ": "إضهار التعليقات الإفتراضية لـ: ",
|
||||||
"Thin mode: ": "الوضع الخفيف: ",
|
"youtube": "يوتيوب",
|
||||||
"Subscription preferences": "تفضيلات الإشتراك",
|
"reddit": "Reddit",
|
||||||
"Redirect homepage to feed: ": "إعادة التوجية من الصفحة الرئيسية لصفحة المشتركين (لرؤية اخر فيديوهات المشتركين): ",
|
"Default captions: ": "الترجمات الإفتراضية: ",
|
||||||
"Number of videos shown in feed: ": "عدد الفيديوهات التى ستظهر فى صفحة المشتركين: ",
|
"Fallback captions: ": "الترجمات المصاحبة: ",
|
||||||
"Sort videos by: ": "ترتيب الفيديو بـ: ",
|
"Show related videos: ": "اعرض الفيديوهات ذات الصلة: ",
|
||||||
"published": "احدث فيديو",
|
"Show annotations by default: ": "اعرض الملاحظات في الفيديو تلقائيا: ",
|
||||||
"published - reverse": "احدث فيديو - عكسى",
|
"Visual preferences": "التفضيلات المرئية",
|
||||||
"alphabetically": "ترتيب ابجدى",
|
"Player style: ": "شكل مشغل الفيديوهات: ",
|
||||||
"alphabetically - reverse": "ابجدى - عكسى",
|
"Dark mode: ": "الوضع الليلى: ",
|
||||||
"channel name": "بإسم القناة",
|
"Theme: ": "المظهر: ",
|
||||||
"channel name - reverse": "بإسم القناة - عكسى",
|
"dark": "غامق (اسود)",
|
||||||
"Only show latest video from channel: ": "فقط إظهر اخر فيديو من القناة: ",
|
"light": "فاتح (ابيض)",
|
||||||
"Only show latest unwatched video from channel: ": "فقط اظهر اخر فيديو لم يتم رؤيتة من القناة: ",
|
"Thin mode: ": "الوضع الخفيف: ",
|
||||||
"Only show unwatched: ": "فقط اظهر الذى لم يتم رؤيتة: ",
|
"Subscription preferences": "تفضيلات الإشتراك",
|
||||||
"Only show notifications (if there are any): ": "إظهار الإشعارات فقط (إذا كان هناك أي): ",
|
"Show annotations by default for subscribed channels: ": "عرض الملاحظات في الفيديوهات تلقائيا في القنوات المشترك بها فقط: ",
|
||||||
"Data preferences": "إعدادات التفضيلات",
|
"Redirect homepage to feed: ": "إعادة التوجية من الصفحة الرئيسية لصفحة المشتركين (لرؤية اخر فيديوهات المشتركين): ",
|
||||||
"Clear watch history": "حذف سجل المشاهدة",
|
"Number of videos shown in feed: ": "عدد الفيديوهات التى ستظهر فى صفحة المشتركين: ",
|
||||||
"Import/Export data": "إضافة\\إستخراج البيانات",
|
"Sort videos by: ": "ترتيب الفيديو بـ: ",
|
||||||
"Manage subscriptions": "إدارة المشتركين",
|
"published": "احدث فيديو",
|
||||||
"Watch history": "سجل المشاهدة",
|
"published - reverse": "احدث فيديو - عكسى",
|
||||||
"Delete account": "حذف الحساب",
|
"alphabetically": "ترتيب ابجدى",
|
||||||
"Administrator preferences": "إعدادات المدير",
|
"alphabetically - reverse": "ابجدى - عكسى",
|
||||||
"Default homepage: ": "الصفحة الرئيسية الافتراضية ",
|
"channel name": "بإسم القناة",
|
||||||
"Feed menu: ": "قائمة التغذية",
|
"channel name - reverse": "بإسم القناة - عكسى",
|
||||||
"Top enabled? ": "تفعيل 'الأفضل' ؟ ",
|
"Only show latest video from channel: ": "فقط إظهر اخر فيديو من القناة: ",
|
||||||
"CAPTCHA enabled? ": "تفعيل الكابتشا ؟",
|
"Only show latest unwatched video from channel: ": "فقط اظهر اخر فيديو لم يتم رؤيتة من القناة: ",
|
||||||
"Login enabled? ": "تفعيل تسجيل الدخول ؟",
|
"Only show unwatched: ": "فقط اظهر الذى لم يتم رؤيتة: ",
|
||||||
"Registration enabled? ": "تفعيل التسجيل ؟",
|
"Only show notifications (if there are any): ": "إظهار الإشعارات فقط (إذا كان هناك أي): ",
|
||||||
"Report statistics? ": "إبلاغ الإحصائيات",
|
"Enable web notifications": "تفعيل إشعارات المتصفح",
|
||||||
"Save preferences": "حفظ التفضيلات",
|
"`x` uploaded a video": "`x` رفع فيديو",
|
||||||
"Subscription manager": "مدير الإشتراكات",
|
"`x` is live": "`x` فى بث مباشر",
|
||||||
"`x` subscriptions": "`x` مشتركين",
|
"Data preferences": "إعدادات التفضيلات",
|
||||||
"Import/Export": "إضافة\\إستخراج",
|
"Clear watch history": "حذف سجل المشاهدة",
|
||||||
"unsubscribe": "إلغاء الإشتراك",
|
"Import/export data": "إضافة\\إستخراج البيانات",
|
||||||
"Subscriptions": "الإشتراكات",
|
"Change password": "غير الرقم السرى",
|
||||||
"`x` unseen notifications": "`x` إشعارات لم تشاهدها بعد ",
|
"Manage subscriptions": "إدارة المشتركين",
|
||||||
"search": "بحث",
|
"Manage tokens": "إدارة الرموز",
|
||||||
"Sign out": "تسجيل الخروج",
|
"Watch history": "سجل المشاهدة",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "تم الإنشاء تحت AGPLv3 بواسطة عمر روث.",
|
"Delete account": "حذف الحساب",
|
||||||
"Source available here.": "الأكواد متوفرة هنا.",
|
"Administrator preferences": "إعدادات المدير",
|
||||||
"View JavaScript license information.": "مشاهدة معلومات حول تراخيص الجافاسكريبت.",
|
"Default homepage: ": "الصفحة الرئيسية الافتراضية ",
|
||||||
"View privacy policy.": "عرض سياسة الخصوصية",
|
"Feed menu: ": "قائمة التدفقات: ",
|
||||||
"Trending": "الشائع",
|
"Top enabled: ": "تفعيل 'الأفضل' ؟ ",
|
||||||
"Unlisted": "غير مصنف",
|
"CAPTCHA enabled: ": "تفعيل الكابتشا: ",
|
||||||
"Watch video on Youtube": "مشاهدة الفيديو على اليوتيوب",
|
"Login enabled: ": "تفعيل الولوج: ",
|
||||||
"Genre: ": "النوع: ",
|
"Registration enabled: ": "تفعيل التسجيل: ",
|
||||||
"License: ": "التراخيص: ",
|
"Report statistics: ": "الإبلاغ عن الإحصائيات: ",
|
||||||
"Family friendly? ": "محتوى عائلى? ",
|
"Save preferences": "حفظ التفضيلات",
|
||||||
"Wilson score: ": "درجة ويلسون: ",
|
"Subscription manager": "مدير الإشتراكات",
|
||||||
"Engagement: ": "نسبة المشاركة (عدد المشاهدات\\عدد الإعجابات): ",
|
"Token manager": "إداره الرمز",
|
||||||
"Whitelisted regions: ": "الدول المسموح فيها هذا الفيديو: ",
|
"Token": "الرمز",
|
||||||
"Blacklisted regions: ": "الدول الحظور فيها هذا الفيديو: ",
|
"`x` subscriptions": "`x` مشتركين",
|
||||||
"Shared `x`": "شارك منذ `x`",
|
"`x` tokens": "`x` رموز",
|
||||||
"Premieres in `x`": "يعرض فى 'x'",
|
"Import/export": "إضافة\\إستخراج",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "اهلا! يبدو ان الجافاسكريبت معطلة. اضغط هنا لعرض التعليقات, ضع فى إعتبارك انها ستأخذ وقت اطول للعرض.",
|
"unsubscribe": "إلغاء الإشتراك",
|
||||||
"View YouTube comments": "عرض تعليقات اليوتيوب",
|
"revoke": "مسح",
|
||||||
"View more comments on Reddit": "عرض المزيد من التعليقات على\\من موقع Reddit",
|
"Subscriptions": "الإشتراكات",
|
||||||
"View `x` comments": "عرض `x` تعليقات",
|
"`x` unseen notifications": "`x` إشعارات لم تشاهدها بعد",
|
||||||
"View Reddit comments": "عرض تعليقات ريدإت Reddit",
|
"search": "بحث",
|
||||||
"Hide replies": "إخفاء الردود",
|
"Log out": "تسجيل الخروج",
|
||||||
"Show replies": "عرض الردود",
|
"Released under the AGPLv3 by Omar Roth.": "تم الإنشاء تحت AGPLv3 بواسطة عمر روث.",
|
||||||
"Incorrect password": "الرقم السرى غير صحيح",
|
"Source available here.": "الأكواد متوفرة هنا.",
|
||||||
"Quota exceeded, try again in a few hours": "تم تجاوز عدد المرات المسموح بها, حاول مرة اخرى بعد عدة ساعات",
|
"View JavaScript license information.": "مشاهدة معلومات حول تراخيص الجافاسكريبت.",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "غير قادر على تسجيل الدخول, تأكد من تشغيل المصادقة الثنائية 2FA.",
|
"View privacy policy.": "عرض سياسة الخصوصية.",
|
||||||
"Invalid TFA code": "كود مصادقة ثنائية 2FA غير صحيح",
|
"Trending": "الشائع",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "لم يتم تسجيل الدخول. هذا ربما بسبب ان المصادقة الثنائية 2FA معطلة فى حسابك.",
|
"Public": "عام",
|
||||||
"Invalid answer": "إجابة خاطئة",
|
"Unlisted": "غير مصنف",
|
||||||
"Invalid CAPTCHA": "الكابتشا CAPTCHA غير صاحلة",
|
"Private": "خاص",
|
||||||
"CAPTCHA is a required field": "مكان الكابتشا CAPTCHA مطلوب",
|
"View all playlists": "عرض جميع قوائم التشغيل",
|
||||||
"User ID is a required field": "مكان إسم المستخدم مطلوب",
|
"Updated `x` ago": "تم تحديثه منذ `x`",
|
||||||
"Password is a required field": "مكان الرقم السرى مطلوب",
|
"Delete playlist `x`?": "حذف قائمه التشغيل `x` ?",
|
||||||
"Invalid username or password": "إسم المستخدم او الرقم السرى غير صحيح",
|
"Delete playlist": "حذف قائمه التغشيل",
|
||||||
"Please sign in using 'Sign in with Google'": "الرجاء تسجيل الدخول 'تسجيل الدخول بواسطة جوجل'",
|
"Create playlist": "إنشاء قائمه تشغيل",
|
||||||
"Password cannot be empty": "الرقم السرى لايمكن ان يكون فارغ",
|
"Title": "العنوان",
|
||||||
"Password cannot be longer than 55 characters": "الرقم السرى لا يتعدى 55 حرف",
|
"Playlist privacy": "إعدادات الخصوصيه",
|
||||||
"Please sign in": "الرجاء تسجيل الدخول",
|
"Editing playlist `x`": "تعديل قائمه التشفيل `x`",
|
||||||
"Invidious Private Feed for `x`": "صفحة Invidious للمشتركين الخاصة\\مخفية لـ `x`",
|
"Watch on YouTube": "مشاهدة الفيديو على اليوتيوب",
|
||||||
"channel:`x`": "قناة:`x`",
|
"Hide annotations": "إخفاء الملاحظات فى الفيديو",
|
||||||
"Deleted or invalid channel": "قناة ممسوحة او غير صالحة",
|
"Show annotations": "عرض الملاحظات فى الفيديو",
|
||||||
"This channel does not exist.": "القناة غير موجودة.",
|
"Genre: ": "النوع: ",
|
||||||
"Could not get channel info.": "لم يستطع الحصول على معلومات القناة.",
|
"License: ": "التراخيص: ",
|
||||||
"Could not fetch comments": "لم يتمكن من إحضار التعليقات",
|
"Family friendly? ": "محتوى عائلى? ",
|
||||||
"View `x` replies": "عرض `x` ردود",
|
"Wilson score: ": "درجة ويلسون: ",
|
||||||
"`x` ago": "`x` منذ",
|
"Engagement: ": "نسبة المشاركة (عدد المشاهدات\\عدد الإعجابات): ",
|
||||||
"Load more": "عرض المزيد",
|
"Whitelisted regions: ": "الدول المسموح فيها هذا الفيديو: ",
|
||||||
"`x` points": "`x` نقاط",
|
"Blacklisted regions: ": "الدول الحظور فيها هذا الفيديو: ",
|
||||||
"Could not create mix.": "لم يستطع عمل خلط.",
|
"Shared `x`": "شارك منذ `x`",
|
||||||
"Playlist is empty": "قائمة التشغيل فارغة",
|
"`x` views": "`x` مشاهدات",
|
||||||
"Invalid playlist.": "قائمة التشغيل غير صالحة.",
|
"Premieres in `x`": "يعرض فى `x`",
|
||||||
"Playlist does not exist.": "قائمة التشغيل غير موجودة.",
|
"Premieres `x`": "يعرض `x`",
|
||||||
"Could not pull trending pages.": "لم يستطع عرض الصفحات الراجئة.",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "اهلا! يبدو ان الجافاسكريبت معطلة. اضغط هنا لعرض التعليقات, ضع فى إعتبارك انها ستأخذ وقت اطول للعرض.",
|
||||||
"Hidden field \"challenge\" is a required field": "مكان مخفى \"تحدى\" مكان مطلوب",
|
"View YouTube comments": "عرض تعليقات اليوتيوب",
|
||||||
"Hidden field \"token\" is a required field": "مكان مخفى \"رمز\" مكان مطلوب",
|
"View more comments on Reddit": "عرض المزيد من التعليقات على\\من موقع Reddit",
|
||||||
"Invalid challenge": "تحدى غير صالح",
|
"View `x` comments": "عرض `x` تعليقات",
|
||||||
"Invalid token": "روز غير صالح",
|
"View Reddit comments": "عرض تعليقات ريدإت Reddit",
|
||||||
"Invalid user": "مستخدم غير صالح",
|
"Hide replies": "إخفاء الردود",
|
||||||
"Token is expired, please try again": "الرمز منتهى الصلاحية , الرجاء المحاولة مرة اخرى",
|
"Show replies": "عرض الردود",
|
||||||
"English": "إنجليزى",
|
"Incorrect password": "الرقم السرى غير صحيح",
|
||||||
"English (auto-generated)": "إنجليزى (تم إنشائة تلقائى)",
|
"Quota exceeded, try again in a few hours": "تم تجاوز عدد المرات المسموح بها, حاول مرة اخرى بعد عدة ساعات",
|
||||||
"Afrikaans": "الأفريكانية",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "غير قادر على تسجيل الدخول, تأكد من تشغيل المصادقة الثنائية 2FA.",
|
||||||
"Albanian": "الألبانية",
|
"Invalid TFA code": "كود مصادقة ثنائية 2FA غير صحيح",
|
||||||
"Amharic": "الأمهرية",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "لم يتم تسجيل الدخول. هذا ربما بسبب ان المصادقة الثنائية 2FA معطلة فى حسابك.",
|
||||||
"Arabic": "العربية",
|
"Wrong answer": "إجابة خاطئة",
|
||||||
"Armenian": "الأرميني",
|
"Erroneous CAPTCHA": "الكابتشا CAPTCHA غير صاحلة",
|
||||||
"Azerbaijani": "أذربيجان",
|
"CAPTCHA is a required field": "مكان الكابتشا CAPTCHA مطلوب",
|
||||||
"Bangla": "البنغالية",
|
"User ID is a required field": "مكان إسم المستخدم مطلوب",
|
||||||
"Basque": "الباسكي",
|
"Password is a required field": "مكان الرقم السرى مطلوب",
|
||||||
"Belarusian": "البيلاروسية",
|
"Wrong username or password": "إسم المستخدم او الرقم السرى غير صحيح",
|
||||||
"Bosnian": "البوسنية",
|
"Please sign in using 'Log in with Google'": "الرجاء تسجيل الدخول 'تسجيل الدخول بواسطة جوجل'",
|
||||||
"Bulgarian": "البلغارية",
|
"Password cannot be empty": "الرقم السرى لايمكن ان يكون فارغ",
|
||||||
"Burmese": "البورمية",
|
"Password cannot be longer than 55 characters": "الرقم السرى لا يتعدى 55 حرف",
|
||||||
"Catalan": "الكاتالونية",
|
"Please log in": "الرجاء تسجيل الدخول",
|
||||||
"Cebuano": "السيبيونو",
|
"Invidious Private Feed for `x`": "صفحة Invidious للمشتركين الخاصة\\مخفية لـ `x`",
|
||||||
"Chinese (Simplified)": "الصينية (المبسطة)",
|
"channel:`x`": "قناة:`x`",
|
||||||
"Chinese (Traditional)": "الصينية (التقليدية)",
|
"Deleted or invalid channel": "قناة ممسوحة او غير صالحة",
|
||||||
"Corsican": "الكورسيكية",
|
"This channel does not exist.": "القناة غير موجودة.",
|
||||||
"Croatian": "الكرواتية",
|
"Could not get channel info.": "لم يستطع الحصول على معلومات القناة.",
|
||||||
"Czech": "تشيكي",
|
"Could not fetch comments": "لم يتمكن من إحضار التعليقات",
|
||||||
"Danish": "دانماركي",
|
"View `x` replies": "عرض `x` ردود",
|
||||||
"Dutch": "هولندي",
|
"`x` ago": "`x` منذ",
|
||||||
"Esperanto": "الاسبرانتو",
|
"Load more": "عرض المزيد",
|
||||||
"Estonian": "الإستونية",
|
"`x` points": "`x` نقاط",
|
||||||
"Filipino": "الفلبينية",
|
"Could not create mix.": "لم يستطع عمل خلط.",
|
||||||
"Finnish": "الفنلندية",
|
"Empty playlist": "قائمة التشغيل فارغة",
|
||||||
"French": "الفرنسية",
|
"Not a playlist.": "قائمة التشغيل غير صالحة.",
|
||||||
"Galician": "الجاليكية",
|
"Playlist does not exist.": "قائمة التشغيل غير موجودة.",
|
||||||
"Georgian": "الجورجية",
|
"Could not pull trending pages.": "لم يستطع عرض الصفحات الراجئة.",
|
||||||
"German": "ألمانية",
|
"Hidden field \"challenge\" is a required field": "مكان مخفى \"تحدى\" مكان مطلوب",
|
||||||
"Greek": "الإغريقي",
|
"Hidden field \"token\" is a required field": "مكان مخفى \"رمز\" مكان مطلوب",
|
||||||
"Gujarati": "الغوجاراتية",
|
"Erroneous challenge": "تحدى غير صالح",
|
||||||
"Haitian Creole": "الكاثوليكية الهايتية",
|
"Erroneous token": "روز غير صالح",
|
||||||
"Hausa": "الهوسا",
|
"No such user": "مستخدم غير صالح",
|
||||||
"Hawaiian": "هاواي",
|
"Token is expired, please try again": "الرمز منتهى الصلاحية , الرجاء المحاولة مرة اخرى",
|
||||||
"Hebrew": "العبرية",
|
"English": "إنجليزى",
|
||||||
"Hindi": "الهندية",
|
"English (auto-generated)": "إنجليزى (تم إنشائة تلقائى)",
|
||||||
"Hmong": "همونغ",
|
"Afrikaans": "الأفريكانية",
|
||||||
"Hungarian": "الهنغارية",
|
"Albanian": "الألبانية",
|
||||||
"Icelandic": "أيسلندي",
|
"Amharic": "الأمهرية",
|
||||||
"Igbo": "الإيبو",
|
"Arabic": "العربية",
|
||||||
"Indonesian": "الأندونيسية",
|
"Armenian": "الأرميني",
|
||||||
"Irish": "الأيرلندية",
|
"Azerbaijani": "أذربيجان",
|
||||||
"Italian": "الإيطالي",
|
"Bangla": "البنغالية",
|
||||||
"Japanese": "اليابانية",
|
"Basque": "الباسكي",
|
||||||
"Javanese": "جاوي",
|
"Belarusian": "البيلاروسية",
|
||||||
"Kannada": "الكانادا",
|
"Bosnian": "البوسنية",
|
||||||
"Kazakh": "الكازاخية",
|
"Bulgarian": "البلغارية",
|
||||||
"Khmer": "الخمير",
|
"Burmese": "البورمية",
|
||||||
"Korean": "الكورية",
|
"Catalan": "الكاتالونية",
|
||||||
"Kurdish": "كردي",
|
"Cebuano": "السيبيونو",
|
||||||
"Kyrgyz": "قيرغيزستان",
|
"Chinese (Simplified)": "الصينية (المبسطة)",
|
||||||
"Lao": "لاو",
|
"Chinese (Traditional)": "الصينية (التقليدية)",
|
||||||
"Latin": "لاتينية",
|
"Corsican": "الكورسيكية",
|
||||||
"Latvian": "اللاتفية",
|
"Croatian": "الكرواتية",
|
||||||
"Lithuanian": "اللتوانية",
|
"Czech": "تشيكي",
|
||||||
"Luxembourgish": "اللوكسمبرجية",
|
"Danish": "دانماركي",
|
||||||
"Macedonian": "المقدونية",
|
"Dutch": "هولندي",
|
||||||
"Malagasy": "مدجشقر\\مدغشقر",
|
"Esperanto": "الاسبرانتو",
|
||||||
"Malay": "الملايو",
|
"Estonian": "الإستونية",
|
||||||
"Malayalam": "المالايالامية",
|
"Filipino": "الفلبينية",
|
||||||
"Maltese": "المالطية",
|
"Finnish": "الفنلندية",
|
||||||
"Maori": "الماوري",
|
"French": "الفرنسية",
|
||||||
"Marathi": "المهاراتية",
|
"Galician": "الجاليكية",
|
||||||
"Mongolian": "المنغولية",
|
"Georgian": "الجورجية",
|
||||||
"Nepali": "النيبالية",
|
"German": "ألمانية",
|
||||||
"Norwegian": "النرويجية",
|
"Greek": "الإغريقي",
|
||||||
"Nyanja": "نيانجا",
|
"Gujarati": "الغوجاراتية",
|
||||||
"Pashto": "الباشتو",
|
"Haitian Creole": "الكاثوليكية الهايتية",
|
||||||
"Persian": "الفارسية",
|
"Hausa": "الهوسا",
|
||||||
"Polish": "البولندي",
|
"Hawaiian": "هاواي",
|
||||||
"Portuguese": "البرتغالية",
|
"Hebrew": "العبرية",
|
||||||
"Punjabi": "البنجابية",
|
"Hindi": "الهندية",
|
||||||
"Romanian": "روماني",
|
"Hmong": "همونغ",
|
||||||
"Russian": "الروسية",
|
"Hungarian": "الهنغارية",
|
||||||
"Samoan": "ساموا",
|
"Icelandic": "أيسلندي",
|
||||||
"Scottish Gaelic": "الغيلية الاسكتلندية",
|
"Igbo": "الإيبو",
|
||||||
"Serbian": "صربي",
|
"Indonesian": "الأندونيسية",
|
||||||
"Shona": "شونا",
|
"Irish": "الأيرلندية",
|
||||||
"Sindhi": "السندية",
|
"Italian": "الإيطالي",
|
||||||
"Sinhala": "السنهالية",
|
"Japanese": "اليابانية",
|
||||||
"Slovak": "السلوفاكية",
|
"Javanese": "جاوي",
|
||||||
"Slovenian": "سلوفيني",
|
"Kannada": "الكانادا",
|
||||||
"Somali": "الصومالية",
|
"Kazakh": "الكازاخية",
|
||||||
"Southern Sotho": "جنوب سوثو",
|
"Khmer": "الخمير",
|
||||||
"Spanish": "الأسبانية",
|
"Korean": "الكورية",
|
||||||
"Spanish (Latin America)": "الأسبانية (أمريكا اللاتينية)",
|
"Kurdish": "كردي",
|
||||||
"Sundanese": "السودانية",
|
"Kyrgyz": "قيرغيزستان",
|
||||||
"Swahili": "السواحلية",
|
"Lao": "لاو",
|
||||||
"Swedish": "السويدية",
|
"Latin": "لاتينية",
|
||||||
"Tajik": "الطاجيكية",
|
"Latvian": "اللاتفية",
|
||||||
"Tamil": "التاميل",
|
"Lithuanian": "اللتوانية",
|
||||||
"Telugu": "التيلجو",
|
"Luxembourgish": "اللوكسمبرجية",
|
||||||
"Thai": "التايلاندية",
|
"Macedonian": "المقدونية",
|
||||||
"Turkish": "التركية",
|
"Malagasy": "مدجشقر\\مدغشقر",
|
||||||
"Ukrainian": "الأوكراني",
|
"Malay": "الملايو",
|
||||||
"Urdu": "الأردية",
|
"Malayalam": "المالايالامية",
|
||||||
"Uzbek": "الأوزبكي",
|
"Maltese": "المالطية",
|
||||||
"Vietnamese": "الفيتنامية",
|
"Maori": "الماوري",
|
||||||
"Welsh": "الولزية",
|
"Marathi": "المهاراتية",
|
||||||
"Western Frisian": "الفريزية الغربية",
|
"Mongolian": "المنغولية",
|
||||||
"Xhosa": "زوسا",
|
"Nepali": "النيبالية",
|
||||||
"Yiddish": "اليديشية",
|
"Norwegian Bokmål": "النرويجية",
|
||||||
"Yoruba": "اليوروبا",
|
"Nyanja": "نيانجا",
|
||||||
"Zulu": "الزولو",
|
"Pashto": "الباشتو",
|
||||||
"`x` years": "`x` سنوات",
|
"Persian": "الفارسية",
|
||||||
"`x` months": "`x` شهور",
|
"Polish": "البولندي",
|
||||||
"`x` weeks": "`x` اسابيع",
|
"Portuguese": "البرتغالية",
|
||||||
"`x` days": "`x` ايام",
|
"Punjabi": "البنجابية",
|
||||||
"`x` hours": "`x` ساعات",
|
"Romanian": "روماني",
|
||||||
"`x` minutes": "`x` دقائق",
|
"Russian": "الروسية",
|
||||||
"`x` seconds": "`x` ثوانى",
|
"Samoan": "ساموا",
|
||||||
"Fallback comments: ": "التعليقات المصاحبة",
|
"Scottish Gaelic": "الغيلية الاسكتلندية",
|
||||||
"Popular": "لاكثر شعبية",
|
"Serbian": "صربي",
|
||||||
"Top": "الأفضل",
|
"Shona": "شونا",
|
||||||
"About": "حول",
|
"Sindhi": "السندية",
|
||||||
"Rating: ": "التقييم",
|
"Sinhala": "السنهالية",
|
||||||
"Language: ": "اللغة",
|
"Slovak": "السلوفاكية",
|
||||||
"Default": "الكل",
|
"Slovenian": "سلوفيني",
|
||||||
"Music": "الاغانى",
|
"Somali": "الصومالية",
|
||||||
"Gaming": "الألعاب",
|
"Southern Sotho": "جنوب سوثو",
|
||||||
"News": "الأخبار",
|
"Spanish": "الأسبانية",
|
||||||
"Movies": "الأفلام",
|
"Spanish (Latin America)": "الأسبانية (أمريكا اللاتينية)",
|
||||||
"Download as: ": "تحميل كـ",
|
"Sundanese": "السودانية",
|
||||||
"Download": "تحميل",
|
"Swahili": "السواحلية",
|
||||||
"%A %B %-d, %Y": "",
|
"Swedish": "السويدية",
|
||||||
"(edited)": "(تم تعديلة)",
|
"Tajik": "الطاجيكية",
|
||||||
"Youtube permalink of the comment": "رابط التعليق على اليوتيوب",
|
"Tamil": "التاميل",
|
||||||
"`x` marked it with a ❤": "'x' اعجب بهذا",
|
"Telugu": "التيلجو",
|
||||||
"Audio mode": "الوضع الصوتى",
|
"Thai": "التايلاندية",
|
||||||
"Video mode": "وضع الفيديو",
|
"Turkish": "التركية",
|
||||||
"Videos": "الفيديوهات",
|
"Ukrainian": "الأوكراني",
|
||||||
"Playlists": "قوائم التشغيل",
|
"Urdu": "الأردية",
|
||||||
"Current version: ": "الإصدار الحالى"
|
"Uzbek": "الأوزبكي",
|
||||||
|
"Vietnamese": "الفيتنامية",
|
||||||
|
"Welsh": "الولزية",
|
||||||
|
"Western Frisian": "الفريزية الغربية",
|
||||||
|
"Xhosa": "زوسا",
|
||||||
|
"Yiddish": "اليديشية",
|
||||||
|
"Yoruba": "اليوروبا",
|
||||||
|
"Zulu": "الزولو",
|
||||||
|
"`x` years": "`x` سنوات",
|
||||||
|
"`x` months": "`x` شهور",
|
||||||
|
"`x` weeks": "`x` اسابيع",
|
||||||
|
"`x` days": "`x` ايام",
|
||||||
|
"`x` hours": "`x` ساعات",
|
||||||
|
"`x` minutes": "`x` دقائق",
|
||||||
|
"`x` seconds": "`x` ثوانى",
|
||||||
|
"Fallback comments: ": "التعليقات البديلة: ",
|
||||||
|
"Popular": "الأكثر شعبية",
|
||||||
|
"Top": "الأفضل",
|
||||||
|
"About": "حول",
|
||||||
|
"Rating: ": "التقييم: ",
|
||||||
|
"Language: ": "اللغة: ",
|
||||||
|
"View as playlist": "عرض كا قائمة التشغيل",
|
||||||
|
"Default": "الكل",
|
||||||
|
"Music": "الاغانى",
|
||||||
|
"Gaming": "الألعاب",
|
||||||
|
"News": "الأخبار",
|
||||||
|
"Movies": "الأفلام",
|
||||||
|
"Download": "نزّل",
|
||||||
|
"Download as: ": "نزّله كـ: ",
|
||||||
|
"%A %B %-d, %Y": "%A %-d %B %Y",
|
||||||
|
"(edited)": "(تم تعديلة)",
|
||||||
|
"YouTube comment permalink": "رابط التعليق على اليوتيوب",
|
||||||
|
"permalink": "الرابط",
|
||||||
|
"`x` marked it with a ❤": "`x` اعجب بهذا",
|
||||||
|
"Audio mode": "الوضع الصوتى",
|
||||||
|
"Video mode": "وضع الفيديو",
|
||||||
|
"Videos": "الفيديوهات",
|
||||||
|
"Playlists": "قوائم التشغيل",
|
||||||
|
"Community": "المجتمع",
|
||||||
|
"Current version: ": "الإصدار الحالي: "
|
||||||
}
|
}
|
||||||
|
|||||||
631
locales/de.json
631
locales/de.json
@@ -1,297 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` Abonnenten",
|
"`x` subscribers": "`x` Abonnenten",
|
||||||
"`x` videos": "`x` Videos",
|
"`x` videos": "`x` Videos",
|
||||||
"LIVE": "LIVE",
|
"`x` playlists": "",
|
||||||
"Shared `x` ago": "Vor `x` geteilt",
|
"LIVE": "LIVE",
|
||||||
"Unsubscribe": "Abbestellen",
|
"Shared `x` ago": "Vor `x` geteilt",
|
||||||
"Subscribe": "Abonnieren",
|
"Unsubscribe": "Abbestellen",
|
||||||
"Login to subscribe to `x`": "Einloggen um `x` zu abonnieren",
|
"Subscribe": "Abonnieren",
|
||||||
"View channel on YouTube": "Kanal auf YouTube anzeigen",
|
"View channel on YouTube": "Kanal auf YouTube anzeigen",
|
||||||
"newest": "neueste",
|
"View playlist on YouTube": "Wiedergabeliste auf YouTube anzeigen",
|
||||||
"oldest": "älteste",
|
"newest": "neueste",
|
||||||
"popular": "beliebt",
|
"oldest": "älteste",
|
||||||
"last": "",
|
"popular": "beliebt",
|
||||||
"Next page": "Nächste Seite",
|
"last": "letzte",
|
||||||
"Previous page": "Vorherige Seite",
|
"Next page": "Nächste Seite",
|
||||||
"Clear watch history?": "Verlauf löschen?",
|
"Previous page": "Vorherige Seite",
|
||||||
"Yes": "Ja",
|
"Clear watch history?": "Verlauf löschen?",
|
||||||
"No": "Nein",
|
"New password": "Neues Passwort",
|
||||||
"Import and Export Data": "Import und Export Daten",
|
"New passwords must match": "Neue Passwörter müssen gleich sein",
|
||||||
"Import": "Importieren",
|
"Cannot change password for Google accounts": "Ich kann das Passwort deines Google Kontos nicht ändern",
|
||||||
"Import Invidious data": "Invidious Daten importieren",
|
"Authorize token?": "Token autorisieren?",
|
||||||
"Import YouTube subscriptions": "YouTube Abonnements importieren",
|
"Authorize token for `x`?": "Token für `x` autorisieren?",
|
||||||
"Import FreeTube subscriptions (.db)": "FreeTube Abonnements importieren (.db)",
|
"Yes": "Ja",
|
||||||
"Import NewPipe subscriptions (.json)": "NewPipe Abonnements importieren (.json)",
|
"No": "Nein",
|
||||||
"Import NewPipe data (.zip)": "NewPipe Daten importieren (.zip)",
|
"Import and Export Data": "Daten importieren und exportieren",
|
||||||
"Export": "Exportieren",
|
"Import": "Importieren",
|
||||||
"Export subscriptions as OPML": "Abonnements als OPML exportieren",
|
"Import Invidious data": "Invidious Daten importieren",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonnements als OPML exportieren (für NewPipe & FreeTube)",
|
"Import YouTube subscriptions": "YouTube Abonnements importieren",
|
||||||
"Export data as JSON": "Daten als JSON exportieren",
|
"Import FreeTube subscriptions (.db)": "FreeTube Abonnements importieren (.db)",
|
||||||
"Delete account?": "Account löschen?",
|
"Import NewPipe subscriptions (.json)": "NewPipe Abonnements importieren (.json)",
|
||||||
"History": "Verlauf",
|
"Import NewPipe data (.zip)": "NewPipe Daten importieren (.zip)",
|
||||||
"An alternative front-end to YouTube": "Eine alternative Oberfläche für YouTube",
|
"Export": "Exportieren",
|
||||||
"JavaScript license information": "JavaScript Lizenzinformationen",
|
"Export subscriptions as OPML": "Abonnements als OPML exportieren",
|
||||||
"source": "Quelle",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonnements als OPML exportieren (für NewPipe & FreeTube)",
|
||||||
"Login": "Einloggen",
|
"Export data as JSON": "Daten als JSON exportieren",
|
||||||
"Login/Register": "Einloggen/Registrieren",
|
"Delete account?": "Account löschen?",
|
||||||
"Login to Google": "In Google einloggen",
|
"History": "Verlauf",
|
||||||
"User ID:": "Benutzer ID:",
|
"An alternative front-end to YouTube": "Eine alternative Oberfläche für YouTube",
|
||||||
"Password:": "Passwort:",
|
"JavaScript license information": "JavaScript Lizenzinformationen",
|
||||||
"Time (h:mm:ss):": "Zeit (h:mm:ss):",
|
"source": "Quelle",
|
||||||
"Text CAPTCHA": "Text CAPTCHA",
|
"Log in": "Einloggen",
|
||||||
"Image CAPTCHA": "Image CAPTCHA",
|
"Log in/register": "Einloggen/Registrieren",
|
||||||
"Sign In": "Einloggen",
|
"Log in with Google": "Mit Google einloggen",
|
||||||
"Register": "Registrieren",
|
"User ID": "Benutzer ID",
|
||||||
"Email:": "Email:",
|
"Password": "Passwort",
|
||||||
"Google verification code:": "Google Bestätigungscode:",
|
"Time (h:mm:ss):": "Zeit (h:mm:ss):",
|
||||||
"Preferences": "Einstellungen",
|
"Text CAPTCHA": "Text CAPTCHA",
|
||||||
"Player preferences": "Playereinstellungen",
|
"Image CAPTCHA": "Bild CAPTCHA",
|
||||||
"Always loop: ": "Immer wiederholen: ",
|
"Sign In": "Anmelden",
|
||||||
"Autoplay: ": "Automatisch abspielen: ",
|
"Register": "Registrieren",
|
||||||
"Autoplay next video: ": "nächstes Video automatisch abspielen: ",
|
"E-mail": "E-Mail",
|
||||||
"Listen by default: ": "Nur Ton als Standard: ",
|
"Google verification code": "Google-Bestätigungscode",
|
||||||
"Proxy videos? ": "",
|
"Preferences": "Einstellungen",
|
||||||
"Default speed: ": "Standardgeschwindigkeit: ",
|
"Player preferences": "Wiedergabeeinstellungen",
|
||||||
"Preferred video quality: ": "Bevorzugte Videoqualität: ",
|
"Always loop: ": "Immer wiederholen: ",
|
||||||
"Player volume: ": "Playerlautstärke: ",
|
"Autoplay: ": "Automatisch abspielen: ",
|
||||||
"Default comments: ": "Standardkommentare: ",
|
"Play next by default: ": "Immer automatisch nächstes Video spielen: ",
|
||||||
"youtube": "youtube",
|
"Autoplay next video: ": "nächstes Video automatisch abspielen: ",
|
||||||
"reddit": "reddit",
|
"Listen by default: ": "Nur Ton als Standard: ",
|
||||||
"Default captions: ": "Standarduntertitel: ",
|
"Proxy videos: ": "Proxy-Videos: ",
|
||||||
"Fallback captions: ": "Ersatzuntertitel: ",
|
"Default speed: ": "Standardgeschwindigkeit: ",
|
||||||
"Show related videos? ": "Ähnliche Videos anzeigen? ",
|
"Preferred video quality: ": "Bevorzugte Videoqualität: ",
|
||||||
"Visual preferences": "Anzeigeeinstellungen",
|
"Player volume: ": "Wiedergabelautstärke: ",
|
||||||
"Dark mode: ": "Nachtmodus: ",
|
"Default comments: ": "Standardkommentare: ",
|
||||||
"Thin mode: ": "Schlanker Modus: ",
|
"youtube": "youtube",
|
||||||
"Subscription preferences": "Abonnementeinstellungen",
|
"reddit": "reddit",
|
||||||
"Redirect homepage to feed: ": "Startseite zu Feed umleiten: ",
|
"Default captions: ": "Standarduntertitel: ",
|
||||||
"Number of videos shown in feed: ": "Anzahl von Videos die im Feed angezeigt werden: ",
|
"Fallback captions: ": "Ersatzuntertitel: ",
|
||||||
"Sort videos by: ": "Videos sortieren nach: ",
|
"Show related videos: ": "Ähnliche Videos anzeigen? ",
|
||||||
"published": "veröffentlicht",
|
"Show annotations by default: ": "Standardmäßig Anmerkungen anzeigen? ",
|
||||||
"published - reverse": "veröffentlicht - invertiert",
|
"Visual preferences": "Anzeigeeinstellungen",
|
||||||
"alphabetically": "alphabetisch",
|
"Player style: ": "Abspielgeräterstil: ",
|
||||||
"alphabetically - reverse": "alphabetisch - invertiert",
|
"Dark mode: ": "Nachtmodus: ",
|
||||||
"channel name": "Kanalname",
|
"Theme: ": "Modus: ",
|
||||||
"channel name - reverse": "Kanalname - invertiert",
|
"dark": "Nachtmodus",
|
||||||
"Only show latest video from channel: ": "Nur neueste Videos des Kanals anzeigen: ",
|
"light": "klarer Modus",
|
||||||
"Only show latest unwatched video from channel: ": "Nur neueste ungesehene Videos des Kanals anzeigen: ",
|
"Thin mode: ": "Schlanker Modus: ",
|
||||||
"Only show unwatched: ": "Nur ungesehene anzeigen: ",
|
"Subscription preferences": "Abonnementeinstellungen",
|
||||||
"Only show notifications (if there are any): ": "Nur Benachrichtigungen anzeigen (wenn es welche gibt): ",
|
"Show annotations by default for subscribed channels: ": "Anmerkungen für abonnierte Kanäle standardmäßig anzeigen? ",
|
||||||
"Data preferences": "Dateneinstellungen",
|
"Redirect homepage to feed: ": "Startseite zu Feed umleiten: ",
|
||||||
"Clear watch history": "Verlauf löschen",
|
"Number of videos shown in feed: ": "Anzahl von Videos die im Feed angezeigt werden: ",
|
||||||
"Import/Export data": "Daten im- exportieren",
|
"Sort videos by: ": "Videos sortieren nach: ",
|
||||||
"Manage subscriptions": "Abonnements verwalten",
|
"published": "veröffentlicht",
|
||||||
"Watch history": "Verlauf",
|
"published - reverse": "veröffentlicht - invertiert",
|
||||||
"Delete account": "Account löschen",
|
"alphabetically": "alphabetisch",
|
||||||
"Administrator preferences": "",
|
"alphabetically - reverse": "alphabetisch - invertiert",
|
||||||
"Default homepage: ": "",
|
"channel name": "Kanalname",
|
||||||
"Feed menu: ": "",
|
"channel name - reverse": "Kanalname - invertiert",
|
||||||
"Top enabled? ": "",
|
"Only show latest video from channel: ": "Nur neueste Videos des Kanals anzeigen: ",
|
||||||
"CAPTCHA enabled? ": "",
|
"Only show latest unwatched video from channel: ": "Nur neueste ungesehene Videos des Kanals anzeigen: ",
|
||||||
"Login enabled? ": "",
|
"Only show unwatched: ": "Nur ungesehene anzeigen: ",
|
||||||
"Registration enabled? ": "",
|
"Only show notifications (if there are any): ": "Nur Benachrichtigungen anzeigen (wenn es welche gibt): ",
|
||||||
"Report statistics? ": "",
|
"Enable web notifications": "Webbenachrichtigungen aktivieren",
|
||||||
"Save preferences": "Einstellungen speichern",
|
"`x` uploaded a video": "`x` hat ein Video hochgeladen",
|
||||||
"Subscription manager": "Abonnementverwaltung",
|
"`x` is live": "`x` ist live",
|
||||||
"`x` subscriptions": "`x` Abonnements",
|
"Data preferences": "Dateneinstellungen",
|
||||||
"Import/Export": "Importieren/Exportieren",
|
"Clear watch history": "Verlauf löschen",
|
||||||
"unsubscribe": "abbestellen",
|
"Import/export data": "Daten im-/exportieren",
|
||||||
"Subscriptions": "Abonnements",
|
"Change password": "Passwort ändern",
|
||||||
"`x` unseen notifications": "`x` ungesehene Benachrichtigungen",
|
"Manage subscriptions": "Abonnements verwalten",
|
||||||
"search": "Suchen",
|
"Manage tokens": "Tokens verwalten",
|
||||||
"Sign out": "Abmelden",
|
"Watch history": "Verlauf",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Veröffentlicht unter AGPLv3 von Omar Roth.",
|
"Delete account": "Account löschen",
|
||||||
"Source available here.": "Quellcode verfügbar hier.",
|
"Administrator preferences": "Administrator-Einstellungen",
|
||||||
"View JavaScript license information.": "Javascript Lizenzinformationen anzeigen.",
|
"Default homepage: ": "Standard-Startseite: ",
|
||||||
"View privacy policy.": "",
|
"Feed menu: ": "Feed-Menü: ",
|
||||||
"Trending": "Trending",
|
"Top enabled: ": "Top aktiviert? ",
|
||||||
"Unlisted": "",
|
"CAPTCHA enabled: ": "CAPTCHA aktiviert? ",
|
||||||
"Watch video on Youtube": "Video auf YouTube ansehen",
|
"Login enabled: ": "Login aktiviert? ",
|
||||||
"Genre: ": "Genre: ",
|
"Registration enabled: ": "Registrierung aktiviert? ",
|
||||||
"License: ": "Lizenz: ",
|
"Report statistics: ": "Statistiken berichten? ",
|
||||||
"Family friendly? ": "Familienfreundlich? ",
|
"Save preferences": "Einstellungen speichern",
|
||||||
"Wilson score: ": "Wilson-Score: ",
|
"Subscription manager": "Abonnementverwaltung",
|
||||||
"Engagement: ": "Engagement: ",
|
"Token manager": "Tokenverwalter",
|
||||||
"Whitelisted regions: ": "Erlaubte Regionen: ",
|
"Token": "Token",
|
||||||
"Blacklisted regions: ": "Unerlaubte Regionen: ",
|
"`x` subscriptions": "`x` Abonnements",
|
||||||
"Shared `x`": "Geteilt `x`",
|
"`x` tokens": "`x` Tokens",
|
||||||
"Premieres in `x`": "",
|
"Import/export": "Importieren/Exportieren",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hallo! Anscheinend haben Sie JavaScript deaktiviert. Klicken Sie hier um Kommentare anzuzeigen, beachten sie dass es etwas länger dauern kann um sie zu laden.",
|
"unsubscribe": "abbestellen",
|
||||||
"View YouTube comments": "YouTube Kommentare anzeigen",
|
"revoke": "widerrufen",
|
||||||
"View more comments on Reddit": "Mehr Kommentare auf Reddit anzeigen",
|
"Subscriptions": "Abonnements",
|
||||||
"View `x` comments": "`x` Kommentare anzeigen",
|
"`x` unseen notifications": "`x` ungesehene Benachrichtigungen",
|
||||||
"View Reddit comments": "Reddit Kommentare anzeigen",
|
"search": "Suchen",
|
||||||
"Hide replies": "Antworten verstecken",
|
"Log out": "Abmelden",
|
||||||
"Show replies": "Antworten anzeigen",
|
"Released under the AGPLv3 by Omar Roth.": "Veröffentlicht unter AGPLv3 von Omar Roth.",
|
||||||
"Incorrect password": "Falsches Passwort",
|
"Source available here.": "Quellcode verfügbar hier.",
|
||||||
"Quota exceeded, try again in a few hours": "Kontingent überschritten, versuche es in ein paar Stunden erneut",
|
"View JavaScript license information.": "Javascript Lizenzinformationen anzeigen.",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Login nicht möglich, stellen Sie sicher dass two-factor Authentifikation (Authentifizierung oder SMS) aktiviert ist.",
|
"View privacy policy.": "Datenschutzerklärung einsehen.",
|
||||||
"Invalid TFA code": "Ungültiger TFA Code",
|
"Trending": "Trending",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Login fehlgeschlagen. Das kann daran liegen dass two-factor Authentifizierung in ihrem Account nicht aktiviert ist.",
|
"Public": "",
|
||||||
"Invalid answer": "Ungültige Antwort",
|
"Unlisted": "Nicht aufgeführt",
|
||||||
"Invalid CAPTCHA": "Ungültiges CAPTCHA",
|
"Private": "",
|
||||||
"CAPTCHA is a required field": "CAPTCHA ist eine erforderliche Eingabe",
|
"View all playlists": "",
|
||||||
"User ID is a required field": "Benutzer ID ist eine erforderliche Eingabe",
|
"Updated `x` ago": "",
|
||||||
"Password is a required field": "Passwort ist eine erforderliche Eingabe",
|
"Delete playlist `x`?": "",
|
||||||
"Invalid username or password": "Ungültiger Benutzername oder Passwort",
|
"Delete playlist": "",
|
||||||
"Please sign in using 'Sign in with Google'": "Bitte melden sie sich mit 'Mit Google anmelden' an",
|
"Create playlist": "",
|
||||||
"Password cannot be empty": "Passwort darf nicht leer sein",
|
"Title": "",
|
||||||
"Password cannot be longer than 55 characters": "Passwort darf nicht länger als 55 Zeichen sein",
|
"Playlist privacy": "",
|
||||||
"Please sign in": "Bitte anmelden",
|
"Editing playlist `x`": "",
|
||||||
"Invidious Private Feed for `x`": "Invidious Persönlicher Feed für `x`",
|
"Watch on YouTube": "Video auf YouTube ansehen",
|
||||||
"channel:`x`": "Kanal:`x`",
|
"Hide annotations": "Anmerkungen ausblenden",
|
||||||
"Deleted or invalid channel": "Gelöschter oder ungültiger Kanal",
|
"Show annotations": "Anmerkungen anzeigen",
|
||||||
"This channel does not exist.": "Dieser Kanal existiert nicht.",
|
"Genre: ": "Genre: ",
|
||||||
"Could not get channel info.": "Kanalinformationen konnten nicht geladen werden.",
|
"License: ": "Lizenz: ",
|
||||||
"Could not fetch comments": "Kommentare konnten nicht geladen werden",
|
"Family friendly? ": "Familienfreundlich? ",
|
||||||
"View `x` replies": "Zeige `x` Antworten",
|
"Wilson score: ": "Wilson-Score: ",
|
||||||
"`x` ago": "vor `x`",
|
"Engagement: ": "Engagement: ",
|
||||||
"Load more": "Mehr laden",
|
"Whitelisted regions: ": "Erlaubte Regionen: ",
|
||||||
"`x` points": "`x` Punkte",
|
"Blacklisted regions: ": "Unerlaubte Regionen: ",
|
||||||
"Could not create mix.": "Mix konnte nicht erstellt werden.",
|
"Shared `x`": "Geteilt `x`",
|
||||||
"Playlist is empty": "Playlist ist leer",
|
"`x` views": "`x` Aufrufe",
|
||||||
"Invalid playlist.": "Ungültige Playlist.",
|
"Premieres in `x`": "Zuerst gesehen in `x`",
|
||||||
"Playlist does not exist.": "Playlist existiert nicht.",
|
"Premieres `x`": "Erster Start `x`",
|
||||||
"Could not pull trending pages.": "Trending Seiten konnten nicht geladen werden.",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hallo! Anscheinend haben Sie JavaScript deaktiviert. Klicken Sie hier um Kommentare anzuzeigen, beachten sie dass es etwas länger dauern kann um sie zu laden.",
|
||||||
"Hidden field \"challenge\" is a required field": "Verstecktes Feld \"challenge\" ist eine erforderliche Eingabe",
|
"View YouTube comments": "YouTube Kommentare anzeigen",
|
||||||
"Hidden field \"token\" is a required field": "Verstecktes Feld \"token\" ist eine erforderliche Eingabe",
|
"View more comments on Reddit": "Mehr Kommentare auf Reddit anzeigen",
|
||||||
"Invalid challenge": "Ungültiger Test",
|
"View `x` comments": "`x` Kommentare anzeigen",
|
||||||
"Invalid token": "Ungöltige Marke",
|
"View Reddit comments": "Reddit Kommentare anzeigen",
|
||||||
"Invalid user": "Ungültiger Benutzer",
|
"Hide replies": "Antworten verstecken",
|
||||||
"Token is expired, please try again": "Marke ist abgelaufen, bitte erneut versuchen",
|
"Show replies": "Antworten anzeigen",
|
||||||
"English": "Englisch",
|
"Incorrect password": "Falsches Passwort",
|
||||||
"English (auto-generated)": "Englisch (automatisch erzeugt)",
|
"Quota exceeded, try again in a few hours": "Kontingent überschritten, versuche es in ein paar Stunden erneut",
|
||||||
"Afrikaans": "Afrikaans",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Login nicht möglich, stellen Sie sicher dass two-factor Authentifikation (Authentifizierung oder SMS) aktiviert ist.",
|
||||||
"Albanian": "Albanisch",
|
"Invalid TFA code": "Ungültiger TFA Code",
|
||||||
"Amharic": "Amharisch",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Login fehlgeschlagen. Das kann daran liegen dass two-factor Authentifizierung in ihrem Account nicht aktiviert ist.",
|
||||||
"Arabic": "Arabisch",
|
"Wrong answer": "Ungültige Antwort",
|
||||||
"Armenian": "Armenisch",
|
"Erroneous CAPTCHA": "Ungültiges CAPTCHA",
|
||||||
"Azerbaijani": "Aserbaidschanisch",
|
"CAPTCHA is a required field": "CAPTCHA ist eine erforderliche Eingabe",
|
||||||
"Bangla": "Bengalisch",
|
"User ID is a required field": "Benutzer ID ist eine erforderliche Eingabe",
|
||||||
"Basque": "Baskisch",
|
"Password is a required field": "Passwort ist eine erforderliche Eingabe",
|
||||||
"Belarusian": "Weißrussisch",
|
"Wrong username or password": "Ungültiger Benutzername oder Passwort",
|
||||||
"Bosnian": "Bosnisch",
|
"Please sign in using 'Log in with Google'": "Bitte melden sie sich mit 'Mit Google anmelden' an",
|
||||||
"Bulgarian": "Bulgarisch",
|
"Password cannot be empty": "Passwort darf nicht leer sein",
|
||||||
"Burmese": "Burmesisch",
|
"Password cannot be longer than 55 characters": "Passwort darf nicht länger als 55 Zeichen sein",
|
||||||
"Catalan": "Katalanisch",
|
"Please log in": "Bitte anmelden",
|
||||||
"Cebuano": "Cebuano",
|
"Invidious Private Feed for `x`": "Invidious Persönlicher Feed für `x`",
|
||||||
"Chinese (Simplified)": "Chinesisch (vereinfacht)",
|
"channel:`x`": "Kanal:`x`",
|
||||||
"Chinese (Traditional)": "Chinesisch (traditionell)",
|
"Deleted or invalid channel": "Gelöschter oder ungültiger Kanal",
|
||||||
"Corsican": "Korsisch",
|
"This channel does not exist.": "Dieser Kanal existiert nicht.",
|
||||||
"Croatian": "Kroatisch",
|
"Could not get channel info.": "Kanalinformationen konnten nicht geladen werden.",
|
||||||
"Czech": "Tschechisch",
|
"Could not fetch comments": "Kommentare konnten nicht geladen werden",
|
||||||
"Danish": "Dänisch",
|
"View `x` replies": "Zeige `x` Antworten",
|
||||||
"Dutch": "Niederländisch",
|
"`x` ago": "vor `x`",
|
||||||
"Esperanto": "Esperanto",
|
"Load more": "Mehr laden",
|
||||||
"Estonian": "Estnisch",
|
"`x` points": "`x` Punkte",
|
||||||
"Filipino": "Philippinisch",
|
"Could not create mix.": "Mix konnte nicht erstellt werden.",
|
||||||
"Finnish": "Finnisch",
|
"Empty playlist": "Playlist ist leer",
|
||||||
"French": "Französisch",
|
"Not a playlist.": "Ungültige Playlist.",
|
||||||
"Galician": "Galizisch",
|
"Playlist does not exist.": "Playlist existiert nicht.",
|
||||||
"Georgian": "Georgisch",
|
"Could not pull trending pages.": "Trending Seiten konnten nicht geladen werden.",
|
||||||
"German": "Deutsch",
|
"Hidden field \"challenge\" is a required field": "Verstecktes Feld \"challenge\" ist eine erforderliche Eingabe",
|
||||||
"Greek": "Griechisch",
|
"Hidden field \"token\" is a required field": "Verstecktes Feld \"token\" ist eine erforderliche Eingabe",
|
||||||
"Gujarati": "Gujarati",
|
"Erroneous challenge": "Ungültiger Test",
|
||||||
"Haitian Creole": "Haitianisches Kreolisch",
|
"Erroneous token": "Ungültiger Token",
|
||||||
"Hausa": "Hausa",
|
"No such user": "Ungültiger Benutzer",
|
||||||
"Hawaiian": "Hawaiianisch",
|
"Token is expired, please try again": "Token ist abgelaufen, bitte erneut versuchen",
|
||||||
"Hebrew": "Hebräisch",
|
"English": "Englisch",
|
||||||
"Hindi": "Hindi",
|
"English (auto-generated)": "Englisch (automatisch erzeugt)",
|
||||||
"Hmong": "Hmong",
|
"Afrikaans": "Afrikaans",
|
||||||
"Hungarian": "Ungarisch",
|
"Albanian": "Albanisch",
|
||||||
"Icelandic": "Isländisch",
|
"Amharic": "Amharisch",
|
||||||
"Igbo": "Igbo",
|
"Arabic": "Arabisch",
|
||||||
"Indonesian": "Indonesisch",
|
"Armenian": "Armenisch",
|
||||||
"Irish": "Irisch",
|
"Azerbaijani": "Aserbaidschanisch",
|
||||||
"Italian": "Italienisch",
|
"Bangla": "Bengalisch",
|
||||||
"Japanese": "Japanisch",
|
"Basque": "Baskisch",
|
||||||
"Javanese": "Javanisch",
|
"Belarusian": "Weißrussisch",
|
||||||
"Kannada": "Kannada",
|
"Bosnian": "Bosnisch",
|
||||||
"Kazakh": "Kasachisch",
|
"Bulgarian": "Bulgarisch",
|
||||||
"Khmer": "Khmer",
|
"Burmese": "Burmesisch",
|
||||||
"Korean": "Koreanisch",
|
"Catalan": "Katalanisch",
|
||||||
"Kurdish": "Kurdisch",
|
"Cebuano": "Cebuano",
|
||||||
"Kyrgyz": "Kirgisisch",
|
"Chinese (Simplified)": "Chinesisch (vereinfacht)",
|
||||||
"Lao": "Laotisch",
|
"Chinese (Traditional)": "Chinesisch (traditionell)",
|
||||||
"Latin": "Lateinisch",
|
"Corsican": "Korsisch",
|
||||||
"Latvian": "Lettisch",
|
"Croatian": "Kroatisch",
|
||||||
"Lithuanian": "Litauisch",
|
"Czech": "Tschechisch",
|
||||||
"Luxembourgish": "Luxemburgisch",
|
"Danish": "Dänisch",
|
||||||
"Macedonian": "Mazedonisch",
|
"Dutch": "Niederländisch",
|
||||||
"Malagasy": "Madagassisch",
|
"Esperanto": "Esperanto",
|
||||||
"Malay": "Malaiisch",
|
"Estonian": "Estnisch",
|
||||||
"Malayalam": "Malayalam",
|
"Filipino": "Philippinisch",
|
||||||
"Maltese": "Maltesisch",
|
"Finnish": "Finnisch",
|
||||||
"Maori": "Maori",
|
"French": "Französisch",
|
||||||
"Marathi": "Marathi",
|
"Galician": "Galizisch",
|
||||||
"Mongolian": "Mongolisch",
|
"Georgian": "Georgisch",
|
||||||
"Nepali": "Nepalesisch",
|
"German": "Deutsch",
|
||||||
"Norwegian": "Norwegisch",
|
"Greek": "Griechisch",
|
||||||
"Nyanja": "Nyanja",
|
"Gujarati": "Gujarati",
|
||||||
"Pashto": "Paschtunisch",
|
"Haitian Creole": "Haitianisches Kreolisch",
|
||||||
"Persian": "Persisch",
|
"Hausa": "Hausa",
|
||||||
"Polish": "Polnisch",
|
"Hawaiian": "Hawaiianisch",
|
||||||
"Portuguese": "Portugiesisch",
|
"Hebrew": "Hebräisch",
|
||||||
"Punjabi": "Pandschabi",
|
"Hindi": "Hindi",
|
||||||
"Romanian": "Rumänisch",
|
"Hmong": "Hmong",
|
||||||
"Russian": "Russisch",
|
"Hungarian": "Ungarisch",
|
||||||
"Samoan": "Samoanisch",
|
"Icelandic": "Isländisch",
|
||||||
"Scottish Gaelic": "Schottisches Gälisch",
|
"Igbo": "Igbo",
|
||||||
"Serbian": "Serbisch",
|
"Indonesian": "Indonesisch",
|
||||||
"Shona": "Schona",
|
"Irish": "Irisch",
|
||||||
"Sindhi": "Sindhi",
|
"Italian": "Italienisch",
|
||||||
"Sinhala": "Singhalesisch",
|
"Japanese": "Japanisch",
|
||||||
"Slovak": "Slowakisch",
|
"Javanese": "Javanisch",
|
||||||
"Slovenian": "Slowenisch",
|
"Kannada": "Kannada",
|
||||||
"Somali": "Somali",
|
"Kazakh": "Kasachisch",
|
||||||
"Southern Sotho": "Südliches Sotho",
|
"Khmer": "Khmer",
|
||||||
"Spanish": "Spanisch",
|
"Korean": "Koreanisch",
|
||||||
"Spanish (Latin America)": "Spanisch (Lateinamerika)",
|
"Kurdish": "Kurdisch",
|
||||||
"Sundanese": "Sundanesisch",
|
"Kyrgyz": "Kirgisisch",
|
||||||
"Swahili": "Suaheli",
|
"Lao": "Laotisch",
|
||||||
"Swedish": "Schwedisch",
|
"Latin": "Lateinisch",
|
||||||
"Tajik": "Tadschikisch",
|
"Latvian": "Lettisch",
|
||||||
"Tamil": "Tamilisch",
|
"Lithuanian": "Litauisch",
|
||||||
"Telugu": "Telugu",
|
"Luxembourgish": "Luxemburgisch",
|
||||||
"Thai": "Thailändisch",
|
"Macedonian": "Mazedonisch",
|
||||||
"Turkish": "Türkisch",
|
"Malagasy": "Madagassisch",
|
||||||
"Ukrainian": "Ukrainisch",
|
"Malay": "Malaiisch",
|
||||||
"Urdu": "Urdu",
|
"Malayalam": "Malayalam",
|
||||||
"Uzbek": "Usbekisch",
|
"Maltese": "Maltesisch",
|
||||||
"Vietnamese": "Vietnamesisch",
|
"Maori": "Maori",
|
||||||
"Welsh": "Walisisch",
|
"Marathi": "Marathi",
|
||||||
"Western Frisian": "Westfriesisch",
|
"Mongolian": "Mongolisch",
|
||||||
"Xhosa": "Xhosa",
|
"Nepali": "Nepalesisch",
|
||||||
"Yiddish": "Jiddisch",
|
"Norwegian Bokmål": "Norwegisch",
|
||||||
"Yoruba": "Joruba",
|
"Nyanja": "Nyanja",
|
||||||
"Zulu": "Zulu",
|
"Pashto": "Paschtunisch",
|
||||||
"`x` years": "`x` Jahre",
|
"Persian": "Persisch",
|
||||||
"`x` months": "`x` Monate",
|
"Polish": "Polnisch",
|
||||||
"`x` weeks": "`x` Wochen",
|
"Portuguese": "Portugiesisch",
|
||||||
"`x` days": "`x` Tage",
|
"Punjabi": "Pandschabi",
|
||||||
"`x` hours": "`x` Stunden",
|
"Romanian": "Rumänisch",
|
||||||
"`x` minutes": "`x` Minuten",
|
"Russian": "Russisch",
|
||||||
"`x` seconds": "`x` Sekunden",
|
"Samoan": "Samoanisch",
|
||||||
"Fallback comments: ": "Alternative Kommentare: ",
|
"Scottish Gaelic": "Schottisches Gälisch",
|
||||||
"Popular": "Populär",
|
"Serbian": "Serbisch",
|
||||||
"Top": "Top",
|
"Shona": "Schona",
|
||||||
"About": "Über",
|
"Sindhi": "Sindhi",
|
||||||
"Rating: ": "Bewertung: ",
|
"Sinhala": "Singhalesisch",
|
||||||
"Language: ": "Sprache: ",
|
"Slovak": "Slowakisch",
|
||||||
"Default": "",
|
"Slovenian": "Slowenisch",
|
||||||
"Music": "",
|
"Somali": "Somali",
|
||||||
"Gaming": "",
|
"Southern Sotho": "Südliches Sotho",
|
||||||
"News": "",
|
"Spanish": "Spanisch",
|
||||||
"Movies": "",
|
"Spanish (Latin America)": "Spanisch (Lateinamerika)",
|
||||||
"Download": "",
|
"Sundanese": "Sundanesisch",
|
||||||
"Download as: ": "",
|
"Swahili": "Suaheli",
|
||||||
"%A %B %-d, %Y": "",
|
"Swedish": "Schwedisch",
|
||||||
"(edited)": "",
|
"Tajik": "Tadschikisch",
|
||||||
"Youtube permalink of the comment": "",
|
"Tamil": "Tamilisch",
|
||||||
"`x` marked it with a ❤": "",
|
"Telugu": "Telugu",
|
||||||
"Audio mode": "",
|
"Thai": "Thailändisch",
|
||||||
"Video mode": "",
|
"Turkish": "Türkisch",
|
||||||
"Videos": "",
|
"Ukrainian": "Ukrainisch",
|
||||||
"Playlists": "",
|
"Urdu": "Urdu",
|
||||||
"Current version: ": ""
|
"Uzbek": "Usbekisch",
|
||||||
}
|
"Vietnamese": "Vietnamesisch",
|
||||||
|
"Welsh": "Walisisch",
|
||||||
|
"Western Frisian": "Westfriesisch",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Jiddisch",
|
||||||
|
"Yoruba": "Joruba",
|
||||||
|
"Zulu": "Zulu",
|
||||||
|
"`x` years": "`x` Jahre",
|
||||||
|
"`x` months": "`x` Monate",
|
||||||
|
"`x` weeks": "`x` Wochen",
|
||||||
|
"`x` days": "`x` Tage",
|
||||||
|
"`x` hours": "`x` Stunden",
|
||||||
|
"`x` minutes": "`x` Minuten",
|
||||||
|
"`x` seconds": "`x` Sekunden",
|
||||||
|
"Fallback comments: ": "Alternative Kommentare: ",
|
||||||
|
"Popular": "Populär",
|
||||||
|
"Top": "Top",
|
||||||
|
"About": "Über",
|
||||||
|
"Rating: ": "Bewertung: ",
|
||||||
|
"Language: ": "Sprache: ",
|
||||||
|
"View as playlist": "Als Wiedergabeliste anzeigen",
|
||||||
|
"Default": "Standard",
|
||||||
|
"Music": "Musik",
|
||||||
|
"Gaming": "Videospiele",
|
||||||
|
"News": "Neuigkeiten",
|
||||||
|
"Movies": "Filme",
|
||||||
|
"Download": "Herunterladen",
|
||||||
|
"Download as: ": "Herunterladen als: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(editiert)",
|
||||||
|
"YouTube comment permalink": "YouTube-Kommentar Permalink",
|
||||||
|
"permalink": "Permalink",
|
||||||
|
"`x` marked it with a ❤": "`x` markierte es mit einem ❤",
|
||||||
|
"Audio mode": "Audiomodus",
|
||||||
|
"Video mode": "Videomodus",
|
||||||
|
"Videos": "Videos",
|
||||||
|
"Playlists": "Wiedergabelisten",
|
||||||
|
"Community": "Gemeinschaft",
|
||||||
|
"Current version: ": "Aktuelle Version: "
|
||||||
|
}
|
||||||
381
locales/el.json
Normal file
381
locales/el.json
Normal file
@@ -0,0 +1,381 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` συνδρομητής",
|
||||||
|
"": "`x` συνδρομητές"
|
||||||
|
},
|
||||||
|
"`x` videos": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` βίντεο",
|
||||||
|
"": "`x` βίντεο"
|
||||||
|
},
|
||||||
|
"`x` playlists": "",
|
||||||
|
"LIVE": "ΖΩΝΤΑΝΑ",
|
||||||
|
"Shared `x` ago": "Μοιράστηκε πριν `x`",
|
||||||
|
"Unsubscribe": "Απεγγραφή",
|
||||||
|
"Subscribe": "Εγγραφή",
|
||||||
|
"View channel on YouTube": "Προβολή καναλιού στο YouTube",
|
||||||
|
"View playlist on YouTube": "",
|
||||||
|
"newest": "νεότερα",
|
||||||
|
"oldest": "παλιότερα",
|
||||||
|
"popular": "δημοφιλή",
|
||||||
|
"last": "τελευταία",
|
||||||
|
"Next page": "Επόμενη σελίδα",
|
||||||
|
"Previous page": "Προηγούμενη σελίδα",
|
||||||
|
"Clear watch history?": "Διαγραφή ιστορικού προβολής;",
|
||||||
|
"New password": "Νέος κωδικός πρόσβασης",
|
||||||
|
"New passwords must match": "Οι νέοι κωδικοί πρόσβασης πρέπει να ταιριάζουν",
|
||||||
|
"Cannot change password for Google accounts": "Δεν επιτρέπεται η αλλαγή κωδικού πρόσβασης λογαριασμών Google",
|
||||||
|
"Authorize token?": "Εξουσιοδότηση διασύνδεσης;",
|
||||||
|
"Authorize token for `x`?": "Εξουσιοδότηση διασύνδεσης με `x`;",
|
||||||
|
"Yes": "Ναι",
|
||||||
|
"No": "Όχι",
|
||||||
|
"Import and Export Data": "Εισαγωγή και Εξαγωγή Δεδομένων",
|
||||||
|
"Import": "Εισαγωγή",
|
||||||
|
"Import Invidious data": "Εισαγωγή δεδομένων Invidious",
|
||||||
|
"Import YouTube subscriptions": "Εισαγωγή συνδρομών YouTube",
|
||||||
|
"Import FreeTube subscriptions (.db)": "Εισαγωγή συνδρομών FreeTube (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "Εισαγωγή συνδρομών NewPipe (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "Εισαγωγή δεδομένων NewPipe (.zip)",
|
||||||
|
"Export": "Εξαγωγή",
|
||||||
|
"Export subscriptions as OPML": "Εξαγωγή συνδρομών ως OPML",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Εξαγωγή συνδρομών ως OPML (για NewPipe & FreeTube)",
|
||||||
|
"Export data as JSON": "Εξαγωγή δεδομένων ως JSON",
|
||||||
|
"Delete account?": "Διαγραφή λογαριασμού;",
|
||||||
|
"History": "Ιστορικό",
|
||||||
|
"An alternative front-end to YouTube": "Μία εναλλακτική πλατφόρμα για το YouTube",
|
||||||
|
"JavaScript license information": "Πληροφορίες άδειας JavaScript",
|
||||||
|
"source": "πηγή",
|
||||||
|
"Log in": "Σύνδεση",
|
||||||
|
"Log in/register": "Σύνδεση/εγγραφή",
|
||||||
|
"Log in with Google": "Σύνδεση με Google",
|
||||||
|
"User ID": "Ταυτότητα χρήστη",
|
||||||
|
"Password": "Κωδικός πρόσβασης",
|
||||||
|
"Time (h:mm:ss):": "Ώρα (ω:λλ:δδ):",
|
||||||
|
"Text CAPTCHA": "Κείμενο CAPTCHA",
|
||||||
|
"Image CAPTCHA": "Εικόνα CAPTCHA",
|
||||||
|
"Sign In": "Σύνδεση",
|
||||||
|
"Register": "Εγγραφή",
|
||||||
|
"E-mail": "E-mail",
|
||||||
|
"Google verification code": "Κωδικός επαλήθευσης Google",
|
||||||
|
"Preferences": "Προτιμήσεις",
|
||||||
|
"Player preferences": "Προτιμήσεις αναπαραγωγής",
|
||||||
|
"Always loop: ": "Αυτόματη επανάληψη: ",
|
||||||
|
"Autoplay: ": "Αυτόματη αναπαραγωγή: ",
|
||||||
|
"Play next by default: ": "Αναπαραγωγή επόμενου: ",
|
||||||
|
"Autoplay next video: ": "Αυτόματη αναπαραγωγή επόμενου: ",
|
||||||
|
"Listen by default: ": "Φόρτωση μόνο ήχου: ",
|
||||||
|
"Proxy videos: ": "Αναπαραγωγή με διακομιστή μεσολάβησης (proxy): ",
|
||||||
|
"Default speed: ": "Προεπιλεγμένη ταχύτητα: ",
|
||||||
|
"Preferred video quality: ": "Προτιμώμενη ανάλυση: ",
|
||||||
|
"Player volume: ": "Ένταση αναπαραγωγής: ",
|
||||||
|
"Default comments: ": "Προεπιλεγμένα σχόλια: ",
|
||||||
|
"youtube": "youtube",
|
||||||
|
"reddit": "reddit",
|
||||||
|
"Default captions: ": "Προεπιλεγμένοι υπότιτλοι: ",
|
||||||
|
"Fallback captions: ": "Εναλλακτικοί υπότιτλοι: ",
|
||||||
|
"Show related videos: ": "Προβολή σχετικών βίντεο; ",
|
||||||
|
"Show annotations by default: ": "Αυτόματη προβολή σημειώσεων; :",
|
||||||
|
"Visual preferences": "Προτιμήσεις εμφάνισης",
|
||||||
|
"Player style: ": "",
|
||||||
|
"Dark mode: ": "Σκοτεινή λειτουργία: ",
|
||||||
|
"Theme: ": "",
|
||||||
|
"dark": "",
|
||||||
|
"light": "",
|
||||||
|
"Thin mode: ": "Ελαφριά λειτουργία: ",
|
||||||
|
"Subscription preferences": "Προτιμήσεις συνδρομών",
|
||||||
|
"Show annotations by default for subscribed channels: ": "Προβολή σημειώσεων μόνο για κανάλια στα οποία είστε συνδρομητής; ",
|
||||||
|
"Redirect homepage to feed: ": "Ανακατεύθυνση αρχικής στη ροή συνδρομών: ",
|
||||||
|
"Number of videos shown in feed: ": "Αριθμός βίντεο ανά σελίδα ροής συνδρομών: ",
|
||||||
|
"Sort videos by: ": "Ταξινόμηση ανά: ",
|
||||||
|
"published": "ημερομηνία δημοσίευσης",
|
||||||
|
"published - reverse": "ημερομηνία δημοσίευσης - ανάποδα",
|
||||||
|
"alphabetically": "αλφαβητικά",
|
||||||
|
"alphabetically - reverse": "αλφαβητικά - ανάποδα",
|
||||||
|
"channel name": "όνομα καναλιού",
|
||||||
|
"channel name - reverse": "όνομα καναλιού - ανάποδα",
|
||||||
|
"Only show latest video from channel: ": "Προβολή μόνο του τελευταίου βίντεο του καναλιού: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "Προβολή μόνο του τελευταίου μη-προβεβλημένου βίντεο του καναλιού: ",
|
||||||
|
"Only show unwatched: ": "Προβολή μόνο μη-προβεβλημένων: ",
|
||||||
|
"Only show notifications (if there are any): ": "Προβολή μόνο ειδοποιήσεων (αν υπάρχουν): ",
|
||||||
|
"Enable web notifications": "",
|
||||||
|
"`x` uploaded a video": "",
|
||||||
|
"`x` is live": "",
|
||||||
|
"Data preferences": "Προτιμήσεις δεδομένων",
|
||||||
|
"Clear watch history": "Εκκαθάριση ιστορικού προβολής",
|
||||||
|
"Import/export data": "Εισαγωγή/εξαγωγή δεδομένων",
|
||||||
|
"Change password": "Αλλαγή κωδικού πρόσβασης",
|
||||||
|
"Manage subscriptions": "Διαχείριση συνδρομών",
|
||||||
|
"Manage tokens": "Διαχείριση διασυνδέσεων",
|
||||||
|
"Watch history": "Ιστορικό προβολής",
|
||||||
|
"Delete account": "Διαγραφή λογαριασμού",
|
||||||
|
"Administrator preferences": "Προτιμήσεις διαχειριστή",
|
||||||
|
"Default homepage: ": "Προεπιλεγμένη αρχική: ",
|
||||||
|
"Feed menu: ": "Μενού ροής συνδρομών: ",
|
||||||
|
"Top enabled: ": "Ενεργοποίηση κορυφαίων; ",
|
||||||
|
"CAPTCHA enabled: ": "Ενεργοποίηση CAPTCHA; ",
|
||||||
|
"Login enabled: ": "Ενεργοποίηση σύνδεσης; ",
|
||||||
|
"Registration enabled: ": "Ενεργοποίηση εγγραφής; ",
|
||||||
|
"Report statistics: ": "Αναφορά στατιστικών; ",
|
||||||
|
"Save preferences": "Αποθήκευση προτιμήσεων",
|
||||||
|
"Subscription manager": "Διαχειριστής συνδρομών",
|
||||||
|
"Token manager": "Διαχειριστής διασυνδέσεων",
|
||||||
|
"Token": "Διασύνδεση",
|
||||||
|
"`x` subscriptions": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` συνδρομή",
|
||||||
|
"": "`x` συνδρομές"
|
||||||
|
},
|
||||||
|
"`x` tokens": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` διασύνδεση",
|
||||||
|
"": "`x` διασυνδέσεις"
|
||||||
|
},
|
||||||
|
"Import/export": "Εισαγωγή/εξαγωγή",
|
||||||
|
"unsubscribe": "κατάργηση συνδρομής",
|
||||||
|
"revoke": "ανάκληση",
|
||||||
|
"Subscriptions": "Συνδρομές",
|
||||||
|
"`x` unseen notifications": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` καινούρια ειδοποίηση",
|
||||||
|
"": "`x` καινούριες ειδοποιήσεις"
|
||||||
|
},
|
||||||
|
"search": "αναζήτηση",
|
||||||
|
"Log out": "Αποσύνδεση",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Κυκλοφορεί υπό την άδεια AGPLv3 από τον Omar Roth.",
|
||||||
|
"Source available here.": "Προβολή πηγαίου κώδικα εδώ.",
|
||||||
|
"View JavaScript license information.": "Προβολή πληροφοριών άδειας JavaScript.",
|
||||||
|
"View privacy policy.": "Προβολή πολιτικής απορρήτου.",
|
||||||
|
"Trending": "Τάσεις",
|
||||||
|
"Public": "",
|
||||||
|
"Unlisted": "Κρυφό",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "Προβολή στο YouTube",
|
||||||
|
"Hide annotations": "Απόκρυψη σημειώσεων",
|
||||||
|
"Show annotations": "Προβολή σημειώσεων",
|
||||||
|
"Genre: ": "Είδος: ",
|
||||||
|
"License: ": "Άδεια: ",
|
||||||
|
"Family friendly? ": "Φιλικό προς την οικογένεια; ",
|
||||||
|
"Wilson score: ": "Wilson score: ",
|
||||||
|
"Engagement: ": "Ενδιαφέρον: ",
|
||||||
|
"Whitelisted regions: ": "Επιτρεπτές περιοχές: ",
|
||||||
|
"Blacklisted regions: ": "Μη-επιτρεπτές περιοχές: ",
|
||||||
|
"Shared `x`": "Μοιράστηκε το `x`",
|
||||||
|
"`x` views": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` προβολή",
|
||||||
|
"": "`x` προβολές"
|
||||||
|
},
|
||||||
|
"Premieres in `x`": "Πρώτη προβολή σε `x`",
|
||||||
|
"Premieres `x`": "",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Γεια! Φαίνεται πως έχετε απενεργοποιήσει το JavaScript. Πατήστε εδώ για προβολή σχολίων, αλλά έχετε υπ'όψιν σας πως ίσως φορτώσουν πιο αργά. ",
|
||||||
|
"View YouTube comments": "Προβολή σχολίων από το YouTube",
|
||||||
|
"View more comments on Reddit": "Προβολή περισσότερων σχολίων στο Reddit",
|
||||||
|
"View `x` comments": "Προβολή `x` σχολίων",
|
||||||
|
"View Reddit comments": "Προβολή σχολίων από το Reddit",
|
||||||
|
"Hide replies": "Απόκρυψη απαντήσεων",
|
||||||
|
"Show replies": "Προβολή απαντήσεων",
|
||||||
|
"Incorrect password": "Λανθασμένος κωδικός πρόσβασης",
|
||||||
|
"Quota exceeded, try again in a few hours": "Έχετε υπερβεί το όριο προσπαθειών, δοκιμάστε ξανα σε λίγες ώρες",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Αδυναμία σύνδεσης, βεβαιωθείτε πως ο έλεγχος ταυτότητας δύο παραγόντων (με Authenticator ή SMS) είναι ενεργοποιημένος.",
|
||||||
|
"Invalid TFA code": "Μη έγκυρος κωδικός ελέγχου ταυτότητας δύο παραγόντων",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Αποτυχία σύνδεσης. Ίσως ευθύνεται η έλλειψη ελέγχου ταυτότητας δύο παραγόντων για το λογαριασμό σας.",
|
||||||
|
"Wrong answer": "Λανθασμένη απάντηση",
|
||||||
|
"Erroneous CAPTCHA": "Λανθασμένο CAPTCHA",
|
||||||
|
"CAPTCHA is a required field": "Το CAPTCHA είναι απαιτούμενο πεδίο",
|
||||||
|
"User ID is a required field": "Η ταυτότητα χρήστη είναι απαιτούμενο πεδίο",
|
||||||
|
"Password is a required field": "Ο κωδικός πρόσβασης είναι απαιτούμενο πεδίο",
|
||||||
|
"Wrong username or password": "Λανθασμένο όνομα χρήστη ή κωδικός πρόσβασης",
|
||||||
|
"Please sign in using 'Log in with Google'": "Συνδεθείτε με την επιλογή 'Σύνδεση με Google'",
|
||||||
|
"Password cannot be empty": "Ο κωδικός πρόσβασης δεν γίνεται να είναι κενός",
|
||||||
|
"Password cannot be longer than 55 characters": "Ο κωδικός πρόσβασης δεν γίνεται να υπερβαίνει τους 55 χαρακτήρες",
|
||||||
|
"Please log in": "Συνδεθείτε",
|
||||||
|
"Invidious Private Feed for `x`": "Ροή RSS του Invidious για το χρήστη `x`",
|
||||||
|
"channel:`x`": "κανάλι:`x`",
|
||||||
|
"Deleted or invalid channel": "Διαγραμμένο ή μη έγκυρο κανάλι",
|
||||||
|
"This channel does not exist.": "Αυτό το κανάλι δεν υπάρχει.",
|
||||||
|
"Could not get channel info.": "Αδύναμια εύρεσης πληροφοριών καναλιού.",
|
||||||
|
"Could not fetch comments": "Αδυναμία λήψης σχολίων",
|
||||||
|
"View `x` replies": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Προβολή `x` απάντησης",
|
||||||
|
"": "Προβολή `x` απαντήσεων"
|
||||||
|
},
|
||||||
|
"`x` ago": "Πριν `x`",
|
||||||
|
"Load more": "Φόρτωση περισσότερων",
|
||||||
|
"`x` points": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` βαθμός",
|
||||||
|
"": "`x` βαθμοί"
|
||||||
|
},
|
||||||
|
"Could not create mix.": "Αδυναμία δημιουργίας μίξης.",
|
||||||
|
"Empty playlist": "Κενή λίστα αναπαραγωγής",
|
||||||
|
"Not a playlist.": "Μη έγκυρη λίστα αναπαραγωγής",
|
||||||
|
"Playlist does not exist.": "Μη υπαρκτή λίστα αναπαραγωγής.",
|
||||||
|
"Could not pull trending pages.": "Αδυναμία λήψης σελίδας τάσεων.",
|
||||||
|
"Hidden field \"challenge\" is a required field": "Το Κρυφό πεδίο \"δοκιμασία\" είναι απαραίτητο",
|
||||||
|
"Hidden field \"token\" is a required field": "Το κρυφό πεδίο \"αναγνωριστικό διασύνδεσης\" είναι απαραίτητο",
|
||||||
|
"Erroneous challenge": "Λανθασμένη δοκιμασία",
|
||||||
|
"Erroneous token": "Λανθασμένο αναγνωριστικό διασύνδεσης",
|
||||||
|
"No such user": "Μη υπαρκτός χρήστης",
|
||||||
|
"Token is expired, please try again": "Το αναγνωριστικό διασύνδεσης έχει λήξει, παρακαλώ ξαναπροσπαθήστε",
|
||||||
|
"English": "Αγγλικά",
|
||||||
|
"English (auto-generated)": "Αγγλικά (αυτόματα)",
|
||||||
|
"Afrikaans": "Αφρικάανς",
|
||||||
|
"Albanian": "Αλβανικά",
|
||||||
|
"Amharic": "Αμχαρικά",
|
||||||
|
"Arabic": "Αραβικά",
|
||||||
|
"Armenian": "Αρμένικα",
|
||||||
|
"Azerbaijani": "Αζερικά",
|
||||||
|
"Bangla": "Μπενγκάλι",
|
||||||
|
"Basque": "Βασκικά",
|
||||||
|
"Belarusian": "Λευκορωσικά",
|
||||||
|
"Bosnian": "Βοσνιακά",
|
||||||
|
"Bulgarian": "Βουλγάρικα",
|
||||||
|
"Burmese": "Βιρμανικά",
|
||||||
|
"Catalan": "Καταλανικά",
|
||||||
|
"Cebuano": "Κεμπουάνο",
|
||||||
|
"Chinese (Simplified)": "Κινέζικα (Απλοποιημένα)",
|
||||||
|
"Chinese (Traditional)": "Κινέζικα (Παραδοσιακά)",
|
||||||
|
"Corsican": "Κορσικανικά",
|
||||||
|
"Croatian": "Κροατικά",
|
||||||
|
"Czech": "Τσέχικα",
|
||||||
|
"Danish": "Δανέζικα",
|
||||||
|
"Dutch": "Ολλανδικά",
|
||||||
|
"Esperanto": "Εσπεράντο",
|
||||||
|
"Estonian": "Εσθονικά",
|
||||||
|
"Filipino": "Φιλιππινέζικα",
|
||||||
|
"Finnish": "Φινλανδικά",
|
||||||
|
"French": "Γαλλικά",
|
||||||
|
"Galician": "Γαλικιακά",
|
||||||
|
"Georgian": "Γεωργιανά",
|
||||||
|
"German": "Γερμανικά",
|
||||||
|
"Greek": "Ελληνικά",
|
||||||
|
"Gujarati": "Γκουτζαρατικά",
|
||||||
|
"Haitian Creole": "Κρεόλ Αϊτής",
|
||||||
|
"Hausa": "Χάουσα",
|
||||||
|
"Hawaiian": "Χαβανέζικα",
|
||||||
|
"Hebrew": "Εβραϊκά",
|
||||||
|
"Hindi": "Χίντι",
|
||||||
|
"Hmong": "Χμονγκ",
|
||||||
|
"Hungarian": "Ουγγαρέζικα",
|
||||||
|
"Icelandic": "Ισλανδικά",
|
||||||
|
"Igbo": "Ιγκμπό",
|
||||||
|
"Indonesian": "Ινδονησιακά",
|
||||||
|
"Irish": "Ιρλανδικά",
|
||||||
|
"Italian": "Ιταλικά",
|
||||||
|
"Japanese": "Ιαπωνικά",
|
||||||
|
"Javanese": "Ιαβανέζικα",
|
||||||
|
"Kannada": "Κανάντα",
|
||||||
|
"Kazakh": "Καζακικά",
|
||||||
|
"Khmer": "Χμερ",
|
||||||
|
"Korean": "Κορεάτικα",
|
||||||
|
"Kurdish": "Κούρδικα",
|
||||||
|
"Kyrgyz": "Κιργιστανικά",
|
||||||
|
"Lao": "Lao",
|
||||||
|
"Latin": "Λατινικά",
|
||||||
|
"Latvian": "Λετονικά",
|
||||||
|
"Lithuanian": "Λιθουανικά",
|
||||||
|
"Luxembourgish": "Λουξεμβουργιανά",
|
||||||
|
"Macedonian": "Μακεδονικά",
|
||||||
|
"Malagasy": "Μαλαγασικά",
|
||||||
|
"Malay": "Μαλαισιανά",
|
||||||
|
"Malayalam": "Μαλαγιαλάμ",
|
||||||
|
"Maltese": "Μαλτέζικα",
|
||||||
|
"Maori": "Μαορί",
|
||||||
|
"Marathi": "Μαράτι",
|
||||||
|
"Mongolian": "Μογγολικά",
|
||||||
|
"Nepali": "Νεπαλικά",
|
||||||
|
"Norwegian Bokmål": "Νορβηγικά Μποκμάλ",
|
||||||
|
"Nyanja": "Νιάντζα",
|
||||||
|
"Pashto": "Αφγανικά",
|
||||||
|
"Persian": "Περσικά",
|
||||||
|
"Polish": "Πολωνικά",
|
||||||
|
"Portuguese": "Πορτογαλικά",
|
||||||
|
"Punjabi": "Παντζάμπι",
|
||||||
|
"Romanian": "Ρουμανικά",
|
||||||
|
"Russian": "Ρώσικα",
|
||||||
|
"Samoan": "Σαμόα",
|
||||||
|
"Scottish Gaelic": "Σκωτικά Γαελικά",
|
||||||
|
"Serbian": "Σέρβικα",
|
||||||
|
"Shona": "Σόνα",
|
||||||
|
"Sindhi": "Σίντι",
|
||||||
|
"Sinhala": "Σιναλεζικά",
|
||||||
|
"Slovak": "Σλοβακικά",
|
||||||
|
"Slovenian": "ΣΛοβενικά",
|
||||||
|
"Somali": "Σομαλικά",
|
||||||
|
"Southern Sotho": "Νότια Σούτου",
|
||||||
|
"Spanish": "Ισπανικά",
|
||||||
|
"Spanish (Latin America)": "Ισπανικά (Λατινική Αμερική)",
|
||||||
|
"Sundanese": "Σουντανέζικα",
|
||||||
|
"Swahili": "Σουαχίλι",
|
||||||
|
"Swedish": "Σουηδικά",
|
||||||
|
"Tajik": "Τατζικικά",
|
||||||
|
"Tamil": "Ταμίλ",
|
||||||
|
"Telugu": "Τελούγκου",
|
||||||
|
"Thai": "Ταϊλανδικά",
|
||||||
|
"Turkish": "Τούρκικα",
|
||||||
|
"Ukrainian": "Ουκρανικά",
|
||||||
|
"Urdu": "Ουρντού",
|
||||||
|
"Uzbek": "Ουζμπεκικά",
|
||||||
|
"Vietnamese": "Βιετναμέζικα",
|
||||||
|
"Welsh": "Ουαλικά",
|
||||||
|
"Western Frisian": "Δυτική Φριζική",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Γίντις",
|
||||||
|
"Yoruba": "Γιορούμπα",
|
||||||
|
"Zulu": "Ζουλού",
|
||||||
|
"`x` years": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` χρόνο",
|
||||||
|
"": "`x` χρόνια"
|
||||||
|
},
|
||||||
|
"`x` months": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` μήνα",
|
||||||
|
"": "`x` μήνες"
|
||||||
|
},
|
||||||
|
"`x` weeks": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` εβδομάδα",
|
||||||
|
"": "`x` εβδομάδες"
|
||||||
|
},
|
||||||
|
"`x` days": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` ημέρα",
|
||||||
|
"": "`x` ημέρες"
|
||||||
|
},
|
||||||
|
"`x` hours": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` ώρα",
|
||||||
|
"": "`x` ώρες"
|
||||||
|
},
|
||||||
|
"`x` minutes": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` λεπτό",
|
||||||
|
"": "`x` λεπτά"
|
||||||
|
},
|
||||||
|
"`x` seconds": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` δευτερόλεπτο",
|
||||||
|
"": "`x` δευτερόλεπτα"
|
||||||
|
},
|
||||||
|
"Fallback comments: ": "Εναλλακτικά σχόλια: ",
|
||||||
|
"Popular": "Δημοφιλή",
|
||||||
|
"Top": "Κορυφαία",
|
||||||
|
"About": "Σχετικά",
|
||||||
|
"Rating: ": "Aξιολόγηση: ",
|
||||||
|
"Language: ": "Γλώσσα: ",
|
||||||
|
"View as playlist": "Προβολή ως λίστα αναπαραγωγής",
|
||||||
|
"Default": "Προεπιλογή",
|
||||||
|
"Music": "Μουσική",
|
||||||
|
"Gaming": "Παιχνίδια",
|
||||||
|
"News": "Ειδήσεις",
|
||||||
|
"Movies": "Ταινίες",
|
||||||
|
"Download": "Λήψη",
|
||||||
|
"Download as: ": "Λήψη ως: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(τροποποιημένο)",
|
||||||
|
"YouTube comment permalink": "Σύνδεσμος YouTube σχολίου",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "Ο χρηστης `x` έβαλε ❤",
|
||||||
|
"Audio mode": "Λειτουργία ήχου",
|
||||||
|
"Video mode": "Λειτουργία βίντεο",
|
||||||
|
"Videos": "Βίντεο",
|
||||||
|
"Playlists": "Λίστες Αναπαραγωγής",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": "Τρέχουσα έκδοση: "
|
||||||
|
}
|
||||||
@@ -1,295 +1,387 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` subscribers",
|
"`x` subscribers": {
|
||||||
"`x` videos": "`x` videos",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` subscriber",
|
||||||
"LIVE": "LIVE",
|
"": "`x` subscribers"
|
||||||
"Shared `x` ago": "Shared `x` ago",
|
},
|
||||||
"Unsubscribe": "Unsubscribe",
|
"`x` videos": {
|
||||||
"Subscribe": "Subscribe",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` video",
|
||||||
"Login to subscribe to `x`": "Login to subscribe to `x`",
|
"": "`x` videos"
|
||||||
"View channel on YouTube": "View channel on YouTube",
|
},
|
||||||
"newest": "newest",
|
"`x` playlists": {
|
||||||
"oldest": "oldest",
|
"(\\D|^)1(\\D|$)": "`x` playlist",
|
||||||
"popular": "popular",
|
"": "`x` playlists"
|
||||||
"last": "last",
|
},
|
||||||
"Next page": "Next page",
|
"LIVE": "LIVE",
|
||||||
"Previous page": "Previous page",
|
"Shared `x` ago": "Shared `x` ago",
|
||||||
"Clear watch history?": "Clear watch history?",
|
"Unsubscribe": "Unsubscribe",
|
||||||
"Yes": "Yes",
|
"Subscribe": "Subscribe",
|
||||||
"No": "No",
|
"View channel on YouTube": "View channel on YouTube",
|
||||||
"Import and Export Data": "Import and Export Data",
|
"View playlist on YouTube": "View playlist on YouTube",
|
||||||
"Import": "Import",
|
"newest": "newest",
|
||||||
"Import Invidious data": "Import Invidious data",
|
"oldest": "oldest",
|
||||||
"Import YouTube subscriptions": "Import YouTube subscriptions",
|
"popular": "popular",
|
||||||
"Import FreeTube subscriptions (.db)": "Import FreeTube subscriptions (.db)",
|
"last": "last",
|
||||||
"Import NewPipe subscriptions (.json)": "Import NewPipe subscriptions (.json)",
|
"Next page": "Next page",
|
||||||
"Import NewPipe data (.zip)": "Import NewPipe data (.zip)",
|
"Previous page": "Previous page",
|
||||||
"Export": "Export",
|
"Clear watch history?": "Clear watch history?",
|
||||||
"Export subscriptions as OPML": "Export subscriptions as OPML",
|
"New password": "New password",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Export subscriptions as OPML (for NewPipe & FreeTube)",
|
"New passwords must match": "New passwords must match",
|
||||||
"Export data as JSON": "Export data as JSON",
|
"Cannot change password for Google accounts": "Cannot change password for Google accounts",
|
||||||
"Delete account?": "Delete account?",
|
"Authorize token?": "Authorize token?",
|
||||||
"History": "History",
|
"Authorize token for `x`?": "Authorize token for `x`?",
|
||||||
"An alternative front-end to YouTube": "An alternative front-end to YouTube",
|
"Yes": "Yes",
|
||||||
"JavaScript license information": "JavaScript license information",
|
"No": "No",
|
||||||
"source": "source",
|
"Import and Export Data": "Import and Export Data",
|
||||||
"Login": "Login",
|
"Import": "Import",
|
||||||
"Login/Register": "Login/Register",
|
"Import Invidious data": "Import Invidious data",
|
||||||
"Login to Google": "Login to Google",
|
"Import YouTube subscriptions": "Import YouTube subscriptions",
|
||||||
"User ID:": "User ID:",
|
"Import FreeTube subscriptions (.db)": "Import FreeTube subscriptions (.db)",
|
||||||
"Password:": "Password:",
|
"Import NewPipe subscriptions (.json)": "Import NewPipe subscriptions (.json)",
|
||||||
"Time (h:mm:ss):": "Time (h:mm:ss):",
|
"Import NewPipe data (.zip)": "Import NewPipe data (.zip)",
|
||||||
"Text CAPTCHA": "Text CAPTCHA",
|
"Export": "Export",
|
||||||
"Image CAPTCHA": "Image CAPTCHA",
|
"Export subscriptions as OPML": "Export subscriptions as OPML",
|
||||||
"Sign In": "Sign In",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Export subscriptions as OPML (for NewPipe & FreeTube)",
|
||||||
"Register": "Register",
|
"Export data as JSON": "Export data as JSON",
|
||||||
"Email:": "Email:",
|
"Delete account?": "Delete account?",
|
||||||
"Google verification code:": "Google verification code:",
|
"History": "History",
|
||||||
"Preferences": "Preferences",
|
"An alternative front-end to YouTube": "An alternative front-end to YouTube",
|
||||||
"Player preferences": "Player preferences",
|
"JavaScript license information": "JavaScript license information",
|
||||||
"Always loop: ": "Always loop: ",
|
"source": "source",
|
||||||
"Autoplay: ": "Autoplay: ",
|
"Log in": "Log in",
|
||||||
"Autoplay next video: ": "Autoplay next video: ",
|
"Log in/register": "Log in/register",
|
||||||
"Listen by default: ": "Listen by default: ",
|
"Log in with Google": "Log in with Google",
|
||||||
"Proxy videos? ": "Proxy videos? ",
|
"User ID": "User ID",
|
||||||
"Default speed: ": "Default speed: ",
|
"Password": "Password",
|
||||||
"Preferred video quality: ": "Preferred video quality: ",
|
"Time (h:mm:ss):": "Time (h:mm:ss):",
|
||||||
"Player volume: ": "Player volume: ",
|
"Text CAPTCHA": "Text CAPTCHA",
|
||||||
"Default comments: ": "Default comments: ",
|
"Image CAPTCHA": "Image CAPTCHA",
|
||||||
"Default captions: ": "Default captions: ",
|
"Sign In": "Sign In",
|
||||||
"Fallback captions: ": "Fallback captions: ",
|
"Register": "Register",
|
||||||
"Show related videos? ": "Show related videos? ",
|
"E-mail": "E-mail",
|
||||||
"Visual preferences": "Visual preferences",
|
"Google verification code": "Google verification code",
|
||||||
"Dark mode: ": "Dark mode: ",
|
"Preferences": "Preferences",
|
||||||
"Thin mode: ": "Thin mode: ",
|
"Player preferences": "Player preferences",
|
||||||
"Subscription preferences": "Subscription preferences",
|
"Always loop: ": "Always loop: ",
|
||||||
"Redirect homepage to feed: ": "Redirect homepage to feed: ",
|
"Autoplay: ": "Autoplay: ",
|
||||||
"Number of videos shown in feed: ": "Number of videos shown in feed: ",
|
"Play next by default: ": "Play next by default: ",
|
||||||
"Sort videos by: ": "Sort videos by: ",
|
"Autoplay next video: ": "Autoplay next video: ",
|
||||||
"published": "published",
|
"Listen by default: ": "Listen by default: ",
|
||||||
"published - reverse": "published - reverse",
|
"Proxy videos: ": "Proxy videos: ",
|
||||||
"alphabetically": "alphabetically",
|
"Default speed: ": "Default speed: ",
|
||||||
"alphabetically - reverse": "alphabetically - reverse",
|
"Preferred video quality: ": "Preferred video quality: ",
|
||||||
"channel name": "channel name",
|
"Player volume: ": "Player volume: ",
|
||||||
"channel name - reverse": "channel name - reverse",
|
"Default comments: ": "Default comments: ",
|
||||||
"Only show latest video from channel: ": "Only show latest video from channel: ",
|
"youtube": "youtube",
|
||||||
"Only show latest unwatched video from channel: ": "Only show latest unwatched video from channel: ",
|
"reddit": "reddit",
|
||||||
"Only show unwatched: ": "Only show unwatched: ",
|
"Default captions: ": "Default captions: ",
|
||||||
"Only show notifications (if there are any): ": "Only show notifications (if there are any): ",
|
"Fallback captions: ": "Fallback captions: ",
|
||||||
"Data preferences": "Data preferences",
|
"Show related videos: ": "Show related videos: ",
|
||||||
"Clear watch history": "Clear watch history",
|
"Show annotations by default: ": "Show annotations by default: ",
|
||||||
"Import/Export data": "Import/Export data",
|
"Visual preferences": "Visual preferences",
|
||||||
"Manage subscriptions": "Manage subscriptions",
|
"Player style: ": "Player style: ",
|
||||||
"Watch history": "Watch history",
|
"Dark mode: ": "Dark mode: ",
|
||||||
"Delete account": "Delete account",
|
"Theme: ": "Theme: ",
|
||||||
"Administrator preferences": "Administrator preferences",
|
"dark": "dark",
|
||||||
"Default homepage: ": "Default homepage: ",
|
"light": "light",
|
||||||
"Feed menu: ": "Feed menu: ",
|
"Thin mode: ": "Thin mode: ",
|
||||||
"Top enabled? ": "Top enabled? ",
|
"Subscription preferences": "Subscription preferences",
|
||||||
"CAPTCHA enabled? ": "CAPTCHA enabled? ",
|
"Show annotations by default for subscribed channels: ": "Show annotations by default for subscribed channels? ",
|
||||||
"Login enabled? ": "Login enabled? ",
|
"Redirect homepage to feed: ": "Redirect homepage to feed: ",
|
||||||
"Registration enabled? ": "Registration enabled? ",
|
"Number of videos shown in feed: ": "Number of videos shown in feed: ",
|
||||||
"Report statistics? ": "Report statistics? ",
|
"Sort videos by: ": "Sort videos by: ",
|
||||||
"Save preferences": "Save preferences",
|
"published": "published",
|
||||||
"Subscription manager": "Subscription manager",
|
"published - reverse": "published - reverse",
|
||||||
"`x` subscriptions": "`x` subscriptions",
|
"alphabetically": "alphabetically",
|
||||||
"Import/Export": "Import/Export",
|
"alphabetically - reverse": "alphabetically - reverse",
|
||||||
"unsubscribe": "unsubscribe",
|
"channel name": "channel name",
|
||||||
"Subscriptions": "Subscriptions",
|
"channel name - reverse": "channel name - reverse",
|
||||||
"`x` unseen notifications": "`x` unseen notifications",
|
"Only show latest video from channel: ": "Only show latest video from channel: ",
|
||||||
"search": "search",
|
"Only show latest unwatched video from channel: ": "Only show latest unwatched video from channel: ",
|
||||||
"Sign out": "Sign out",
|
"Only show unwatched: ": "Only show unwatched: ",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Released under the AGPLv3 by Omar Roth.",
|
"Only show notifications (if there are any): ": "Only show notifications (if there are any): ",
|
||||||
"Source available here.": "Source available here.",
|
"Enable web notifications": "Enable web notifications",
|
||||||
"View JavaScript license information.": "View JavaScript license information.",
|
"`x` uploaded a video": "`x` uploaded a video",
|
||||||
"View privacy policy.": "View privacy policy.",
|
"`x` is live": "`x` is live",
|
||||||
"Trending": "Trending",
|
"Data preferences": "Data preferences",
|
||||||
"Unlisted": "",
|
"Clear watch history": "Clear watch history",
|
||||||
"Watch video on Youtube": "Watch video on Youtube",
|
"Import/export data": "Import/export data",
|
||||||
"Genre: ": "Genre: ",
|
"Change password": "Change password",
|
||||||
"License: ": "License: ",
|
"Manage subscriptions": "Manage subscriptions",
|
||||||
"Family friendly? ": "Family friendly? ",
|
"Manage tokens": "Manage tokens",
|
||||||
"Wilson score: ": "Wilson score: ",
|
"Watch history": "Watch history",
|
||||||
"Engagement: ": "Engagement: ",
|
"Delete account": "Delete account",
|
||||||
"Whitelisted regions: ": "Whitelisted regions: ",
|
"Administrator preferences": "Administrator preferences",
|
||||||
"Blacklisted regions: ": "Blacklisted regions: ",
|
"Default homepage: ": "Default homepage: ",
|
||||||
"Shared `x`": "Shared `x`",
|
"Feed menu: ": "Feed menu: ",
|
||||||
"Premieres in `x`": "",
|
"Top enabled: ": "Top enabled: ",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.",
|
"CAPTCHA enabled: ": "CAPTCHA enabled: ",
|
||||||
"View YouTube comments": "View YouTube comments",
|
"Login enabled: ": "Login enabled: ",
|
||||||
"View more comments on Reddit": "View more comments on Reddit",
|
"Registration enabled: ": "Registration enabled: ",
|
||||||
"View `x` comments": "View `x` comments",
|
"Report statistics: ": "Report statistics: ",
|
||||||
"View Reddit comments": "View Reddit comments",
|
"Save preferences": "Save preferences",
|
||||||
"Hide replies": "Hide replies",
|
"Subscription manager": "Subscription manager",
|
||||||
"Show replies": "Show replies",
|
"Token manager": "Token manager",
|
||||||
"Incorrect password": "Incorrect password",
|
"Token": "Token",
|
||||||
"Quota exceeded, try again in a few hours": "Quota exceeded, try again in a few hours",
|
"`x` subscriptions": {
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` subscription",
|
||||||
"Invalid TFA code": "Invalid TFA code",
|
"": "`x` subscriptions"
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Login failed. This may be because two-factor authentication is not enabled on your account.",
|
},
|
||||||
"Invalid answer": "Invalid answer",
|
"`x` tokens": {
|
||||||
"Invalid CAPTCHA": "Invalid CAPTCHA",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` token",
|
||||||
"CAPTCHA is a required field": "CAPTCHA is a required field",
|
"": "`x` tokens"
|
||||||
"User ID is a required field": "User ID is a required field",
|
},
|
||||||
"Password is a required field": "Password is a required field",
|
"Import/export": "Import/export",
|
||||||
"Invalid username or password": "Invalid username or password",
|
"unsubscribe": "unsubscribe",
|
||||||
"Please sign in using 'Sign in with Google'": "Please sign in using 'Sign in with Google'",
|
"revoke": "revoke",
|
||||||
"Password cannot be empty": "Password cannot be empty",
|
"Subscriptions": "Subscriptions",
|
||||||
"Password cannot be longer than 55 characters": "Password cannot be longer than 55 characters",
|
"`x` unseen notifications": {
|
||||||
"Please sign in": "Please sign in",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` unseen notification",
|
||||||
"Invidious Private Feed for `x`": "Invidious Private Feed for `x`",
|
"": "`x` unseen notifications"
|
||||||
"channel:`x`": "channel:`x`",
|
},
|
||||||
"Deleted or invalid channel": "Deleted or invalid channel",
|
"search": "search",
|
||||||
"This channel does not exist.": "This channel does not exist.",
|
"Log out": "Log out",
|
||||||
"Could not get channel info.": "Could not get channel info.",
|
"Released under the AGPLv3 by Omar Roth.": "Released under the AGPLv3 by Omar Roth.",
|
||||||
"Could not fetch comments": "Could not fetch comments",
|
"Source available here.": "Source available here.",
|
||||||
"View `x` replies": "View `x` replies",
|
"View JavaScript license information.": "View JavaScript license information.",
|
||||||
"`x` ago": "`x` ago",
|
"View privacy policy.": "View privacy policy.",
|
||||||
"Load more": "Load more",
|
"Trending": "Trending",
|
||||||
"`x` points": "`x` points",
|
"Public": "Public",
|
||||||
"Could not create mix.": "Could not create mix.",
|
"Unlisted": "Unlisted",
|
||||||
"Playlist is empty": "Playlist is empty",
|
"Private": "Private",
|
||||||
"Invalid playlist.": "Invalid playlist.",
|
"View all playlists": "View all playlists",
|
||||||
"Playlist does not exist.": "Playlist does not exist.",
|
"Updated `x` ago": "Updated `x` ago",
|
||||||
"Could not pull trending pages.": "Could not pull trending pages.",
|
"Delete playlist `x`?": "Delete playlist `x`?",
|
||||||
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
|
"Delete playlist": "Delete playlist",
|
||||||
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
|
"Create playlist": "Create playlist",
|
||||||
"Invalid challenge": "Invalid challenge",
|
"Title": "Title",
|
||||||
"Invalid token": "Invalid token",
|
"Playlist privacy": "Playlist privacy",
|
||||||
"Invalid user": "Invalid user",
|
"Editing playlist `x`": "Editing playlist `x`",
|
||||||
"Token is expired, please try again": "Token is expired, please try again",
|
"Watch on YouTube": "Watch on YouTube",
|
||||||
"English": "English",
|
"Hide annotations": "Hide annotations",
|
||||||
"English (auto-generated)": "English (auto-generated)",
|
"Show annotations": "Show annotations",
|
||||||
"Afrikaans": "Afrikaans",
|
"Genre: ": "Genre: ",
|
||||||
"Albanian": "Albanian",
|
"License: ": "License: ",
|
||||||
"Amharic": "Amharic",
|
"Family friendly? ": "Family friendly? ",
|
||||||
"Arabic": "Arabic",
|
"Wilson score: ": "Wilson score: ",
|
||||||
"Armenian": "Armenian",
|
"Engagement: ": "Engagement: ",
|
||||||
"Azerbaijani": "Azerbaijani",
|
"Whitelisted regions: ": "Whitelisted regions: ",
|
||||||
"Bangla": "Bangla",
|
"Blacklisted regions: ": "Blacklisted regions: ",
|
||||||
"Basque": "Basque",
|
"Shared `x`": "Shared `x`",
|
||||||
"Belarusian": "Belarusian",
|
"`x` views": {
|
||||||
"Bosnian": "Bosnian",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` view",
|
||||||
"Bulgarian": "Bulgarian",
|
"": "`x` views"
|
||||||
"Burmese": "Burmese",
|
},
|
||||||
"Catalan": "Catalan",
|
"Premieres in `x`": "Premieres in `x`",
|
||||||
"Cebuano": "Cebuano",
|
"Premieres `x`": "Premieres `x`",
|
||||||
"Chinese (Simplified)": "Chinese (Simplified)",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.",
|
||||||
"Chinese (Traditional)": "Chinese (Traditional)",
|
"View YouTube comments": "View YouTube comments",
|
||||||
"Corsican": "Corsican",
|
"View more comments on Reddit": "View more comments on Reddit",
|
||||||
"Croatian": "Croatian",
|
"View `x` comments": {
|
||||||
"Czech": "Czech",
|
"(\\D|^)1(\\D|$)": "View `x` comment",
|
||||||
"Danish": "Danish",
|
"": "View `x` comments"
|
||||||
"Dutch": "Dutch",
|
},
|
||||||
"Esperanto": "Esperanto",
|
"View Reddit comments": "View Reddit comments",
|
||||||
"Estonian": "Estonian",
|
"Hide replies": "Hide replies",
|
||||||
"Filipino": "Filipino",
|
"Show replies": "Show replies",
|
||||||
"Finnish": "Finnish",
|
"Incorrect password": "Incorrect password",
|
||||||
"French": "French",
|
"Quota exceeded, try again in a few hours": "Quota exceeded, try again in a few hours",
|
||||||
"Galician": "Galician",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.",
|
||||||
"Georgian": "Georgian",
|
"Invalid TFA code": "Invalid TFA code",
|
||||||
"German": "German",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Login failed. This may be because two-factor authentication is not turned on for your account.",
|
||||||
"Greek": "Greek",
|
"Wrong answer": "Wrong answer",
|
||||||
"Gujarati": "Gujarati",
|
"Erroneous CAPTCHA": "Erroneous CAPTCHA",
|
||||||
"Haitian Creole": "Haitian Creole",
|
"CAPTCHA is a required field": "CAPTCHA is a required field",
|
||||||
"Hausa": "Hausa",
|
"User ID is a required field": "User ID is a required field",
|
||||||
"Hawaiian": "Hawaiian",
|
"Password is a required field": "Password is a required field",
|
||||||
"Hebrew": "Hebrew",
|
"Wrong username or password": "Wrong username or password",
|
||||||
"Hindi": "Hindi",
|
"Please sign in using 'Log in with Google'": "Please sign in using 'Log in with Google'",
|
||||||
"Hmong": "Hmong",
|
"Password cannot be empty": "Password cannot be empty",
|
||||||
"Hungarian": "Hungarian",
|
"Password cannot be longer than 55 characters": "Password cannot be longer than 55 characters",
|
||||||
"Icelandic": "Icelandic",
|
"Please log in": "Please log in",
|
||||||
"Igbo": "Igbo",
|
"Invidious Private Feed for `x`": "Invidious Private Feed for `x`",
|
||||||
"Indonesian": "Indonesian",
|
"channel:`x`": "channel:`x`",
|
||||||
"Irish": "Irish",
|
"Deleted or invalid channel": "Deleted or invalid channel",
|
||||||
"Italian": "Italian",
|
"This channel does not exist.": "This channel does not exist.",
|
||||||
"Japanese": "Japanese",
|
"Could not get channel info.": "Could not get channel info.",
|
||||||
"Javanese": "Javanese",
|
"Could not fetch comments": "Could not fetch comments",
|
||||||
"Kannada": "Kannada",
|
"View `x` replies": {
|
||||||
"Kazakh": "Kazakh",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "View `x` reply",
|
||||||
"Khmer": "Khmer",
|
"": "View `x` replies"
|
||||||
"Korean": "Korean",
|
},
|
||||||
"Kurdish": "Kurdish",
|
"`x` ago": "`x` ago",
|
||||||
"Kyrgyz": "Kyrgyz",
|
"Load more": "Load more",
|
||||||
"Lao": "Lao",
|
"`x` points": {
|
||||||
"Latin": "Latin",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` point",
|
||||||
"Latvian": "Latvian",
|
"": "`x` points"
|
||||||
"Lithuanian": "Lithuanian",
|
},
|
||||||
"Luxembourgish": "Luxembourgish",
|
"Could not create mix.": "Could not create mix.",
|
||||||
"Macedonian": "Macedonian",
|
"Empty playlist": "Empty playlist",
|
||||||
"Malagasy": "Malagasy",
|
"Not a playlist.": "Not a playlist.",
|
||||||
"Malay": "Malay",
|
"Playlist does not exist.": "Playlist does not exist.",
|
||||||
"Malayalam": "Malayalam",
|
"Could not pull trending pages.": "Could not pull trending pages.",
|
||||||
"Maltese": "Maltese",
|
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
|
||||||
"Maori": "Maori",
|
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
|
||||||
"Marathi": "Marathi",
|
"Erroneous challenge": "Erroneous challenge",
|
||||||
"Mongolian": "Mongolian",
|
"Erroneous token": "Erroneous token",
|
||||||
"Nepali": "Nepali",
|
"No such user": "No such user",
|
||||||
"Norwegian": "Norwegian",
|
"Token is expired, please try again": "Token is expired, please try again",
|
||||||
"Nyanja": "Nyanja",
|
"English": "English",
|
||||||
"Pashto": "Pashto",
|
"English (auto-generated)": "English (auto-generated)",
|
||||||
"Persian": "Persian",
|
"Afrikaans": "Afrikaans",
|
||||||
"Polish": "Polish",
|
"Albanian": "Albanian",
|
||||||
"Portuguese": "Portuguese",
|
"Amharic": "Amharic",
|
||||||
"Punjabi": "Punjabi",
|
"Arabic": "Arabic",
|
||||||
"Romanian": "Romanian",
|
"Armenian": "Armenian",
|
||||||
"Russian": "Russian",
|
"Azerbaijani": "Azerbaijani",
|
||||||
"Samoan": "Samoan",
|
"Bangla": "Bangla",
|
||||||
"Scottish Gaelic": "Scottish Gaelic",
|
"Basque": "Basque",
|
||||||
"Serbian": "Serbian",
|
"Belarusian": "Belarusian",
|
||||||
"Shona": "Shona",
|
"Bosnian": "Bosnian",
|
||||||
"Sindhi": "Sindhi",
|
"Bulgarian": "Bulgarian",
|
||||||
"Sinhala": "Sinhala",
|
"Burmese": "Burmese",
|
||||||
"Slovak": "Slovak",
|
"Catalan": "Catalan",
|
||||||
"Slovenian": "Slovenian",
|
"Cebuano": "Cebuano",
|
||||||
"Somali": "Somali",
|
"Chinese (Simplified)": "Chinese (Simplified)",
|
||||||
"Southern Sotho": "Southern Sotho",
|
"Chinese (Traditional)": "Chinese (Traditional)",
|
||||||
"Spanish": "Spanish",
|
"Corsican": "Corsican",
|
||||||
"Spanish (Latin America)": "Spanish (Latin America)",
|
"Croatian": "Croatian",
|
||||||
"Sundanese": "Sundanese",
|
"Czech": "Czech",
|
||||||
"Swahili": "Swahili",
|
"Danish": "Danish",
|
||||||
"Swedish": "Swedish",
|
"Dutch": "Dutch",
|
||||||
"Tajik": "Tajik",
|
"Esperanto": "Esperanto",
|
||||||
"Tamil": "Tamil",
|
"Estonian": "Estonian",
|
||||||
"Telugu": "Telugu",
|
"Filipino": "Filipino",
|
||||||
"Thai": "Thai",
|
"Finnish": "Finnish",
|
||||||
"Turkish": "Turkish",
|
"French": "French",
|
||||||
"Ukrainian": "Ukrainian",
|
"Galician": "Galician",
|
||||||
"Urdu": "Urdu",
|
"Georgian": "Georgian",
|
||||||
"Uzbek": "Uzbek",
|
"German": "German",
|
||||||
"Vietnamese": "Vietnamese",
|
"Greek": "Greek",
|
||||||
"Welsh": "Welsh",
|
"Gujarati": "Gujarati",
|
||||||
"Western Frisian": "Western Frisian",
|
"Haitian Creole": "Haitian Creole",
|
||||||
"Xhosa": "Xhosa",
|
"Hausa": "Hausa",
|
||||||
"Yiddish": "Yiddish",
|
"Hawaiian": "Hawaiian",
|
||||||
"Yoruba": "Yoruba",
|
"Hebrew": "Hebrew",
|
||||||
"Zulu": "Zulu",
|
"Hindi": "Hindi",
|
||||||
"`x` years": "`x` years",
|
"Hmong": "Hmong",
|
||||||
"`x` months": "`x` months",
|
"Hungarian": "Hungarian",
|
||||||
"`x` weeks": "`x` weeks",
|
"Icelandic": "Icelandic",
|
||||||
"`x` days": "`x` days",
|
"Igbo": "Igbo",
|
||||||
"`x` hours": "`x` hours",
|
"Indonesian": "Indonesian",
|
||||||
"`x` minutes": "`x` minutes",
|
"Irish": "Irish",
|
||||||
"`x` seconds": "`x` seconds",
|
"Italian": "Italian",
|
||||||
"Fallback comments: ": "Fallback comments: ",
|
"Japanese": "Japanese",
|
||||||
"Popular": "Popular",
|
"Javanese": "Javanese",
|
||||||
"Top": "Top",
|
"Kannada": "Kannada",
|
||||||
"About": "About",
|
"Kazakh": "Kazakh",
|
||||||
"Rating: ": "Rating: ",
|
"Khmer": "Khmer",
|
||||||
"Language: ": "Language: ",
|
"Korean": "Korean",
|
||||||
"Default": "Default",
|
"Kurdish": "Kurdish",
|
||||||
"Music": "Music",
|
"Kyrgyz": "Kyrgyz",
|
||||||
"Gaming": "Gaming",
|
"Lao": "Lao",
|
||||||
"News": "News",
|
"Latin": "Latin",
|
||||||
"Movies": "Movies",
|
"Latvian": "Latvian",
|
||||||
"Download": "Download",
|
"Lithuanian": "Lithuanian",
|
||||||
"Download as: ": "Download as: ",
|
"Luxembourgish": "Luxembourgish",
|
||||||
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
"Macedonian": "Macedonian",
|
||||||
"(edited)": "(edited)",
|
"Malagasy": "Malagasy",
|
||||||
"Youtube permalink of the comment": "Youtube permalink of the comment",
|
"Malay": "Malay",
|
||||||
"`x` marked it with a ❤": "`x` marked it with a ❤",
|
"Malayalam": "Malayalam",
|
||||||
"Audio mode": "Audio mode",
|
"Maltese": "Maltese",
|
||||||
"Video mode": "Video mode",
|
"Maori": "Maori",
|
||||||
"Videos": "Videos",
|
"Marathi": "Marathi",
|
||||||
"Playlists": "Playlists",
|
"Mongolian": "Mongolian",
|
||||||
"Current version: ": "Current version: "
|
"Nepali": "Nepali",
|
||||||
}
|
"Norwegian Bokmål": "Norwegian Bokmål",
|
||||||
|
"Nyanja": "Nyanja",
|
||||||
|
"Pashto": "Pashto",
|
||||||
|
"Persian": "Persian",
|
||||||
|
"Polish": "Polish",
|
||||||
|
"Portuguese": "Portuguese",
|
||||||
|
"Punjabi": "Punjabi",
|
||||||
|
"Romanian": "Romanian",
|
||||||
|
"Russian": "Russian",
|
||||||
|
"Samoan": "Samoan",
|
||||||
|
"Scottish Gaelic": "Scottish Gaelic",
|
||||||
|
"Serbian": "Serbian",
|
||||||
|
"Shona": "Shona",
|
||||||
|
"Sindhi": "Sindhi",
|
||||||
|
"Sinhala": "Sinhala",
|
||||||
|
"Slovak": "Slovak",
|
||||||
|
"Slovenian": "Slovenian",
|
||||||
|
"Somali": "Somali",
|
||||||
|
"Southern Sotho": "Southern Sotho",
|
||||||
|
"Spanish": "Spanish",
|
||||||
|
"Spanish (Latin America)": "Spanish (Latin America)",
|
||||||
|
"Sundanese": "Sundanese",
|
||||||
|
"Swahili": "Swahili",
|
||||||
|
"Swedish": "Swedish",
|
||||||
|
"Tajik": "Tajik",
|
||||||
|
"Tamil": "Tamil",
|
||||||
|
"Telugu": "Telugu",
|
||||||
|
"Thai": "Thai",
|
||||||
|
"Turkish": "Turkish",
|
||||||
|
"Ukrainian": "Ukrainian",
|
||||||
|
"Urdu": "Urdu",
|
||||||
|
"Uzbek": "Uzbek",
|
||||||
|
"Vietnamese": "Vietnamese",
|
||||||
|
"Welsh": "Welsh",
|
||||||
|
"Western Frisian": "Western Frisian",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Yiddish",
|
||||||
|
"Yoruba": "Yoruba",
|
||||||
|
"Zulu": "Zulu",
|
||||||
|
"`x` years": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` year",
|
||||||
|
"": "`x` years"
|
||||||
|
},
|
||||||
|
"`x` months": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` month",
|
||||||
|
"": "`x` months"
|
||||||
|
},
|
||||||
|
"`x` weeks": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` week",
|
||||||
|
"": "`x` weeks"
|
||||||
|
},
|
||||||
|
"`x` days": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` day",
|
||||||
|
"": "`x` days"
|
||||||
|
},
|
||||||
|
"`x` hours": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` hour",
|
||||||
|
"": "`x` hours"
|
||||||
|
},
|
||||||
|
"`x` minutes": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` minute",
|
||||||
|
"": "`x` minutes"
|
||||||
|
},
|
||||||
|
"`x` seconds": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` second",
|
||||||
|
"": "`x` seconds"
|
||||||
|
},
|
||||||
|
"Fallback comments: ": "Fallback comments: ",
|
||||||
|
"Popular": "Popular",
|
||||||
|
"Top": "Top",
|
||||||
|
"About": "About",
|
||||||
|
"Rating: ": "Rating: ",
|
||||||
|
"Language: ": "Language: ",
|
||||||
|
"View as playlist": "View as playlist",
|
||||||
|
"Default": "Default",
|
||||||
|
"Music": "Music",
|
||||||
|
"Gaming": "Gaming",
|
||||||
|
"News": "News",
|
||||||
|
"Movies": "Movies",
|
||||||
|
"Download": "Download",
|
||||||
|
"Download as: ": "Download as: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(edited)",
|
||||||
|
"YouTube comment permalink": "YouTube comment permalink",
|
||||||
|
"permalink": "permalink",
|
||||||
|
"`x` marked it with a ❤": "`x` marked it with a ❤",
|
||||||
|
"Audio mode": "Audio mode",
|
||||||
|
"Video mode": "Video mode",
|
||||||
|
"Videos": "Videos",
|
||||||
|
"Playlists": "Playlists",
|
||||||
|
"Community": "Community",
|
||||||
|
"Current version: ": "Current version: "
|
||||||
|
}
|
||||||
336
locales/eo.json
Normal file
336
locales/eo.json
Normal file
@@ -0,0 +1,336 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": "`x` abonantoj",
|
||||||
|
"`x` videos": "`x` videoj",
|
||||||
|
"`x` playlists": "",
|
||||||
|
"LIVE": "NUNA",
|
||||||
|
"Shared `x` ago": "Konigita antaŭ `x`",
|
||||||
|
"Unsubscribe": "Malaboni",
|
||||||
|
"Subscribe": "Aboni",
|
||||||
|
"View channel on YouTube": "Vidi kanalon en YouTube",
|
||||||
|
"View playlist on YouTube": "Vidi ludliston en YouTube",
|
||||||
|
"newest": "pli novaj",
|
||||||
|
"oldest": "pli malnovaj",
|
||||||
|
"popular": "popularaj",
|
||||||
|
"last": "lasta",
|
||||||
|
"Next page": "Sekva paĝo",
|
||||||
|
"Previous page": "Antaŭa paĝo",
|
||||||
|
"Clear watch history?": "Ĉu forigi vidohistorion?",
|
||||||
|
"New password": "Nova pasvorto",
|
||||||
|
"New passwords must match": "Novaj pasvortoj devas kongrui",
|
||||||
|
"Cannot change password for Google accounts": "Ne eblas ŝanĝi pasvorton por kontoj de Google",
|
||||||
|
"Authorize token?": "Ĉu rajtigi ĵetonon?",
|
||||||
|
"Authorize token for `x`?": "Ĉu rajtigi ĵetonon por `x`?",
|
||||||
|
"Yes": "Jes",
|
||||||
|
"No": "Ne",
|
||||||
|
"Import and Export Data": "Importi kaj Eksporti Datumojn",
|
||||||
|
"Import": "Importi",
|
||||||
|
"Import Invidious data": "Importi datumojn de Invidious",
|
||||||
|
"Import YouTube subscriptions": "Importi abonojn de YouTube",
|
||||||
|
"Import FreeTube subscriptions (.db)": "Importi abonojn de FreeTube (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "Importi abonojn de NewPipe (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "Importi datumojn de NewPipe (.zip)",
|
||||||
|
"Export": "Eksporti",
|
||||||
|
"Export subscriptions as OPML": "Eksporti abonojn kiel OPML",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksporti abonojn kiel OPML (por NewPipe kaj FreeTube)",
|
||||||
|
"Export data as JSON": "Eksporti datumojn kiel JSON",
|
||||||
|
"Delete account?": "Ĉu forigi konton?",
|
||||||
|
"History": "Historio",
|
||||||
|
"An alternative front-end to YouTube": "Alternativa fasado al YouTube",
|
||||||
|
"JavaScript license information": "Ĝavoskripta licenca informo",
|
||||||
|
"source": "fonto",
|
||||||
|
"Log in": "Ensaluti",
|
||||||
|
"Log in/register": "Ensaluti/Registriĝi",
|
||||||
|
"Log in with Google": "Ensaluti al Google",
|
||||||
|
"User ID": "Uzula identigilo",
|
||||||
|
"Password": "Pasvorto",
|
||||||
|
"Time (h:mm:ss):": "Horo (h:mm:ss):",
|
||||||
|
"Text CAPTCHA": "Teksta CAPTCHA",
|
||||||
|
"Image CAPTCHA": "Bilda CAPTCHA",
|
||||||
|
"Sign In": "Ensaluti",
|
||||||
|
"Register": "Registriĝi",
|
||||||
|
"E-mail": "Retpoŝto",
|
||||||
|
"Google verification code": "Kontrolkodo de Google",
|
||||||
|
"Preferences": "Agordoj",
|
||||||
|
"Player preferences": "Spektilaj agordoj",
|
||||||
|
"Always loop: ": "Ĉiam ripeti: ",
|
||||||
|
"Autoplay: ": "Aŭtomate ludi: ",
|
||||||
|
"Play next by default: ": "Ludi sekvan defaŭlte: ",
|
||||||
|
"Autoplay next video: ": "Aŭtomate ludi sekvan videon: ",
|
||||||
|
"Listen by default: ": "Aŭskulti defaŭlte: ",
|
||||||
|
"Proxy videos: ": "Ĉu uzi prokuran servilon por videoj? ",
|
||||||
|
"Default speed: ": "Defaŭlta rapido: ",
|
||||||
|
"Preferred video quality: ": "Preferita videkvalito: ",
|
||||||
|
"Player volume: ": "Ludila sonforteco: ",
|
||||||
|
"Default comments: ": "Defaŭltaj komentoj: ",
|
||||||
|
"youtube": "youtube",
|
||||||
|
"reddit": "reddit",
|
||||||
|
"Default captions: ": "Defaŭltaj subtekstoj: ",
|
||||||
|
"Fallback captions: ": "Retrodefaŭltaj subtekstoj: ",
|
||||||
|
"Show related videos: ": "Ĉu montri rilatajn videojn? ",
|
||||||
|
"Show annotations by default: ": "Ĉu montri prinotojn defaŭlte? ",
|
||||||
|
"Visual preferences": "Vidaj preferoj",
|
||||||
|
"Player style: ": "Ludila stilo: ",
|
||||||
|
"Dark mode: ": "Malhela reĝimo: ",
|
||||||
|
"Theme: ": "Etoso: ",
|
||||||
|
"dark": "malhela",
|
||||||
|
"light": "hela",
|
||||||
|
"Thin mode: ": "Maldika reĝimo: ",
|
||||||
|
"Subscription preferences": "Abonaj agordoj",
|
||||||
|
"Show annotations by default for subscribed channels: ": "Ĉu montri prinotojn defaŭlte por abonitaj kanaloj? ",
|
||||||
|
"Redirect homepage to feed: ": "Alidirekti hejmpâgon al fluo: ",
|
||||||
|
"Number of videos shown in feed: ": "Nombro da videoj montritaj en fluo: ",
|
||||||
|
"Sort videos by: ": "Ordi videojn laŭ: ",
|
||||||
|
"published": "publikigo",
|
||||||
|
"published - reverse": "publitigo - renverse",
|
||||||
|
"alphabetically": "alfabete",
|
||||||
|
"alphabetically - reverse": "alfabete - renverse",
|
||||||
|
"channel name": "kanala nombro",
|
||||||
|
"channel name - reverse": "kanala nombro - renverse",
|
||||||
|
"Only show latest video from channel: ": "Nur montri pli novan videon el kanalo: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "Nur montri pli novan malviditan videon el kanalo: ",
|
||||||
|
"Only show unwatched: ": "Nur montri malviditajn: ",
|
||||||
|
"Only show notifications (if there are any): ": "Nur montri sciigojn (se estas): ",
|
||||||
|
"Enable web notifications": "Ebligi retejajn sciigojn",
|
||||||
|
"`x` uploaded a video": "`x` alŝutis videon",
|
||||||
|
"`x` is live": "`x` estas nuna",
|
||||||
|
"Data preferences": "Datumagordoj",
|
||||||
|
"Clear watch history": "Forigi vidohistorion",
|
||||||
|
"Import/export data": "Importi/Eksporti datumojn",
|
||||||
|
"Change password": "Ŝanĝi pasvorton",
|
||||||
|
"Manage subscriptions": "Administri abonojn",
|
||||||
|
"Manage tokens": "Administri ĵetonojn",
|
||||||
|
"Watch history": "Vidohistorio",
|
||||||
|
"Delete account": "Forigi konton",
|
||||||
|
"Administrator preferences": "Agordoj de administranto",
|
||||||
|
"Default homepage: ": "Defaŭlta hejmpaĝo: ",
|
||||||
|
"Feed menu: ": "Flua menuo: ",
|
||||||
|
"Top enabled: ": "Ĉu pli bonaj ŝaltitaj? ",
|
||||||
|
"CAPTCHA enabled: ": "Ĉu CAPTCHA ŝaltita? ",
|
||||||
|
"Login enabled: ": "Ĉu ensaluto aktivita? ",
|
||||||
|
"Registration enabled: ": "Ĉu registriĝo aktivita? ",
|
||||||
|
"Report statistics: ": "Ĉu raporti statistikojn? ",
|
||||||
|
"Save preferences": "Konservi agordojn",
|
||||||
|
"Subscription manager": "Administrilo de abonoj",
|
||||||
|
"Token manager": "Ĵetona administrilo",
|
||||||
|
"Token": "Ĵetono",
|
||||||
|
"`x` subscriptions": "`x` abonoj",
|
||||||
|
"`x` tokens": "`x` ĵetonoj",
|
||||||
|
"Import/export": "Importi/Eksporti",
|
||||||
|
"unsubscribe": "malaboni",
|
||||||
|
"revoke": "senvalidigi",
|
||||||
|
"Subscriptions": "Abonoj",
|
||||||
|
"`x` unseen notifications": "`x` neviditaj sciigoj",
|
||||||
|
"search": "serĉi",
|
||||||
|
"Log out": "Elsaluti",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Eldonita sub la AGPLv3 de Omar Roth.",
|
||||||
|
"Source available here.": "Fonto havebla ĉi tie.",
|
||||||
|
"View JavaScript license information.": "Vidi Ĝavoskriptan licencan informon.",
|
||||||
|
"View privacy policy.": "Vidi regularon pri privateco.",
|
||||||
|
"Trending": "Tendencoj",
|
||||||
|
"Public": "",
|
||||||
|
"Unlisted": "Ne listigita",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "Vidi videon en Youtube",
|
||||||
|
"Hide annotations": "Kaŝi prinotojn",
|
||||||
|
"Show annotations": "Montri prinotojn",
|
||||||
|
"Genre: ": "Ĝenro: ",
|
||||||
|
"License: ": "Licenco: ",
|
||||||
|
"Family friendly? ": "Ĉu familie amika? ",
|
||||||
|
"Wilson score: ": "Poentaro de Wilson: ",
|
||||||
|
"Engagement: ": "Intereso: ",
|
||||||
|
"Whitelisted regions: ": "Regionoj listigitaj en blanka listo: ",
|
||||||
|
"Blacklisted regions: ": "Regionoj listigitaj en nigra listo: ",
|
||||||
|
"Shared `x`": "Konigita `x`",
|
||||||
|
"`x` views": "`x` spektaĵoj",
|
||||||
|
"Premieres in `x`": "Premieras en `x`",
|
||||||
|
"Premieres `x`": "Premieras `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Saluton! Ŝajnas, ke vi havas Ĝavoskripton malebligitan. Klaku ĉi tie por vidi komentojn, memoru, ke la ŝargado povus daŭri iom pli.",
|
||||||
|
"View YouTube comments": "Vidi komentojn de YouTube",
|
||||||
|
"View more comments on Reddit": "Vidi pli komentoj en Reddit",
|
||||||
|
"View `x` comments": "Vidi `x` komentojn",
|
||||||
|
"View Reddit comments": "Vidi komentojn de Reddit",
|
||||||
|
"Hide replies": "Kaŝi respondojn",
|
||||||
|
"Show replies": "Montri respondojn",
|
||||||
|
"Incorrect password": "Malbona pasvorto",
|
||||||
|
"Quota exceeded, try again in a few hours": "Kvoto transpasita, provu denove post iuj horoj",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Ne povas ensaluti, certigu, ke dufaktora aŭtentigo (Authenticator aŭ SMS) estas ebligita.",
|
||||||
|
"Invalid TFA code": "Nevalida TFA-kodo",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Ensalutado fiaskis. Eble ĉar la dufaktora aŭtentigo estas malebligita en via konto.",
|
||||||
|
"Wrong answer": "Nevalida respondo",
|
||||||
|
"Erroneous CAPTCHA": "Nevalida CAPTCHA",
|
||||||
|
"CAPTCHA is a required field": "CAPTCHA estas deviga kampo",
|
||||||
|
"User ID is a required field": "Uzula identigilo estas deviga kampo",
|
||||||
|
"Password is a required field": "Pasvorto estas deviga kampo",
|
||||||
|
"Wrong username or password": "Nevalida uzantnomo aŭ pasvorto",
|
||||||
|
"Please sign in using 'Log in with Google'": "Bonvolu ensaluti per 'Ensaluti per Google'",
|
||||||
|
"Password cannot be empty": "Pasvorto ne povas esti malplena",
|
||||||
|
"Password cannot be longer than 55 characters": "Pasvorto ne povas esti pli longa ol 55 signoj",
|
||||||
|
"Please log in": "Bonvolu ensaluti",
|
||||||
|
"Invidious Private Feed for `x`": "Privata Fluo de Invidious por `x`",
|
||||||
|
"channel:`x`": "kanalo:`x`",
|
||||||
|
"Deleted or invalid channel": "Forigita aŭ nevalida kanalo",
|
||||||
|
"This channel does not exist.": "Ĉi tiu kanalo ne ekzistas.",
|
||||||
|
"Could not get channel info.": "Ne povis havigi kanalan informon.",
|
||||||
|
"Could not fetch comments": "Ne povis venigi komentojn",
|
||||||
|
"View `x` replies": "Vidi `x` respondojn",
|
||||||
|
"`x` ago": "antaŭ `x`",
|
||||||
|
"Load more": "Ŝarĝi pli",
|
||||||
|
"`x` points": "`x` poentoj",
|
||||||
|
"Could not create mix.": "Ne povis krei mikson.",
|
||||||
|
"Empty playlist": "Ludlisto estas malplena",
|
||||||
|
"Not a playlist.": "Nevalida ludlisto.",
|
||||||
|
"Playlist does not exist.": "Ludlisto ne ekzistas.",
|
||||||
|
"Could not pull trending pages.": "Ne povis venigi tendencajn paĝojn.",
|
||||||
|
"Hidden field \"challenge\" is a required field": "Kaŝita kampo \"challenge\" estas deviga kampo",
|
||||||
|
"Hidden field \"token\" is a required field": "Kaŝita kampo \"token\" estas deviga kampo",
|
||||||
|
"Erroneous challenge": "Nevalida defio",
|
||||||
|
"Erroneous token": "Nevalida ĵetono",
|
||||||
|
"No such user": "Nevalida uzanto",
|
||||||
|
"Token is expired, please try again": "Ĵetono senvalidiĝis, bonvolu provi denove",
|
||||||
|
"English": "Angla",
|
||||||
|
"English (auto-generated)": "Angla (aŭtomate generita)",
|
||||||
|
"Afrikaans": "Afrikansa",
|
||||||
|
"Albanian": "Albana",
|
||||||
|
"Amharic": "Amhara",
|
||||||
|
"Arabic": "Araba",
|
||||||
|
"Armenian": "Armena",
|
||||||
|
"Azerbaijani": "Azerbajĝana",
|
||||||
|
"Bangla": "Bengala",
|
||||||
|
"Basque": "Eŭska",
|
||||||
|
"Belarusian": "Belorusa",
|
||||||
|
"Bosnian": "Bosna",
|
||||||
|
"Bulgarian": "Bulgara",
|
||||||
|
"Burmese": "Birma",
|
||||||
|
"Catalan": "Kataluna",
|
||||||
|
"Cebuano": "Cebua",
|
||||||
|
"Chinese (Simplified)": "Ĉina (simpligita)",
|
||||||
|
"Chinese (Traditional)": "Ĉina (tradicia)",
|
||||||
|
"Corsican": "Korsika",
|
||||||
|
"Croatian": "Kroata",
|
||||||
|
"Czech": "Ĉeĥa",
|
||||||
|
"Danish": "Dana",
|
||||||
|
"Dutch": "Nederlanda",
|
||||||
|
"Esperanto": "Esperanto",
|
||||||
|
"Estonian": "Estona",
|
||||||
|
"Filipino": "Filipina",
|
||||||
|
"Finnish": "Finna",
|
||||||
|
"French": "Franca",
|
||||||
|
"Galician": "Galega",
|
||||||
|
"Georgian": "Kartvela",
|
||||||
|
"German": "Germana",
|
||||||
|
"Greek": "Greka",
|
||||||
|
"Gujarati": "Guĝarata",
|
||||||
|
"Haitian Creole": "Haitia kreola",
|
||||||
|
"Hausa": "Haŭsa",
|
||||||
|
"Hawaiian": "Havaja",
|
||||||
|
"Hebrew": "Hebrea",
|
||||||
|
"Hindi": "Hindia",
|
||||||
|
"Hmong": "Miaa",
|
||||||
|
"Hungarian": "Hungara",
|
||||||
|
"Icelandic": "Islanda",
|
||||||
|
"Igbo": "Igba",
|
||||||
|
"Indonesian": "Indonezia",
|
||||||
|
"Irish": "Irlanda",
|
||||||
|
"Italian": "Itala",
|
||||||
|
"Japanese": "Japana",
|
||||||
|
"Javanese": "Java",
|
||||||
|
"Kannada": "Kanara",
|
||||||
|
"Kazakh": "Kazaĥa",
|
||||||
|
"Khmer": "Kmera",
|
||||||
|
"Korean": "Korea",
|
||||||
|
"Kurdish": "Kurda",
|
||||||
|
"Kyrgyz": "Kirgiza",
|
||||||
|
"Lao": "Laosa",
|
||||||
|
"Latin": "Latina",
|
||||||
|
"Latvian": "Latva",
|
||||||
|
"Lithuanian": "Litova",
|
||||||
|
"Luxembourgish": "Luksemburga",
|
||||||
|
"Macedonian": "Makedona",
|
||||||
|
"Malagasy": "Malagasa",
|
||||||
|
"Malay": "Malaja",
|
||||||
|
"Malayalam": "Malajala",
|
||||||
|
"Maltese": "Malta",
|
||||||
|
"Maori": "Maoria",
|
||||||
|
"Marathi": "Marata",
|
||||||
|
"Mongolian": "Mongola",
|
||||||
|
"Nepali": "Nepala",
|
||||||
|
"Norwegian Bokmål": "Norvega",
|
||||||
|
"Nyanja": "Njanĝa",
|
||||||
|
"Pashto": "Paŝtuna",
|
||||||
|
"Persian": "Persa",
|
||||||
|
"Polish": "Pola",
|
||||||
|
"Portuguese": "Portugala",
|
||||||
|
"Punjabi": "Panĝaba",
|
||||||
|
"Romanian": "Rumana",
|
||||||
|
"Russian": "Rusa",
|
||||||
|
"Samoan": "Samoa",
|
||||||
|
"Scottish Gaelic": "Skotgaela",
|
||||||
|
"Serbian": "Serba",
|
||||||
|
"Shona": "Ŝona",
|
||||||
|
"Sindhi": "Sinda",
|
||||||
|
"Sinhala": "Sinhala",
|
||||||
|
"Slovak": "Slovaka",
|
||||||
|
"Slovenian": "Slovena",
|
||||||
|
"Somali": "Somala",
|
||||||
|
"Southern Sotho": "Sota",
|
||||||
|
"Spanish": "Hispana",
|
||||||
|
"Spanish (Latin America)": "Hispana (Latinameriko)",
|
||||||
|
"Sundanese": "Sunda",
|
||||||
|
"Swahili": "Svahila",
|
||||||
|
"Swedish": "Sveda",
|
||||||
|
"Tajik": "Taĝika",
|
||||||
|
"Tamil": "Tamila",
|
||||||
|
"Telugu": "Telugua",
|
||||||
|
"Thai": "Taja",
|
||||||
|
"Turkish": "Turka",
|
||||||
|
"Ukrainian": "Ukraina",
|
||||||
|
"Urdu": "Urduo",
|
||||||
|
"Uzbek": "Uzbeka",
|
||||||
|
"Vietnamese": "Vjetnama",
|
||||||
|
"Welsh": "Kimra",
|
||||||
|
"Western Frisian": "Okcidentfrisa",
|
||||||
|
"Xhosa": "Kosa",
|
||||||
|
"Yiddish": "Jida",
|
||||||
|
"Yoruba": "Joruba",
|
||||||
|
"Zulu": "Zulua",
|
||||||
|
"`x` years": "`x` jaroj",
|
||||||
|
"`x` months": "`x` monatoj",
|
||||||
|
"`x` weeks": "`x` semajnoj",
|
||||||
|
"`x` days": "`x` tagoj",
|
||||||
|
"`x` hours": "`x` horoj",
|
||||||
|
"`x` minutes": "`x` minutoj",
|
||||||
|
"`x` seconds": "`x` sekundoj",
|
||||||
|
"Fallback comments: ": "Retrodefaŭltaj komentoj: ",
|
||||||
|
"Popular": "Popularaj",
|
||||||
|
"Top": "Supraj",
|
||||||
|
"About": "Pri",
|
||||||
|
"Rating: ": "Takso: ",
|
||||||
|
"Language: ": "Lingvo: ",
|
||||||
|
"View as playlist": "Vidi kiel ludlisto",
|
||||||
|
"Default": "Defaŭlte",
|
||||||
|
"Music": "Musiko",
|
||||||
|
"Gaming": "Komputiloludoj",
|
||||||
|
"News": "Novaĵoj",
|
||||||
|
"Movies": "Filmoj",
|
||||||
|
"Download": "Elŝuti",
|
||||||
|
"Download as: ": "Elŝuti kiel: ",
|
||||||
|
"%A %B %-d, %Y": "%A %-d de %B %Y",
|
||||||
|
"(edited)": "(redaktita)",
|
||||||
|
"YouTube comment permalink": "Fiksligilo de la komento en YouTube",
|
||||||
|
"permalink": "konstanta ligilo",
|
||||||
|
"`x` marked it with a ❤": "`x` markis ĝin per ❤",
|
||||||
|
"Audio mode": "Aŭda reĝimo",
|
||||||
|
"Video mode": "Videa reĝimo",
|
||||||
|
"Videos": "Videoj",
|
||||||
|
"Playlists": "Ludlistoj",
|
||||||
|
"Community": "Komunumo",
|
||||||
|
"Current version: ": "Nuna versio: "
|
||||||
|
}
|
||||||
629
locales/es.json
629
locales/es.json
@@ -1,295 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` suscriptores",
|
"`x` subscribers": "`x` suscriptores",
|
||||||
"`x` videos": "`x` vídeos",
|
"`x` videos": "`x` vídeos",
|
||||||
"LIVE": "DIRECTO",
|
"`x` playlists": "",
|
||||||
"Shared `x` ago": "Compartido hace `x`",
|
"LIVE": "DIRECTO",
|
||||||
"Unsubscribe": "Desuscribirse",
|
"Shared `x` ago": "Compartido hace `x`",
|
||||||
"Subscribe": "Suscribirse",
|
"Unsubscribe": "Desuscribirse",
|
||||||
"Login to subscribe to `x`": "Inicie sesión para suscribirse a `x`",
|
"Subscribe": "Suscribirse",
|
||||||
"View channel on YouTube": "Ver el canal en YouTube",
|
"View channel on YouTube": "Ver el canal en YouTube",
|
||||||
"newest": "más nuevos",
|
"View playlist on YouTube": "Ver lista de reproducción en YouTube",
|
||||||
"oldest": "más viejos",
|
"newest": "más nuevos",
|
||||||
"popular": "populares",
|
"oldest": "más viejos",
|
||||||
"last": "último",
|
"popular": "populares",
|
||||||
"Next page": "Página siguiente",
|
"last": "último",
|
||||||
"Previous page": "Página anterior",
|
"Next page": "Página siguiente",
|
||||||
"Clear watch history?": "¿Quiere borrar el historial de reproducción?",
|
"Previous page": "Página anterior",
|
||||||
"Yes": "Sí",
|
"Clear watch history?": "¿Quiere borrar el historial de reproducción?",
|
||||||
"No": "No",
|
"New password": "Nueva contraseña",
|
||||||
"Import and Export Data": "Importación y exportación de datos",
|
"New passwords must match": "Las nuevas contraseñas deben coincidir",
|
||||||
"Import": "Importar",
|
"Cannot change password for Google accounts": "No se puede cambiar la contraseña de la cuenta de Google",
|
||||||
"Import Invidious data": "Importar datos de Invidious",
|
"Authorize token?": "¿Autorizar el token?",
|
||||||
"Import YouTube subscriptions": "Importar suscripciones de YouTube",
|
"Authorize token for `x`?": "¿Autorizar el token para `x`?",
|
||||||
"Import FreeTube subscriptions (.db)": "Importar suscripciones de FreeTube (.db)",
|
"Yes": "Sí",
|
||||||
"Import NewPipe subscriptions (.json)": "Importar suscripciones de NewPipe (.json)",
|
"No": "No",
|
||||||
"Import NewPipe data (.zip)": "Importar datos de NewPipe (.zip)",
|
"Import and Export Data": "Importación y exportación de datos",
|
||||||
"Export": "Exportar",
|
"Import": "Importar",
|
||||||
"Export subscriptions as OPML": "Exportar suscripciones como OPML",
|
"Import Invidious data": "Importar datos de Invidious",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exportar suscripciones como OPML (para NewPipe y FreeTube)",
|
"Import YouTube subscriptions": "Importar suscripciones de YouTube",
|
||||||
"Export data as JSON": "Exportar datos como JSON",
|
"Import FreeTube subscriptions (.db)": "Importar suscripciones de FreeTube (.db)",
|
||||||
"Delete account?": "¿Quiere borrar la cuenta?",
|
"Import NewPipe subscriptions (.json)": "Importar suscripciones de NewPipe (.json)",
|
||||||
"History": "Historial",
|
"Import NewPipe data (.zip)": "Importar datos de NewPipe (.zip)",
|
||||||
"An alternative front-end to YouTube": "Una interfaz alternativa para YouTube",
|
"Export": "Exportar",
|
||||||
"JavaScript license information": "Información de licencia de JavaScript",
|
"Export subscriptions as OPML": "Exportar suscripciones como OPML",
|
||||||
"source": "código fuente",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exportar suscripciones como OPML (para NewPipe y FreeTube)",
|
||||||
"Login": "Iniciar sesión",
|
"Export data as JSON": "Exportar datos como JSON",
|
||||||
"Login/Register": "Iniciar sesión/Registrarse",
|
"Delete account?": "¿Quiere borrar la cuenta?",
|
||||||
"Login to Google": "Iniciar sesión en Google",
|
"History": "Historial",
|
||||||
"User ID:": "Nombre:",
|
"An alternative front-end to YouTube": "Una interfaz alternativa para YouTube",
|
||||||
"Password:": "Contraseña:",
|
"JavaScript license information": "Información de licencia de JavaScript",
|
||||||
"Time (h:mm:ss):": "Hora (h:mm:ss):",
|
"source": "código fuente",
|
||||||
"Text CAPTCHA": "CAPTCHA en texto",
|
"Log in": "Iniciar sesión",
|
||||||
"Image CAPTCHA": "CAPTCHA en imagen",
|
"Log in/register": "Iniciar sesión/Registrarse",
|
||||||
"Sign In": "Iniciar sesión",
|
"Log in with Google": "Iniciar sesión en Google",
|
||||||
"Register": "Registrarse",
|
"User ID": "Nombre",
|
||||||
"Email:": "Correo:",
|
"Password": "Contraseña",
|
||||||
"Google verification code:": "Código de verificación de Google:",
|
"Time (h:mm:ss):": "Hora (h:mm:ss):",
|
||||||
"Preferences": "Preferencias",
|
"Text CAPTCHA": "CAPTCHA en texto",
|
||||||
"Player preferences": "Preferencias del reproductor",
|
"Image CAPTCHA": "CAPTCHA en imagen",
|
||||||
"Always loop: ": "Repetir siempre: ",
|
"Sign In": "Iniciar sesión",
|
||||||
"Autoplay: ": "Reproducción automática: ",
|
"Register": "Registrarse",
|
||||||
"Autoplay next video: ": "Reproducir automáticamente el vídeo siguiente: ",
|
"E-mail": "Correo",
|
||||||
"Listen by default: ": "Activar el sonido por defecto: ",
|
"Google verification code": "Código de verificación de Google",
|
||||||
"Proxy videos? ": "¿Usar un proxy para los vídeos? ",
|
"Preferences": "Preferencias",
|
||||||
"Default speed: ": "Velocidad por defecto: ",
|
"Player preferences": "Preferencias del reproductor",
|
||||||
"Preferred video quality: ": "Calidad de vídeo preferida: ",
|
"Always loop: ": "Repetir siempre: ",
|
||||||
"Player volume: ": "Volumen del reproductor: ",
|
"Autoplay: ": "Reproducción automática: ",
|
||||||
"Default comments: ": "Comentarios por defecto: ",
|
"Play next by default: ": "Reproducir siguiente por defecto: ",
|
||||||
"Default captions: ": "Subtítulos por defecto: ",
|
"Autoplay next video: ": "Reproducir automáticamente el vídeo siguiente: ",
|
||||||
"Fallback captions: ": "Subtítulos alternativos: ",
|
"Listen by default: ": "Activar el sonido por defecto: ",
|
||||||
"Show related videos? ": "¿Mostrar vídeos relacionados? ",
|
"Proxy videos: ": "¿Usar un proxy para los vídeos? ",
|
||||||
"Visual preferences": "Preferencias visuales",
|
"Default speed: ": "Velocidad por defecto: ",
|
||||||
"Dark mode: ": "Modo oscuro: ",
|
"Preferred video quality: ": "Calidad de vídeo preferida: ",
|
||||||
"Thin mode: ": "Modo compacto: ",
|
"Player volume: ": "Volumen del reproductor: ",
|
||||||
"Subscription preferences": "Preferencias de la suscripción",
|
"Default comments: ": "Comentarios por defecto: ",
|
||||||
"Redirect homepage to feed: ": "Redirigir la página de inicio a la fuente: ",
|
"youtube": "YouTube",
|
||||||
"Number of videos shown in feed: ": "Número de vídeos mostrados en la fuente: ",
|
"reddit": "Reddit",
|
||||||
"Sort videos by: ": "Ordenar los vídeos por: ",
|
"Default captions: ": "Subtítulos por defecto: ",
|
||||||
"published": "fecha de publicación",
|
"Fallback captions: ": "Subtítulos alternativos: ",
|
||||||
"published - reverse": "fecha de publicación: orden inverso",
|
"Show related videos: ": "¿Mostrar vídeos relacionados? ",
|
||||||
"alphabetically": "alfabéticamente",
|
"Show annotations by default: ": "¿Mostrar anotaciones por defecto? ",
|
||||||
"alphabetically - reverse": "alfabéticamente: orden inverso",
|
"Visual preferences": "Preferencias visuales",
|
||||||
"channel name": "nombre del canal",
|
"Player style: ": "",
|
||||||
"channel name - reverse": "nombre del canal: orden inverso",
|
"Dark mode: ": "Modo oscuro: ",
|
||||||
"Only show latest video from channel: ": "Mostrar solo el último vídeo del canal: ",
|
"Theme: ": "",
|
||||||
"Only show latest unwatched video from channel: ": "Mostrar solo el último vídeo sin ver del canal: ",
|
"dark": "",
|
||||||
"Only show unwatched: ": "Mostrar solo los no vistos: ",
|
"light": "",
|
||||||
"Only show notifications (if there are any): ": "Mostrar solo notificaciones (si hay alguna): ",
|
"Thin mode: ": "Modo compacto: ",
|
||||||
"Data preferences": "Preferencias de los datos",
|
"Subscription preferences": "Preferencias de la suscripción",
|
||||||
"Clear watch history": "Borrar el historial de reproducción",
|
"Show annotations by default for subscribed channels: ": "¿Mostrar anotaciones por defecto para los canales suscritos? ",
|
||||||
"Import/Export data": "Importar/Exportar datos",
|
"Redirect homepage to feed: ": "Redirigir la página de inicio a la fuente: ",
|
||||||
"Manage subscriptions": "Gestionar las suscripciones",
|
"Number of videos shown in feed: ": "Número de vídeos mostrados en la fuente: ",
|
||||||
"Watch history": "Historial de reproducción",
|
"Sort videos by: ": "Ordenar los vídeos por: ",
|
||||||
"Delete account": "Borrar cuenta",
|
"published": "fecha de publicación",
|
||||||
"Administrator preferences": "Preferencias de administrador",
|
"published - reverse": "fecha de publicación: orden inverso",
|
||||||
"Default homepage: ": "Página de inicio por defecto: ",
|
"alphabetically": "alfabéticamente",
|
||||||
"Feed menu: ": "Menú de fuentes: ",
|
"alphabetically - reverse": "alfabéticamente: orden inverso",
|
||||||
"Top enabled? ": "¿Habilitar los destacados? ",
|
"channel name": "nombre del canal",
|
||||||
"CAPTCHA enabled? ": "¿Habilitar los CAPTCHA? ",
|
"channel name - reverse": "nombre del canal: orden inverso",
|
||||||
"Login enabled? ": "¿Habilitar el inicio de sesión? ",
|
"Only show latest video from channel: ": "Mostrar solo el último vídeo del canal: ",
|
||||||
"Registration enabled? ": "¿Habilitar el registro? ",
|
"Only show latest unwatched video from channel: ": "Mostrar solo el último vídeo sin ver del canal: ",
|
||||||
"Report statistics? ": "¿Enviar estadísticas? ",
|
"Only show unwatched: ": "Mostrar solo los no vistos: ",
|
||||||
"Save preferences": "Guardar las preferencias",
|
"Only show notifications (if there are any): ": "Mostrar solo notificaciones (si hay alguna): ",
|
||||||
"Subscription manager": "Gestor de suscripciones",
|
"Enable web notifications": "",
|
||||||
"`x` subscriptions": "`x` suscripciones",
|
"`x` uploaded a video": "",
|
||||||
"Import/Export": "Importar/Exportar",
|
"`x` is live": "",
|
||||||
"unsubscribe": "Desuscribirse",
|
"Data preferences": "Preferencias de los datos",
|
||||||
"Subscriptions": "Suscripciones",
|
"Clear watch history": "Borrar el historial de reproducción",
|
||||||
"`x` unseen notifications": "`x` notificaciones sin ver",
|
"Import/export data": "Importar/Exportar datos",
|
||||||
"search": "buscar",
|
"Change password": "Cambiar contraseña",
|
||||||
"Sign out": "Cerrar la sesión",
|
"Manage subscriptions": "Gestionar las suscripciones",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Publicado bajo licencia AGPLv3 por Omar Roth.",
|
"Manage tokens": "Gestionar tokens",
|
||||||
"Source available here.": "Código fuente disponible aquí.",
|
"Watch history": "Historial de reproducción",
|
||||||
"View JavaScript license information.": "Ver información de licencia de JavaScript.",
|
"Delete account": "Borrar cuenta",
|
||||||
"View privacy policy.": "Ver la política de privacidad.",
|
"Administrator preferences": "Preferencias de administrador",
|
||||||
"Trending": "Tendencias",
|
"Default homepage: ": "Página de inicio por defecto: ",
|
||||||
"Unlisted": "",
|
"Feed menu: ": "Menú de fuentes: ",
|
||||||
"Watch video on Youtube": "Ver el vídeo en Youtube",
|
"Top enabled: ": "¿Habilitar los destacados? ",
|
||||||
"Genre: ": "Género: ",
|
"CAPTCHA enabled: ": "¿Habilitar los CAPTCHA? ",
|
||||||
"License: ": "Licencia: ",
|
"Login enabled: ": "¿Habilitar el inicio de sesión? ",
|
||||||
"Family friendly? ": "¿Filtrar contenidos? ",
|
"Registration enabled: ": "¿Habilitar el registro? ",
|
||||||
"Wilson score: ": "Puntuación Wilson: ",
|
"Report statistics: ": "¿Enviar estadísticas? ",
|
||||||
"Engagement: ": "Compromiso: ",
|
"Save preferences": "Guardar las preferencias",
|
||||||
"Whitelisted regions: ": "Regiones permitidas: ",
|
"Subscription manager": "Gestor de suscripciones",
|
||||||
"Blacklisted regions: ": "Regiones bloqueadas: ",
|
"Token manager": "Gestor de tokens",
|
||||||
"Shared `x`": "Compartido `x`",
|
"Token": "Token",
|
||||||
"Premieres in `x`": "",
|
"`x` subscriptions": "`x` suscripciones",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "¡Hola! Parece que tiene JavaScript desactivado. Haga clic aquí para ver los comentarios, pero tenga en cuenta que pueden tardar un poco más en cargarse.",
|
"`x` tokens": "`x` tokens",
|
||||||
"View YouTube comments": "Ver los comentarios de YouTube",
|
"Import/export": "Importar/Exportar",
|
||||||
"View more comments on Reddit": "Ver más comentarios en Reddit",
|
"unsubscribe": "Desuscribirse",
|
||||||
"View `x` comments": "Ver `x` comentarios",
|
"revoke": "revocar",
|
||||||
"View Reddit comments": "Ver los comentarios de Reddit",
|
"Subscriptions": "Suscripciones",
|
||||||
"Hide replies": "Ocultar las respuestas",
|
"`x` unseen notifications": "`x` notificaciones sin ver",
|
||||||
"Show replies": "Mostrar las respuestas",
|
"search": "buscar",
|
||||||
"Incorrect password": "Contraseña incorrecta",
|
"Log out": "Cerrar la sesión",
|
||||||
"Quota exceeded, try again in a few hours": "Cuota excedida, pruebe otra vez en unas horas",
|
"Released under the AGPLv3 by Omar Roth.": "Publicado bajo licencia AGPLv3 por Omar Roth.",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "No se puede iniciar sesión, asegúrese de que la autentificación de dos factores (autentificador o SMS) esté habilitada.",
|
"Source available here.": "Código fuente disponible aquí.",
|
||||||
"Invalid TFA code": "Código TFA no válido",
|
"View JavaScript license information.": "Ver información de licencia de JavaScript.",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Error de inicio de sesion. Puede deberse a que la autentificación de dos factores no está habilitada en su cuenta.",
|
"View privacy policy.": "Ver la política de privacidad.",
|
||||||
"Invalid answer": "Respuesta no válida",
|
"Trending": "Tendencias",
|
||||||
"Invalid CAPTCHA": "CAPTCHA no válido",
|
"Public": "",
|
||||||
"CAPTCHA is a required field": "El CAPTCHA es un campo obligatorio",
|
"Unlisted": "No listado",
|
||||||
"User ID is a required field": "El nombre es un campo obligatorio",
|
"Private": "",
|
||||||
"Password is a required field": "La contraseña es un campo obligatorio",
|
"View all playlists": "",
|
||||||
"Invalid username or password": "Nombre o contraseña incorrecto",
|
"Updated `x` ago": "",
|
||||||
"Please sign in using 'Sign in with Google'": "Inicie sesión con «Iniciar sesión con Google»",
|
"Delete playlist `x`?": "",
|
||||||
"Password cannot be empty": "La contraseña no puede estar en blanco",
|
"Delete playlist": "",
|
||||||
"Password cannot be longer than 55 characters": "La contraseña no puede tener más de 55 caracteres",
|
"Create playlist": "",
|
||||||
"Please sign in": "Inicie sesión, por favor",
|
"Title": "",
|
||||||
"Invidious Private Feed for `x`": "Fuente privada de Invidious para `x`",
|
"Playlist privacy": "",
|
||||||
"channel:`x`": "canal: `x`",
|
"Editing playlist `x`": "",
|
||||||
"Deleted or invalid channel": "El canal no es válido o ha sido borrado",
|
"Watch on YouTube": "Ver el vídeo en Youtube",
|
||||||
"This channel does not exist.": "El canal no existe.",
|
"Hide annotations": "Ocultar anotaciones",
|
||||||
"Could not get channel info.": "No se ha podido obtener información del canal.",
|
"Show annotations": "Mostrar anotaciones",
|
||||||
"Could not fetch comments": "No se han podido recuperar los comentarios.",
|
"Genre: ": "Género: ",
|
||||||
"View `x` replies": "Ver `x` respuestas",
|
"License: ": "Licencia: ",
|
||||||
"`x` ago": "hace `x`",
|
"Family friendly? ": "¿Filtrar contenidos? ",
|
||||||
"Load more": "Cargar más",
|
"Wilson score: ": "Puntuación Wilson: ",
|
||||||
"`x` points": "`x` puntos",
|
"Engagement: ": "Compromiso: ",
|
||||||
"Could not create mix.": "No se ha podido crear la mezcla.",
|
"Whitelisted regions: ": "Regiones permitidas: ",
|
||||||
"Playlist is empty": "La lista de reproducción está vacía",
|
"Blacklisted regions: ": "Regiones bloqueadas: ",
|
||||||
"Invalid playlist.": "Lista de reproducción no válida.",
|
"Shared `x`": "Compartido `x`",
|
||||||
"Playlist does not exist.": "La lista de reproducción no existe.",
|
"`x` views": "`x` visualizaciones",
|
||||||
"Could not pull trending pages.": "No se han podido obtener las páginas de tendencias.",
|
"Premieres in `x`": "Se estrena en `x`",
|
||||||
"Hidden field \"challenge\" is a required field": "El campo oculto «desafío» es un campo obligatorio",
|
"Premieres `x`": "",
|
||||||
"Hidden field \"token\" is a required field": "El campo oculto «símbolo» es un campo obligatorio",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "¡Hola! Parece que tiene JavaScript desactivado. Haga clic aquí para ver los comentarios, pero tenga en cuenta que pueden tardar un poco más en cargarse.",
|
||||||
"Invalid challenge": "Desafío no válido",
|
"View YouTube comments": "Ver los comentarios de YouTube",
|
||||||
"Invalid token": "Símbolo no válido",
|
"View more comments on Reddit": "Ver más comentarios en Reddit",
|
||||||
"Invalid user": "Usuario no válido",
|
"View `x` comments": "Ver `x` comentarios",
|
||||||
"Token is expired, please try again": "El símbolo ha caducado, inténtelo de nuevo",
|
"View Reddit comments": "Ver los comentarios de Reddit",
|
||||||
"English": "Inglés",
|
"Hide replies": "Ocultar las respuestas",
|
||||||
"English (auto-generated)": "Inglés (autogenerado)",
|
"Show replies": "Mostrar las respuestas",
|
||||||
"Afrikaans": "Afrikáans",
|
"Incorrect password": "Contraseña incorrecta",
|
||||||
"Albanian": "Albanés",
|
"Quota exceeded, try again in a few hours": "Cuota excedida, pruebe otra vez en unas horas",
|
||||||
"Amharic": "Amárico",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "No se puede iniciar sesión, asegúrese de que la autentificación de dos factores (autentificador o SMS) esté habilitada.",
|
||||||
"Arabic": "Árabe",
|
"Invalid TFA code": "Código TFA no válido",
|
||||||
"Armenian": "Armenio",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Error de inicio de sesion. Puede deberse a que la autentificación de dos factores no está habilitada en su cuenta.",
|
||||||
"Azerbaijani": "Azerbaiyano",
|
"Wrong answer": "Respuesta no válida",
|
||||||
"Bangla": "Bengalí",
|
"Erroneous CAPTCHA": "CAPTCHA no válido",
|
||||||
"Basque": "Euskera",
|
"CAPTCHA is a required field": "El CAPTCHA es un campo obligatorio",
|
||||||
"Belarusian": "Bielorruso",
|
"User ID is a required field": "El nombre es un campo obligatorio",
|
||||||
"Bosnian": "Bosnio",
|
"Password is a required field": "La contraseña es un campo obligatorio",
|
||||||
"Bulgarian": "Búlgaro",
|
"Wrong username or password": "Nombre o contraseña incorrecto",
|
||||||
"Burmese": "Birmano",
|
"Please sign in using 'Log in with Google'": "Inicie sesión con «Iniciar sesión con Google»",
|
||||||
"Catalan": "Catalán",
|
"Password cannot be empty": "La contraseña no puede estar en blanco",
|
||||||
"Cebuano": "Cebuano",
|
"Password cannot be longer than 55 characters": "La contraseña no puede tener más de 55 caracteres",
|
||||||
"Chinese (Simplified)": "Chino (simplificado)",
|
"Please log in": "Inicie sesión, por favor",
|
||||||
"Chinese (Traditional)": "Chino (tradicional)",
|
"Invidious Private Feed for `x`": "Fuente privada de Invidious para `x`",
|
||||||
"Corsican": "Corso",
|
"channel:`x`": "canal: `x`",
|
||||||
"Croatian": "Croata",
|
"Deleted or invalid channel": "El canal no es válido o ha sido borrado",
|
||||||
"Czech": "Checo",
|
"This channel does not exist.": "El canal no existe.",
|
||||||
"Danish": "Danés",
|
"Could not get channel info.": "No se ha podido obtener información del canal.",
|
||||||
"Dutch": "Holandés",
|
"Could not fetch comments": "No se han podido recuperar los comentarios",
|
||||||
"Esperanto": "Esperanto",
|
"View `x` replies": "Ver `x` respuestas",
|
||||||
"Estonian": "Estonio",
|
"`x` ago": "hace `x`",
|
||||||
"Filipino": "Filipino",
|
"Load more": "Cargar más",
|
||||||
"Finnish": "Finés",
|
"`x` points": "`x` puntos",
|
||||||
"French": "Francés",
|
"Could not create mix.": "No se ha podido crear la mezcla.",
|
||||||
"Galician": "Gallego",
|
"Empty playlist": "La lista de reproducción está vacía",
|
||||||
"Georgian": "Georgiano",
|
"Not a playlist.": "Lista de reproducción no válida.",
|
||||||
"German": "Alemán",
|
"Playlist does not exist.": "La lista de reproducción no existe.",
|
||||||
"Greek": "Griego",
|
"Could not pull trending pages.": "No se han podido obtener las páginas de tendencias.",
|
||||||
"Gujarati": "Guyaratí",
|
"Hidden field \"challenge\" is a required field": "El campo oculto «desafío» es un campo obligatorio",
|
||||||
"Haitian Creole": "Criollo haitiano",
|
"Hidden field \"token\" is a required field": "El campo oculto «símbolo» es un campo obligatorio",
|
||||||
"Hausa": "Hausa",
|
"Erroneous challenge": "Desafío no válido",
|
||||||
"Hawaiian": "Hawaiano",
|
"Erroneous token": "Símbolo no válido",
|
||||||
"Hebrew": "Hebreo",
|
"No such user": "Usuario no válido",
|
||||||
"Hindi": "Hindi",
|
"Token is expired, please try again": "El símbolo ha caducado, inténtelo de nuevo",
|
||||||
"Hmong": "Hmong",
|
"English": "Inglés",
|
||||||
"Hungarian": "Húngaro",
|
"English (auto-generated)": "Inglés (autogenerado)",
|
||||||
"Icelandic": "Islandés",
|
"Afrikaans": "Afrikáans",
|
||||||
"Igbo": "Igbo",
|
"Albanian": "Albanés",
|
||||||
"Indonesian": "Indonesio",
|
"Amharic": "Amárico",
|
||||||
"Irish": "Irlandés",
|
"Arabic": "Árabe",
|
||||||
"Italian": "Italiano",
|
"Armenian": "Armenio",
|
||||||
"Japanese": "Japonés",
|
"Azerbaijani": "Azerbaiyano",
|
||||||
"Javanese": "Javanés",
|
"Bangla": "Bengalí",
|
||||||
"Kannada": "Canarés",
|
"Basque": "Euskera",
|
||||||
"Kazakh": "Kazajo",
|
"Belarusian": "Bielorruso",
|
||||||
"Khmer": "Camboyano",
|
"Bosnian": "Bosnio",
|
||||||
"Korean": "Coreano",
|
"Bulgarian": "Búlgaro",
|
||||||
"Kurdish": "Kurdo",
|
"Burmese": "Birmano",
|
||||||
"Kyrgyz": "Kirguís",
|
"Catalan": "Catalán",
|
||||||
"Lao": "Laosiano",
|
"Cebuano": "Cebuano",
|
||||||
"Latin": "Latín",
|
"Chinese (Simplified)": "Chino (simplificado)",
|
||||||
"Latvian": "Letón",
|
"Chinese (Traditional)": "Chino (tradicional)",
|
||||||
"Lithuanian": "Lituano",
|
"Corsican": "Corso",
|
||||||
"Luxembourgish": "Luxemburgués",
|
"Croatian": "Croata",
|
||||||
"Macedonian": "Macedonio",
|
"Czech": "Checo",
|
||||||
"Malagasy": "Malgache",
|
"Danish": "Danés",
|
||||||
"Malay": "Malayo",
|
"Dutch": "Holandés",
|
||||||
"Malayalam": "Malabar",
|
"Esperanto": "Esperanto",
|
||||||
"Maltese": "Maltés",
|
"Estonian": "Estonio",
|
||||||
"Maori": "Maorí",
|
"Filipino": "Filipino",
|
||||||
"Marathi": "Maratí",
|
"Finnish": "Finés",
|
||||||
"Mongolian": "Mongol",
|
"French": "Francés",
|
||||||
"Nepali": "Nepalí",
|
"Galician": "Gallego",
|
||||||
"Norwegian": "Noruego",
|
"Georgian": "Georgiano",
|
||||||
"Nyanja": "Chichewa",
|
"German": "Alemán",
|
||||||
"Pashto": "Pastún",
|
"Greek": "Griego",
|
||||||
"Persian": "Persa",
|
"Gujarati": "Guyaratí",
|
||||||
"Polish": "Polaco",
|
"Haitian Creole": "Criollo haitiano",
|
||||||
"Portuguese": "Portugués",
|
"Hausa": "Hausa",
|
||||||
"Punjabi": "Panyabí",
|
"Hawaiian": "Hawaiano",
|
||||||
"Romanian": "Rumano",
|
"Hebrew": "Hebreo",
|
||||||
"Russian": "Ruso",
|
"Hindi": "Hindi",
|
||||||
"Samoan": "Samoano",
|
"Hmong": "Hmong",
|
||||||
"Scottish Gaelic": "Gaélico escocés",
|
"Hungarian": "Húngaro",
|
||||||
"Serbian": "Serbio",
|
"Icelandic": "Islandés",
|
||||||
"Shona": "Shona",
|
"Igbo": "Igbo",
|
||||||
"Sindhi": "Sindi",
|
"Indonesian": "Indonesio",
|
||||||
"Sinhala": "Cingalés",
|
"Irish": "Irlandés",
|
||||||
"Slovak": "Eslovaco",
|
"Italian": "Italiano",
|
||||||
"Slovenian": "Esloveno",
|
"Japanese": "Japonés",
|
||||||
"Somali": "Somalí",
|
"Javanese": "Javanés",
|
||||||
"Southern Sotho": "Sesoto",
|
"Kannada": "Canarés",
|
||||||
"Spanish": "Español",
|
"Kazakh": "Kazajo",
|
||||||
"Spanish (Latin America)": "Español (Hispanoamérica)",
|
"Khmer": "Camboyano",
|
||||||
"Sundanese": "Sondanés",
|
"Korean": "Coreano",
|
||||||
"Swahili": "Suajili",
|
"Kurdish": "Kurdo",
|
||||||
"Swedish": "Sueco",
|
"Kyrgyz": "Kirguís",
|
||||||
"Tajik": "Tayiko",
|
"Lao": "Laosiano",
|
||||||
"Tamil": "Tamil",
|
"Latin": "Latín",
|
||||||
"Telugu": "Telugu",
|
"Latvian": "Letón",
|
||||||
"Thai": "Tailandés",
|
"Lithuanian": "Lituano",
|
||||||
"Turkish": "Turco",
|
"Luxembourgish": "Luxemburgués",
|
||||||
"Ukrainian": "Ucraniano",
|
"Macedonian": "Macedonio",
|
||||||
"Urdu": "Urdu",
|
"Malagasy": "Malgache",
|
||||||
"Uzbek": "Uzbeko",
|
"Malay": "Malayo",
|
||||||
"Vietnamese": "Vietnamita",
|
"Malayalam": "Malabar",
|
||||||
"Welsh": "Galés",
|
"Maltese": "Maltés",
|
||||||
"Western Frisian": "Frisón",
|
"Maori": "Maorí",
|
||||||
"Xhosa": "Xhosa",
|
"Marathi": "Maratí",
|
||||||
"Yiddish": "Yidis",
|
"Mongolian": "Mongol",
|
||||||
"Yoruba": "Yoruba",
|
"Nepali": "Nepalí",
|
||||||
"Zulu": "Zulú",
|
"Norwegian Bokmål": "Noruego",
|
||||||
"`x` years": "`x` años",
|
"Nyanja": "Chichewa",
|
||||||
"`x` months": "`x` meses",
|
"Pashto": "Pastún",
|
||||||
"`x` weeks": "`x` semanas",
|
"Persian": "Persa",
|
||||||
"`x` days": "`x` días",
|
"Polish": "Polaco",
|
||||||
"`x` hours": "`x` horas",
|
"Portuguese": "Portugués",
|
||||||
"`x` minutes": "`x` minutos",
|
"Punjabi": "Panyabí",
|
||||||
"`x` seconds": "`x` segundos",
|
"Romanian": "Rumano",
|
||||||
"Fallback comments: ": "Comentarios alternativos: ",
|
"Russian": "Ruso",
|
||||||
"Popular": "Populares",
|
"Samoan": "Samoano",
|
||||||
"Top": "Destacados",
|
"Scottish Gaelic": "Gaélico escocés",
|
||||||
"About": "Acerca de",
|
"Serbian": "Serbio",
|
||||||
"Rating: ": "Valoración: ",
|
"Shona": "Shona",
|
||||||
"Language: ": "Idioma: ",
|
"Sindhi": "Sindi",
|
||||||
"Default": "Por defecto",
|
"Sinhala": "Cingalés",
|
||||||
"Music": "Música",
|
"Slovak": "Eslovaco",
|
||||||
"Gaming": "Videojuegos",
|
"Slovenian": "Esloveno",
|
||||||
"News": "Noticias",
|
"Somali": "Somalí",
|
||||||
"Movies": "Películas",
|
"Southern Sotho": "Sesoto",
|
||||||
"Download": "Descargar",
|
"Spanish": "Español",
|
||||||
"Download as: ": "Descargar como: ",
|
"Spanish (Latin America)": "Español (Hispanoamérica)",
|
||||||
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
"Sundanese": "Sondanés",
|
||||||
"(edited)": "(editado)",
|
"Swahili": "Suajili",
|
||||||
"Youtube permalink of the comment": "Enlace permanente de YouTube del comentario",
|
"Swedish": "Sueco",
|
||||||
"`x` marked it with a ❤": "`x` lo ha marcado con un ❤",
|
"Tajik": "Tayiko",
|
||||||
"Audio mode": "Modo de audio",
|
"Tamil": "Tamil",
|
||||||
"Video mode": "Modo de vídeo",
|
"Telugu": "Telugu",
|
||||||
"Videos": "Vídeos",
|
"Thai": "Tailandés",
|
||||||
"Playlists": "Listas de reproducción",
|
"Turkish": "Turco",
|
||||||
"Current version: ": "Versión actual: "
|
"Ukrainian": "Ucraniano",
|
||||||
}
|
"Urdu": "Urdu",
|
||||||
|
"Uzbek": "Uzbeko",
|
||||||
|
"Vietnamese": "Vietnamita",
|
||||||
|
"Welsh": "Galés",
|
||||||
|
"Western Frisian": "Frisón",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Yidis",
|
||||||
|
"Yoruba": "Yoruba",
|
||||||
|
"Zulu": "Zulú",
|
||||||
|
"`x` years": "`x` años",
|
||||||
|
"`x` months": "`x` meses",
|
||||||
|
"`x` weeks": "`x` semanas",
|
||||||
|
"`x` days": "`x` días",
|
||||||
|
"`x` hours": "`x` horas",
|
||||||
|
"`x` minutes": "`x` minutos",
|
||||||
|
"`x` seconds": "`x` segundos",
|
||||||
|
"Fallback comments: ": "Comentarios alternativos: ",
|
||||||
|
"Popular": "Populares",
|
||||||
|
"Top": "Destacados",
|
||||||
|
"About": "Acerca de",
|
||||||
|
"Rating: ": "Valoración: ",
|
||||||
|
"Language: ": "Idioma: ",
|
||||||
|
"View as playlist": "Ver como lista de reproducción",
|
||||||
|
"Default": "Por defecto",
|
||||||
|
"Music": "Música",
|
||||||
|
"Gaming": "Videojuegos",
|
||||||
|
"News": "Noticias",
|
||||||
|
"Movies": "Películas",
|
||||||
|
"Download": "Descargar",
|
||||||
|
"Download as: ": "Descargar como: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(editado)",
|
||||||
|
"YouTube comment permalink": "Enlace permanente de YouTube del comentario",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "`x` lo ha marcado con un ❤",
|
||||||
|
"Audio mode": "Modo de audio",
|
||||||
|
"Video mode": "Modo de vídeo",
|
||||||
|
"Videos": "Vídeos",
|
||||||
|
"Playlists": "Listas de reproducción",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": "Versión actual: "
|
||||||
|
}
|
||||||
629
locales/eu.json
629
locales/eu.json
@@ -1,295 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` harpidedun",
|
"`x` subscribers": "`x` harpidedun",
|
||||||
"`x` videos": "`x` bideo",
|
"`x` videos": "`x` bideo",
|
||||||
"LIVE": "ZUZENEAN",
|
"`x` playlists": "",
|
||||||
"Shared `x` ago": "Duela `x` partekatua",
|
"LIVE": "ZUZENEAN",
|
||||||
"Unsubscribe": "Harpidetza kendu",
|
"Shared `x` ago": "Duela `x` partekatua",
|
||||||
"Subscribe": "Harpidetu",
|
"Unsubscribe": "Harpidetza kendu",
|
||||||
"Login to subscribe to `x`": "Saioa hasi `x`(e)ra harpidetzeko",
|
"Subscribe": "Harpidetu",
|
||||||
"View channel on YouTube": "Ikusi kanala YouTuben",
|
"View channel on YouTube": "Ikusi kanala YouTuben",
|
||||||
"newest": "berrienak",
|
"View playlist on YouTube": "",
|
||||||
"oldest": "zaharrenak",
|
"newest": "berrienak",
|
||||||
"popular": "ospetsuenak",
|
"oldest": "zaharrenak",
|
||||||
"last": "",
|
"popular": "ospetsuenak",
|
||||||
"Next page": "Hurrengo orria",
|
"last": "azkena",
|
||||||
"Previous page": "Aurreko orria",
|
"Next page": "Hurrengo orria",
|
||||||
"Clear watch history?": "Garbitu ikusitakoen historia?",
|
"Previous page": "Aurreko orria",
|
||||||
"Yes": "Bai",
|
"Clear watch history?": "Garbitu ikusitakoen historia?",
|
||||||
"No": "Ez",
|
"New password": "Pasahitz berria",
|
||||||
"Import and Export Data": "Datuak inportatu eta esportatu",
|
"New passwords must match": "",
|
||||||
"Import": "Inportatu",
|
"Cannot change password for Google accounts": "",
|
||||||
"Import Invidious data": "Invidiouseko datuak inportatu",
|
"Authorize token?": "",
|
||||||
"Import YouTube subscriptions": "YouTubeko harpidetzak inportatu",
|
"Authorize token for `x`?": "",
|
||||||
"Import FreeTube subscriptions (.db)": "FreeTubeko harpidetzak inportatu (.db)",
|
"Yes": "Bai",
|
||||||
"Import NewPipe subscriptions (.json)": "NewPipeko harpidetzak inportatu (.json)",
|
"No": "Ez",
|
||||||
"Import NewPipe data (.zip)": "NewPipeko datuak inportatu (.zip)",
|
"Import and Export Data": "Datuak inportatu eta esportatu",
|
||||||
"Export": "Esportatu",
|
"Import": "Inportatu",
|
||||||
"Export subscriptions as OPML": "Esportatu harpidetzak OPML bezala",
|
"Import Invidious data": "Invidiouseko datuak inportatu",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Harpidetzak OPML bezala esportatu (NewPipe eta FreeTuberako)",
|
"Import YouTube subscriptions": "YouTubeko harpidetzak inportatu",
|
||||||
"Export data as JSON": "Datuak JSON bezala esportatu",
|
"Import FreeTube subscriptions (.db)": "FreeTubeko harpidetzak inportatu (.db)",
|
||||||
"Delete account?": "Kontua ezabatu?",
|
"Import NewPipe subscriptions (.json)": "NewPipeko harpidetzak inportatu (.json)",
|
||||||
"History": "Historia",
|
"Import NewPipe data (.zip)": "NewPipeko datuak inportatu (.zip)",
|
||||||
"An alternative front-end to YouTube": "YouTuberako interfaze alternatibo bat",
|
"Export": "Esportatu",
|
||||||
"JavaScript license information": "JavaScript lizentzia informazioa",
|
"Export subscriptions as OPML": "Esportatu harpidetzak OPML bezala",
|
||||||
"source": "iturburua",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Harpidetzak OPML bezala esportatu (NewPipe eta FreeTuberako)",
|
||||||
"Login": "Saioa hasi",
|
"Export data as JSON": "Datuak JSON bezala esportatu",
|
||||||
"Login/Register": "Saioa hasi/Izena eman",
|
"Delete account?": "Kontua ezabatu?",
|
||||||
"Login to Google": "Googlekin hasi saioa",
|
"History": "Historia",
|
||||||
"User ID:": "Erabiltzaile IDa:",
|
"An alternative front-end to YouTube": "YouTuberako interfaze alternatibo bat",
|
||||||
"Password:": "Pasahitza:",
|
"JavaScript license information": "JavaScript lizentzia informazioa",
|
||||||
"Time (h:mm:ss):": "Denbora (o:mm:ss):",
|
"source": "iturburua",
|
||||||
"Text CAPTCHA": "Testu CAPTCHA",
|
"Log in": "Saioa hasi",
|
||||||
"Image CAPTCHA": "Irudi CAPTCHA",
|
"Log in/register": "Saioa hasi/Izena eman",
|
||||||
"Sign In": "",
|
"Log in with Google": "Googlekin hasi saioa",
|
||||||
"Register": "",
|
"User ID": "Erabiltzaile IDa",
|
||||||
"Email:": "",
|
"Password": "Pasahitza",
|
||||||
"Google verification code:": "",
|
"Time (h:mm:ss):": "Denbora (o:mm:ss):",
|
||||||
"Preferences": "",
|
"Text CAPTCHA": "Testu CAPTCHA",
|
||||||
"Player preferences": "",
|
"Image CAPTCHA": "Irudi CAPTCHA",
|
||||||
"Always loop: ": "",
|
"Sign In": "",
|
||||||
"Autoplay: ": "",
|
"Register": "",
|
||||||
"Autoplay next video: ": "",
|
"E-mail": "",
|
||||||
"Listen by default: ": "",
|
"Google verification code": "",
|
||||||
"Proxy videos? ": "",
|
"Preferences": "",
|
||||||
"Default speed: ": "",
|
"Player preferences": "",
|
||||||
"Preferred video quality: ": "",
|
"Always loop: ": "",
|
||||||
"Player volume: ": "",
|
"Autoplay: ": "",
|
||||||
"Default comments: ": "",
|
"Play next by default: ": "",
|
||||||
"Default captions: ": "",
|
"Autoplay next video: ": "",
|
||||||
"Fallback captions: ": "",
|
"Listen by default: ": "",
|
||||||
"Show related videos? ": "",
|
"Proxy videos: ": "",
|
||||||
"Visual preferences": "",
|
"Default speed: ": "",
|
||||||
"Dark mode: ": "",
|
"Preferred video quality: ": "",
|
||||||
"Thin mode: ": "",
|
"Player volume: ": "",
|
||||||
"Subscription preferences": "",
|
"Default comments: ": "",
|
||||||
"Redirect homepage to feed: ": "",
|
"youtube": "",
|
||||||
"Number of videos shown in feed: ": "",
|
"reddit": "",
|
||||||
"Sort videos by: ": "",
|
"Default captions: ": "",
|
||||||
"published": "",
|
"Fallback captions: ": "",
|
||||||
"published - reverse": "",
|
"Show related videos: ": "",
|
||||||
"alphabetically": "",
|
"Show annotations by default: ": "",
|
||||||
"alphabetically - reverse": "",
|
"Visual preferences": "",
|
||||||
"channel name": "",
|
"Player style: ": "",
|
||||||
"channel name - reverse": "",
|
"Dark mode: ": "",
|
||||||
"Only show latest video from channel: ": "",
|
"Theme: ": "",
|
||||||
"Only show latest unwatched video from channel: ": "",
|
"dark": "",
|
||||||
"Only show unwatched: ": "",
|
"light": "",
|
||||||
"Only show notifications (if there are any): ": "",
|
"Thin mode: ": "",
|
||||||
"Data preferences": "",
|
"Subscription preferences": "",
|
||||||
"Clear watch history": "",
|
"Show annotations by default for subscribed channels: ": "",
|
||||||
"Import/Export data": "",
|
"Redirect homepage to feed: ": "",
|
||||||
"Manage subscriptions": "",
|
"Number of videos shown in feed: ": "",
|
||||||
"Watch history": "",
|
"Sort videos by: ": "",
|
||||||
"Delete account": "",
|
"published": "",
|
||||||
"Administrator preferences": "",
|
"published - reverse": "",
|
||||||
"Default homepage: ": "",
|
"alphabetically": "",
|
||||||
"Feed menu: ": "",
|
"alphabetically - reverse": "",
|
||||||
"Top enabled? ": "",
|
"channel name": "",
|
||||||
"CAPTCHA enabled? ": "",
|
"channel name - reverse": "",
|
||||||
"Login enabled? ": "",
|
"Only show latest video from channel: ": "",
|
||||||
"Registration enabled? ": "",
|
"Only show latest unwatched video from channel: ": "",
|
||||||
"Report statistics? ": "",
|
"Only show unwatched: ": "",
|
||||||
"Save preferences": "",
|
"Only show notifications (if there are any): ": "",
|
||||||
"Subscription manager": "",
|
"Enable web notifications": "",
|
||||||
"`x` subscriptions": "",
|
"`x` uploaded a video": "",
|
||||||
"Import/Export": "",
|
"`x` is live": "",
|
||||||
"unsubscribe": "",
|
"Data preferences": "",
|
||||||
"Subscriptions": "",
|
"Clear watch history": "",
|
||||||
"`x` unseen notifications": "",
|
"Import/export data": "",
|
||||||
"search": "",
|
"Change password": "",
|
||||||
"Sign out": "",
|
"Manage subscriptions": "",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "",
|
"Manage tokens": "",
|
||||||
"Source available here.": "",
|
"Watch history": "",
|
||||||
"View JavaScript license information.": "",
|
"Delete account": "",
|
||||||
"View privacy policy.": "",
|
"Administrator preferences": "",
|
||||||
"Unlisted": "",
|
"Default homepage: ": "",
|
||||||
"Trending": "",
|
"Feed menu: ": "",
|
||||||
"Watch video on Youtube": "",
|
"Top enabled: ": "",
|
||||||
"Genre: ": "",
|
"CAPTCHA enabled: ": "",
|
||||||
"License: ": "",
|
"Login enabled: ": "",
|
||||||
"Family friendly? ": "",
|
"Registration enabled: ": "",
|
||||||
"Wilson score: ": "",
|
"Report statistics: ": "",
|
||||||
"Engagement: ": "",
|
"Save preferences": "",
|
||||||
"Whitelisted regions: ": "",
|
"Subscription manager": "",
|
||||||
"Blacklisted regions: ": "",
|
"Token manager": "",
|
||||||
"Shared `x`": "",
|
"Token": "",
|
||||||
"Premieres in `x`": "",
|
"`x` subscriptions": "",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "",
|
"`x` tokens": "",
|
||||||
"View YouTube comments": "",
|
"Import/export": "",
|
||||||
"View more comments on Reddit": "",
|
"unsubscribe": "",
|
||||||
"View `x` comments": "",
|
"revoke": "",
|
||||||
"View Reddit comments": "",
|
"Subscriptions": "",
|
||||||
"Hide replies": "",
|
"`x` unseen notifications": "",
|
||||||
"Show replies": "",
|
"search": "",
|
||||||
"Incorrect password": "",
|
"Log out": "",
|
||||||
"Quota exceeded, try again in a few hours": "",
|
"Released under the AGPLv3 by Omar Roth.": "",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "",
|
"Source available here.": "",
|
||||||
"Invalid TFA code": "",
|
"View JavaScript license information.": "",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "",
|
"View privacy policy.": "",
|
||||||
"Invalid answer": "",
|
"Trending": "",
|
||||||
"Invalid CAPTCHA": "",
|
"Public": "",
|
||||||
"CAPTCHA is a required field": "",
|
"Unlisted": "",
|
||||||
"User ID is a required field": "",
|
"Private": "",
|
||||||
"Password is a required field": "",
|
"View all playlists": "",
|
||||||
"Invalid username or password": "",
|
"Updated `x` ago": "",
|
||||||
"Please sign in using 'Sign in with Google'": "",
|
"Delete playlist `x`?": "",
|
||||||
"Password cannot be empty": "",
|
"Delete playlist": "",
|
||||||
"Password cannot be longer than 55 characters": "",
|
"Create playlist": "",
|
||||||
"Please sign in": "",
|
"Title": "",
|
||||||
"Invidious Private Feed for `x`": "",
|
"Playlist privacy": "",
|
||||||
"channel:`x`": "",
|
"Editing playlist `x`": "",
|
||||||
"Deleted or invalid channel": "",
|
"Watch on YouTube": "",
|
||||||
"This channel does not exist.": "",
|
"Hide annotations": "",
|
||||||
"Could not get channel info.": "",
|
"Show annotations": "",
|
||||||
"Could not fetch comments": "",
|
"Genre: ": "",
|
||||||
"View `x` replies": "",
|
"License: ": "",
|
||||||
"`x` ago": "",
|
"Family friendly? ": "",
|
||||||
"Load more": "",
|
"Wilson score: ": "",
|
||||||
"`x` points": "",
|
"Engagement: ": "",
|
||||||
"Could not create mix.": "",
|
"Whitelisted regions: ": "",
|
||||||
"Playlist is empty": "",
|
"Blacklisted regions: ": "",
|
||||||
"Invalid playlist.": "",
|
"Shared `x`": "",
|
||||||
"Playlist does not exist.": "",
|
"`x` views": "",
|
||||||
"Could not pull trending pages.": "",
|
"Premieres in `x`": "",
|
||||||
"Hidden field \"challenge\" is a required field": "",
|
"Premieres `x`": "",
|
||||||
"Hidden field \"token\" is a required field": "",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "",
|
||||||
"Invalid challenge": "",
|
"View YouTube comments": "",
|
||||||
"Invalid token": "",
|
"View more comments on Reddit": "",
|
||||||
"Invalid user": "",
|
"View `x` comments": "",
|
||||||
"Token is expired, please try again": "",
|
"View Reddit comments": "",
|
||||||
"English": "",
|
"Hide replies": "",
|
||||||
"English (auto-generated)": "",
|
"Show replies": "",
|
||||||
"Afrikaans": "",
|
"Incorrect password": "",
|
||||||
"Albanian": "",
|
"Quota exceeded, try again in a few hours": "",
|
||||||
"Amharic": "",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "",
|
||||||
"Arabic": "",
|
"Invalid TFA code": "",
|
||||||
"Armenian": "",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "",
|
||||||
"Azerbaijani": "",
|
"Wrong answer": "",
|
||||||
"Bangla": "",
|
"Erroneous CAPTCHA": "",
|
||||||
"Basque": "",
|
"CAPTCHA is a required field": "",
|
||||||
"Belarusian": "",
|
"User ID is a required field": "",
|
||||||
"Bosnian": "",
|
"Password is a required field": "",
|
||||||
"Bulgarian": "",
|
"Wrong username or password": "",
|
||||||
"Burmese": "",
|
"Please sign in using 'Log in with Google'": "",
|
||||||
"Catalan": "",
|
"Password cannot be empty": "",
|
||||||
"Cebuano": "",
|
"Password cannot be longer than 55 characters": "",
|
||||||
"Chinese (Simplified)": "",
|
"Please log in": "",
|
||||||
"Chinese (Traditional)": "",
|
"Invidious Private Feed for `x`": "",
|
||||||
"Corsican": "",
|
"channel:`x`": "",
|
||||||
"Croatian": "",
|
"Deleted or invalid channel": "",
|
||||||
"Czech": "",
|
"This channel does not exist.": "",
|
||||||
"Danish": "",
|
"Could not get channel info.": "",
|
||||||
"Dutch": "",
|
"Could not fetch comments": "",
|
||||||
"Esperanto": "",
|
"View `x` replies": "",
|
||||||
"Estonian": "",
|
"`x` ago": "",
|
||||||
"Filipino": "",
|
"Load more": "",
|
||||||
"Finnish": "",
|
"`x` points": "",
|
||||||
"French": "",
|
"Could not create mix.": "",
|
||||||
"Galician": "",
|
"Empty playlist": "",
|
||||||
"Georgian": "",
|
"Not a playlist.": "",
|
||||||
"German": "",
|
"Playlist does not exist.": "",
|
||||||
"Greek": "",
|
"Could not pull trending pages.": "",
|
||||||
"Gujarati": "",
|
"Hidden field \"challenge\" is a required field": "",
|
||||||
"Haitian Creole": "",
|
"Hidden field \"token\" is a required field": "",
|
||||||
"Hausa": "",
|
"Erroneous challenge": "",
|
||||||
"Hawaiian": "",
|
"Erroneous token": "",
|
||||||
"Hebrew": "",
|
"No such user": "",
|
||||||
"Hindi": "",
|
"Token is expired, please try again": "",
|
||||||
"Hmong": "",
|
"English": "",
|
||||||
"Hungarian": "",
|
"English (auto-generated)": "",
|
||||||
"Icelandic": "",
|
"Afrikaans": "",
|
||||||
"Igbo": "",
|
"Albanian": "",
|
||||||
"Indonesian": "",
|
"Amharic": "",
|
||||||
"Irish": "",
|
"Arabic": "",
|
||||||
"Italian": "",
|
"Armenian": "",
|
||||||
"Japanese": "",
|
"Azerbaijani": "",
|
||||||
"Javanese": "",
|
"Bangla": "",
|
||||||
"Kannada": "",
|
"Basque": "",
|
||||||
"Kazakh": "",
|
"Belarusian": "",
|
||||||
"Khmer": "",
|
"Bosnian": "",
|
||||||
"Korean": "",
|
"Bulgarian": "",
|
||||||
"Kurdish": "",
|
"Burmese": "",
|
||||||
"Kyrgyz": "",
|
"Catalan": "",
|
||||||
"Lao": "",
|
"Cebuano": "",
|
||||||
"Latin": "",
|
"Chinese (Simplified)": "",
|
||||||
"Latvian": "",
|
"Chinese (Traditional)": "",
|
||||||
"Lithuanian": "",
|
"Corsican": "",
|
||||||
"Luxembourgish": "",
|
"Croatian": "",
|
||||||
"Macedonian": "",
|
"Czech": "",
|
||||||
"Malagasy": "",
|
"Danish": "",
|
||||||
"Malay": "",
|
"Dutch": "",
|
||||||
"Malayalam": "",
|
"Esperanto": "",
|
||||||
"Maltese": "",
|
"Estonian": "",
|
||||||
"Maori": "",
|
"Filipino": "",
|
||||||
"Marathi": "",
|
"Finnish": "",
|
||||||
"Mongolian": "",
|
"French": "",
|
||||||
"Nepali": "",
|
"Galician": "",
|
||||||
"Norwegian": "",
|
"Georgian": "",
|
||||||
"Nyanja": "",
|
"German": "",
|
||||||
"Pashto": "",
|
"Greek": "",
|
||||||
"Persian": "",
|
"Gujarati": "",
|
||||||
"Polish": "",
|
"Haitian Creole": "",
|
||||||
"Portuguese": "",
|
"Hausa": "",
|
||||||
"Punjabi": "",
|
"Hawaiian": "",
|
||||||
"Romanian": "",
|
"Hebrew": "",
|
||||||
"Russian": "",
|
"Hindi": "",
|
||||||
"Samoan": "",
|
"Hmong": "",
|
||||||
"Scottish Gaelic": "",
|
"Hungarian": "",
|
||||||
"Serbian": "",
|
"Icelandic": "",
|
||||||
"Shona": "",
|
"Igbo": "",
|
||||||
"Sindhi": "",
|
"Indonesian": "",
|
||||||
"Sinhala": "",
|
"Irish": "",
|
||||||
"Slovak": "",
|
"Italian": "",
|
||||||
"Slovenian": "",
|
"Japanese": "",
|
||||||
"Somali": "",
|
"Javanese": "",
|
||||||
"Southern Sotho": "",
|
"Kannada": "",
|
||||||
"Spanish": "",
|
"Kazakh": "",
|
||||||
"Spanish (Latin America)": "",
|
"Khmer": "",
|
||||||
"Sundanese": "",
|
"Korean": "",
|
||||||
"Swahili": "",
|
"Kurdish": "",
|
||||||
"Swedish": "",
|
"Kyrgyz": "",
|
||||||
"Tajik": "",
|
"Lao": "",
|
||||||
"Tamil": "",
|
"Latin": "",
|
||||||
"Telugu": "",
|
"Latvian": "",
|
||||||
"Thai": "",
|
"Lithuanian": "",
|
||||||
"Turkish": "",
|
"Luxembourgish": "",
|
||||||
"Ukrainian": "",
|
"Macedonian": "",
|
||||||
"Urdu": "",
|
"Malagasy": "",
|
||||||
"Uzbek": "",
|
"Malay": "",
|
||||||
"Vietnamese": "",
|
"Malayalam": "",
|
||||||
"Welsh": "",
|
"Maltese": "",
|
||||||
"Western Frisian": "",
|
"Maori": "",
|
||||||
"Xhosa": "",
|
"Marathi": "",
|
||||||
"Yiddish": "",
|
"Mongolian": "",
|
||||||
"Yoruba": "",
|
"Nepali": "",
|
||||||
"Zulu": "",
|
"Norwegian Bokmål": "",
|
||||||
"`x` years": "",
|
"Nyanja": "",
|
||||||
"`x` months": "",
|
"Pashto": "",
|
||||||
"`x` weeks": "",
|
"Persian": "",
|
||||||
"`x` days": "",
|
"Polish": "",
|
||||||
"`x` hours": "",
|
"Portuguese": "",
|
||||||
"`x` minutes": "",
|
"Punjabi": "",
|
||||||
"`x` seconds": "",
|
"Romanian": "",
|
||||||
"Fallback comments: ": "",
|
"Russian": "",
|
||||||
"Popular": "",
|
"Samoan": "",
|
||||||
"Top": "",
|
"Scottish Gaelic": "",
|
||||||
"About": "",
|
"Serbian": "",
|
||||||
"Rating: ": "",
|
"Shona": "",
|
||||||
"Language: ": "",
|
"Sindhi": "",
|
||||||
"Default": "",
|
"Sinhala": "",
|
||||||
"Music": "",
|
"Slovak": "",
|
||||||
"Gaming": "",
|
"Slovenian": "",
|
||||||
"News": "",
|
"Somali": "",
|
||||||
"Movies": "",
|
"Southern Sotho": "",
|
||||||
"Download": "",
|
"Spanish": "",
|
||||||
"Download as: ": "",
|
"Spanish (Latin America)": "",
|
||||||
"%A %B %-d, %Y": "",
|
"Sundanese": "",
|
||||||
"(edited)": "",
|
"Swahili": "",
|
||||||
"Youtube permalink of the comment": "",
|
"Swedish": "",
|
||||||
"`x` marked it with a ❤": "",
|
"Tajik": "",
|
||||||
"Audio mode": "",
|
"Tamil": "",
|
||||||
"Video mode": "",
|
"Telugu": "",
|
||||||
"Videos": "",
|
"Thai": "",
|
||||||
"Playlists": "",
|
"Turkish": "",
|
||||||
"Current version: ": ""
|
"Ukrainian": "",
|
||||||
}
|
"Urdu": "",
|
||||||
|
"Uzbek": "",
|
||||||
|
"Vietnamese": "",
|
||||||
|
"Welsh": "",
|
||||||
|
"Western Frisian": "",
|
||||||
|
"Xhosa": "",
|
||||||
|
"Yiddish": "",
|
||||||
|
"Yoruba": "",
|
||||||
|
"Zulu": "",
|
||||||
|
"`x` years": "",
|
||||||
|
"`x` months": "",
|
||||||
|
"`x` weeks": "",
|
||||||
|
"`x` days": "",
|
||||||
|
"`x` hours": "",
|
||||||
|
"`x` minutes": "",
|
||||||
|
"`x` seconds": "",
|
||||||
|
"Fallback comments: ": "",
|
||||||
|
"Popular": "",
|
||||||
|
"Top": "",
|
||||||
|
"About": "",
|
||||||
|
"Rating: ": "",
|
||||||
|
"Language: ": "",
|
||||||
|
"View as playlist": "",
|
||||||
|
"Default": "",
|
||||||
|
"Music": "",
|
||||||
|
"Gaming": "",
|
||||||
|
"News": "",
|
||||||
|
"Movies": "",
|
||||||
|
"Download": "",
|
||||||
|
"Download as: ": "",
|
||||||
|
"%A %B %-d, %Y": "",
|
||||||
|
"(edited)": "",
|
||||||
|
"YouTube comment permalink": "",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "",
|
||||||
|
"Audio mode": "",
|
||||||
|
"Video mode": "",
|
||||||
|
"Videos": "",
|
||||||
|
"Playlists": "",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": ""
|
||||||
|
}
|
||||||
627
locales/fr.json
627
locales/fr.json
@@ -1,295 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` abonnés",
|
"`x` subscribers": "`x` abonnés",
|
||||||
"`x` videos": "`x` vidéos",
|
"`x` videos": "`x` vidéos",
|
||||||
"LIVE": "EN DIRECT",
|
"`x` playlists": "`x` listes de lecture",
|
||||||
"Shared `x` ago": "Ajoutée il y a `x`",
|
"LIVE": "EN DIRECT",
|
||||||
"Unsubscribe": "Se désabonner",
|
"Shared `x` ago": "Ajoutée il y a `x`",
|
||||||
"Subscribe": "S'abonner",
|
"Unsubscribe": "Se désabonner",
|
||||||
"Login to subscribe to `x`": "Vous devez vous connecter pour vous abonner à `x`",
|
"Subscribe": "S'abonner",
|
||||||
"View channel on YouTube": "Voir la chaîne sur YouTube",
|
"View channel on YouTube": "Voir la chaîne sur YouTube",
|
||||||
"newest": "Date d'ajout (la plus récente)",
|
"View playlist on YouTube": "Voir la liste de lecture sur YouTube",
|
||||||
"oldest": "Date d'ajout (la plus ancienne)",
|
"newest": "Date d'ajout (la plus récente)",
|
||||||
"popular": "Les plus populaires",
|
"oldest": "Date d'ajout (la plus ancienne)",
|
||||||
"last": "Dernières",
|
"popular": "Les plus populaires",
|
||||||
"Next page": "Page suivante",
|
"last": "Dernières",
|
||||||
"Previous page": "Page précédente",
|
"Next page": "Page suivante",
|
||||||
"Clear watch history?": "Êtes-vous sûr de vouloir supprimer l'historique des vidéos regardées ?",
|
"Previous page": "Page précédente",
|
||||||
"Yes": "Oui",
|
"Clear watch history?": "Êtes-vous sûr de vouloir supprimer l'historique des vidéos regardées ?",
|
||||||
"No": "Non",
|
"New password": "Nouveau mot de passe",
|
||||||
"Import and Export Data": "Importer et exporter des données",
|
"New passwords must match": "Les champs \"Nouveau mot de passe\" doivent être identiques",
|
||||||
"Import": "Importer",
|
"Cannot change password for Google accounts": "Le mot de passe d'un compte Google ne peut pas être changé depuis Invidious",
|
||||||
"Import Invidious data": "Importer des données Invidious",
|
"Authorize token?": "Autoriser le token ?",
|
||||||
"Import YouTube subscriptions": "Importer des abonnements YouTube",
|
"Authorize token for `x`?": "Autoriser le token pour `x` ?",
|
||||||
"Import FreeTube subscriptions (.db)": "Importer des abonnements FreeTube (.db)",
|
"Yes": "Oui",
|
||||||
"Import NewPipe subscriptions (.json)": "Importer des abonnements NewPipe (.json)",
|
"No": "Non",
|
||||||
"Import NewPipe data (.zip)": "Importer des données NewPipe (.zip)",
|
"Import and Export Data": "Importer et exporter des données",
|
||||||
"Export": "Exporter",
|
"Import": "Importer",
|
||||||
"Export subscriptions as OPML": "Exporter les abonnements en OPML",
|
"Import Invidious data": "Importer des données Invidious",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exporter les abonnements en OPML (pour NewPipe & FreeTube)",
|
"Import YouTube subscriptions": "Importer des abonnements YouTube",
|
||||||
"Export data as JSON": "Exporter les données au format JSON",
|
"Import FreeTube subscriptions (.db)": "Importer des abonnements FreeTube (.db)",
|
||||||
"Delete account?": "Êtes-vous sûr de vouloir supprimer votre compte ?",
|
"Import NewPipe subscriptions (.json)": "Importer des abonnements NewPipe (.json)",
|
||||||
"History": "Historique",
|
"Import NewPipe data (.zip)": "Importer des données NewPipe (.zip)",
|
||||||
"An alternative front-end to YouTube": "Un front-end alternatif à YouTube",
|
"Export": "Exporter",
|
||||||
"JavaScript license information": "Informations sur les licences JavaScript",
|
"Export subscriptions as OPML": "Exporter les abonnements au format OPML",
|
||||||
"source": "source",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exporter les abonnements au format OPML (pour NewPipe & FreeTube)",
|
||||||
"Login": "Se connecter",
|
"Export data as JSON": "Exporter les données au format JSON",
|
||||||
"Login/Register": "Se connecter/Créer un compte",
|
"Delete account?": "Êtes-vous sûr de vouloir supprimer votre compte ?",
|
||||||
"Login to Google": "Se connecter avec Google",
|
"History": "Historique",
|
||||||
"User ID:": "Identifiant utilisateur :",
|
"An alternative front-end to YouTube": "Un front-end alternatif à YouTube",
|
||||||
"Password:": "Mot de passe :",
|
"JavaScript license information": "Informations sur les licences JavaScript",
|
||||||
"Time (h:mm:ss):": "Heure (h:mm:ss) :",
|
"source": "source",
|
||||||
"Text CAPTCHA": "CAPTCHA Texte",
|
"Log in": "Se connecter",
|
||||||
"Image CAPTCHA": "CAPTCHA Image",
|
"Log in/register": "Se connecter/Créer un compte",
|
||||||
"Sign In": "Se connecter",
|
"Log in with Google": "Se connecter avec Google",
|
||||||
"Register": "S'inscrire",
|
"User ID": "Identifiant utilisateur",
|
||||||
"Email:": "E-mail :",
|
"Password": "Mot de passe",
|
||||||
"Google verification code:": "Code de vérification Google :",
|
"Time (h:mm:ss):": "Heure (h:mm:ss) :",
|
||||||
"Preferences": "Préférences",
|
"Text CAPTCHA": "CAPTCHA Texte",
|
||||||
"Player preferences": "Préférences du lecteur",
|
"Image CAPTCHA": "CAPTCHA Image",
|
||||||
"Always loop: ": "Lire en boucle : ",
|
"Sign In": "Se connecter",
|
||||||
"Autoplay: ": "Lire automatiquement : ",
|
"Register": "S'inscrire",
|
||||||
"Autoplay next video: ": "Lire automatiquement la vidéo suivante : ",
|
"E-mail": "E-mail",
|
||||||
"Listen by default: ": "Audio uniquement : ",
|
"Google verification code": "Code de vérification Google",
|
||||||
"Proxy videos? ": "Charger les vidéos à travers un proxy ? ",
|
"Preferences": "Préférences",
|
||||||
"Default speed: ": "Vitesse par défaut : ",
|
"Player preferences": "Préférences du lecteur",
|
||||||
"Preferred video quality: ": "Qualité vidéo souhaitée : ",
|
"Always loop: ": "Lire en boucle : ",
|
||||||
"Player volume: ": "Volume du lecteur : ",
|
"Autoplay: ": "Lancer la lecture automatiquement : ",
|
||||||
"Default comments: ": "Source des commentaires : ",
|
"Play next by default: ": "Lire les vidéos suivantes par défaut : ",
|
||||||
"Default captions: ": "Sous-titres par défaut : ",
|
"Autoplay next video: ": "Lancer la lecture automatiquement pour la vidéo suivant la vidéo regardée : ",
|
||||||
"Fallback captions: ": "Fallback captions: ",
|
"Listen by default: ": "Audio uniquement : ",
|
||||||
"Show related videos? ": "Voir les vidéos liées ? ",
|
"Proxy videos: ": "Charger les vidéos à travers un proxy : ",
|
||||||
"Visual preferences": "Préférences du site",
|
"Default speed: ": "Vitesse par défaut : ",
|
||||||
"Dark mode: ": "Mode Sombre : ",
|
"Preferred video quality: ": "Qualité vidéo souhaitée : ",
|
||||||
"Thin mode: ": "Mode Simplifié : ",
|
"Player volume: ": "Volume du lecteur : ",
|
||||||
"Subscription preferences": "Préférences de la page d'abonnements",
|
"Default comments: ": "Source des commentaires : ",
|
||||||
"Redirect homepage to feed: ": "Rediriger la page d'accueil vers la page d'abonnements : ",
|
"youtube": "YouTube",
|
||||||
"Number of videos shown in feed: ": "Nombre de vidéos montrées dans la page d'abonnements : ",
|
"reddit": "Reddit",
|
||||||
"Sort videos by: ": "Trier les vidéos par : ",
|
"Default captions: ": "Sous-titres par défaut : ",
|
||||||
"published": "publication",
|
"Fallback captions: ": "Sous-titres alternatifs : ",
|
||||||
"published - reverse": "publication - inversé",
|
"Show related videos: ": "Voir les vidéos liées : ",
|
||||||
"alphabetically": "alphabétiquement",
|
"Show annotations by default: ": "Afficher les annotations par défaut : ",
|
||||||
"alphabetically - reverse": "alphabétiquement - inversé",
|
"Visual preferences": "Préférences du site",
|
||||||
"channel name": "nom de la chaîne",
|
"Player style: ": "Style du lecteur : ",
|
||||||
"channel name - reverse": "nom de la chaîne - inversé",
|
"Dark mode: ": "Mode sombre : ",
|
||||||
"Only show latest video from channel: ": "Afficher uniquement la dernière vidéo de la chaîne : ",
|
"Theme: ": "Thème : ",
|
||||||
"Only show latest unwatched video from channel: ": "Afficher uniquement la dernière vidéo de la chaîne non regardée : ",
|
"dark": "sombre",
|
||||||
"Only show unwatched: ": "Afficher uniquement les vidéos non regardées : ",
|
"light": "clair",
|
||||||
"Only show notifications (if there are any): ": "Afficher uniquement les notifications (s'il y en a) : ",
|
"Thin mode: ": "Mode léger : ",
|
||||||
"Data preferences": "Préférences liées aux données",
|
"Subscription preferences": "Préférences de la page d'abonnements",
|
||||||
"Clear watch history": "Supprimer l'historique des vidéos regardées",
|
"Show annotations by default for subscribed channels: ": "Afficher les annotations par défaut sur les chaînes auxquelles vous êtes abonnés : ",
|
||||||
"Import/Export data": "Importer/exporter les données",
|
"Redirect homepage to feed: ": "Rediriger la page d'accueil vers la page d'abonnements : ",
|
||||||
"Manage subscriptions": "Gérer les abonnements",
|
"Number of videos shown in feed: ": "Nombre de vidéos affichées dans la page d'abonnements : ",
|
||||||
"Watch history": "Historique de visionnage",
|
"Sort videos by: ": "Trier les vidéos par : ",
|
||||||
"Delete account": "Supprimer votre compte",
|
"published": "date de publication",
|
||||||
"Administrator preferences": "Préferences d'Administrateur",
|
"published - reverse": "date de publication - inversé",
|
||||||
"Default homepage: ": "Page d'accueil par défaut : ",
|
"alphabetically": "alphabétiquement",
|
||||||
"Feed menu: ": "Menu des Flux : ",
|
"alphabetically - reverse": "alphabétiquement - inversé",
|
||||||
"Top enabled? ": "Top activé ? ",
|
"channel name": "nom de la chaîne",
|
||||||
"CAPTCHA enabled? ": "CAPTCHA activé ? ",
|
"channel name - reverse": "nom de la chaîne - inversé",
|
||||||
"Login enabled? ": "Connexion activé ? ",
|
"Only show latest video from channel: ": "Afficher uniquement la dernière vidéo des chaînes auxquelles vous êtes abonnés : ",
|
||||||
"Registration enabled? ": "Inscription activée ? ",
|
"Only show latest unwatched video from channel: ": "Afficher uniquement la dernière vidéo des chaînes auxquelles vous êtes abonnés qui n'a pas était regardée : ",
|
||||||
"Report statistics? ": "Télémétrie activé ? ",
|
"Only show unwatched: ": "Afficher uniquement les vidéos qui n'ont pas étaient regardées : ",
|
||||||
"Save preferences": "Enregistrer les préférences",
|
"Only show notifications (if there are any): ": "Afficher uniquement les notifications (s'il y en a) : ",
|
||||||
"Subscription manager": "Gestionnaire d'abonnement",
|
"Enable web notifications": "Activer les notifications web",
|
||||||
"`x` subscriptions": "`x` abonnements",
|
"`x` uploaded a video": "`x` a partagé(e) une vidéo",
|
||||||
"Import/Export": "Importer/Exporter",
|
"`x` is live": "`x` est en direct",
|
||||||
"unsubscribe": "se désabonner",
|
"Data preferences": "Préférences liées aux données",
|
||||||
"Subscriptions": "Abonnements",
|
"Clear watch history": "Supprimer l'historique des vidéos regardées",
|
||||||
"`x` unseen notifications": "`x` notifications non vues",
|
"Import/export data": "Importer/exporter les données",
|
||||||
"search": "Rechercher",
|
"Change password": "Modifier le mot de passe",
|
||||||
"Sign out": "Déconnexion",
|
"Manage subscriptions": "Gérer les abonnements",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Publié sous licence AGPLv3 par Omar Roth.",
|
"Manage tokens": "Gérer les tokens",
|
||||||
"Source available here.": "Code Source.",
|
"Watch history": "Historique de visionnage",
|
||||||
"View JavaScript license information.": "Voir les informations des licences JavaScript.",
|
"Delete account": "Supprimer votre compte",
|
||||||
"View privacy policy.": "Politique de confidentialité",
|
"Administrator preferences": "Préferences d'Administration",
|
||||||
"Trending": "Tendances",
|
"Default homepage: ": "Page d'accueil par défaut : ",
|
||||||
"Unlisted": "Non répertoriée",
|
"Feed menu: ": "Préferences des abonnements : ",
|
||||||
"Watch video on Youtube": "Voir la vidéo sur Youtube",
|
"Top enabled: ": "Top activé : ",
|
||||||
"Genre: ": "Genre : ",
|
"CAPTCHA enabled: ": "CAPTCHA activé : ",
|
||||||
"License: ": "Licence : ",
|
"Login enabled: ": "Connexion activé : ",
|
||||||
"Family friendly? ": "Tout Public ? ",
|
"Registration enabled: ": "Inscription activée : ",
|
||||||
"Wilson score: ": "Score de Wilson : ",
|
"Report statistics: ": "Télémétrie activé : ",
|
||||||
"Engagement: ": "Poucentage de spectateur aillant aimé Like ou Dislike la vidéo : ",
|
"Save preferences": "Enregistrer les préférences",
|
||||||
"Whitelisted regions: ": "Régions en liste blanche : ",
|
"Subscription manager": "Gestionnaire d'abonnement",
|
||||||
"Blacklisted regions: ": "Régions sur liste noire : ",
|
"Token manager": "Gestionnaire de tokens",
|
||||||
"Shared `x`": "Ajoutée le `x`",
|
"Token": "Token",
|
||||||
"Premieres in `x`": "Première dans `x`",
|
"`x` subscriptions": "`x` abonnements",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Il semblerait que JavaScript soit désactivé. Cliquez ici pour voir les commentaires sans. Gardez à l'esprit que le chargement peut prendre plus de temps.",
|
"`x` tokens": "`x` tokens",
|
||||||
"View YouTube comments": "Voir les commentaires YouTube",
|
"Import/export": "Importer/Exporter",
|
||||||
"View more comments on Reddit": "Voir plus de commentaires sur Reddit",
|
"unsubscribe": "se désabonner",
|
||||||
"View `x` comments": "Voir `x` commentaires",
|
"revoke": "révoquer",
|
||||||
"View Reddit comments": "Voir les commentaires Reddit",
|
"Subscriptions": "Abonnements",
|
||||||
"Hide replies": "Masquer les réponses",
|
"`x` unseen notifications": "`x` notifications non vues",
|
||||||
"Show replies": "Afficher les réponses",
|
"search": "rechercher",
|
||||||
"Incorrect password": "Mot de passe incorrect",
|
"Log out": "Déconnexion",
|
||||||
"Quota exceeded, try again in a few hours": "Nombre de tentative de connexion dépassé, réessayez dans quelques heures",
|
"Released under the AGPLv3 by Omar Roth.": "Publié sous licence AGPLv3 par Omar Roth.",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Si vous ne parvenez pas à vous connecter, assurez-vous que l'authentification à deux facteurs (Authenticator ou SMS) est activée.",
|
"Source available here.": "Code source disponible ici.",
|
||||||
"Invalid TFA code": "Code d'authentification à deux facteurs invalide",
|
"View JavaScript license information.": "Informations des licences JavaScript.",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "La connexion a échoué. Cela peut être dû au fait que l'authentification à deux facteurs n'est pas activée sur votre compte.",
|
"View privacy policy.": "Politique de confidentialité.",
|
||||||
"Invalid answer": "Réponse invalide",
|
"Trending": "Tendances",
|
||||||
"Invalid CAPTCHA": "CAPTCHA invalide",
|
"Public": "Publique",
|
||||||
"CAPTCHA is a required field": "Veuillez entrer un CAPTCHA",
|
"Unlisted": "Non répertoriée",
|
||||||
"User ID is a required field": "Veuillez entrer un Identifiant Utilisateur",
|
"Private": "Privée",
|
||||||
"Password is a required field": "Veuillez entrer un Mot de passe",
|
"View all playlists": "Voir toutes vos playlists",
|
||||||
"Invalid username or password": "Nom d'utilisateur ou mot de passe invalide",
|
"Updated `x` ago": "Dernière mise à jour il y a `x`",
|
||||||
"Please sign in using 'Sign in with Google'": "Veuillez vous connecter en utilisant \"Se connecter avec Google\"",
|
"Delete playlist `x`?": "Êtes-vous sûr de vouloir supprimer la liste de lecture ?",
|
||||||
"Password cannot be empty": "Le mot de passe ne peut pas être vide",
|
"Delete playlist": "Supprimer la liste de lecture",
|
||||||
"Password cannot be longer than 55 characters": "Le mot de passe ne doit pas comporter plus de 55 caractères",
|
"Create playlist": "Créer une liste de lecture",
|
||||||
"Please sign in": "Veuillez vous connecter",
|
"Title": "Titre",
|
||||||
"Invidious Private Feed for `x`": "Flux RSS privé pour `x`",
|
"Playlist privacy": "Paramètres de confidentialité de la liste de lecture",
|
||||||
"channel:`x`": "chaîne:`x`",
|
"Editing playlist `x`": "Liste de lecture modifier le `x`",
|
||||||
"Deleted or invalid channel": "Chaîne supprimée ou invalide",
|
"Watch on YouTube": "Voir la vidéo sur Youtube",
|
||||||
"This channel does not exist.": "Cette chaine n'existe pas.",
|
"Hide annotations": "Masquer les annotations",
|
||||||
"Could not get channel info.": "Impossible de charger les informations de cette chaîne.",
|
"Show annotations": "Afficher les annotations",
|
||||||
"Could not fetch comments": "Impossible de charger les commentaires",
|
"Genre: ": "Genre : ",
|
||||||
"View `x` replies": "Voir `x` réponses",
|
"License: ": "Licence : ",
|
||||||
"`x` ago": "il y a `x`",
|
"Family friendly? ": "Vidéo tout public ? ",
|
||||||
"Load more": "Charger plus",
|
"Wilson score: ": "Score de Wilson : ",
|
||||||
"`x` points": "`x` points",
|
"Engagement: ": "Pourcentage de spectateur aillant appuyé sur \"J'aime\" ou \"J'aime Pas\" : ",
|
||||||
"Could not create mix.": "Impossible de charger cette liste de lecture.",
|
"Whitelisted regions: ": "Régions sur liste blanche : ",
|
||||||
"Playlist is empty": "La liste de lecture est vide",
|
"Blacklisted regions: ": "Régions sur liste noire : ",
|
||||||
"Invalid playlist.": "Liste de lecture invalide.",
|
"Shared `x`": "Ajoutée le `x`",
|
||||||
"Playlist does not exist.": "La liste de lecture n'existe pas.",
|
"`x` views": "`x` vues",
|
||||||
"Could not pull trending pages.": "Impossible de charger les pages de tendances.",
|
"Premieres in `x`": "Première dans `x`",
|
||||||
"Hidden field \"challenge\" is a required field": "Hidden field \"challenge\" is a required field",
|
"Premieres `x`": "Première le `x`",
|
||||||
"Hidden field \"token\" is a required field": "Hidden field \"token\" is a required field",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Il semblerait que JavaScript soit désactivé. Cliquez ici pour voir les commentaires sans. Gardez à l'esprit que le chargement peut prendre plus de temps.",
|
||||||
"Invalid challenge": "Invalid challenge",
|
"View YouTube comments": "Voir les commentaires YouTube",
|
||||||
"Invalid token": "Invalid token",
|
"View more comments on Reddit": "Voir plus de commentaires sur Reddit",
|
||||||
"Invalid user": "Invalid user",
|
"View `x` comments": "Voir `x` commentaires",
|
||||||
"Token is expired, please try again": "Token is expired, please try again",
|
"View Reddit comments": "Voir les commentaires Reddit",
|
||||||
"English": "Anglais",
|
"Hide replies": "Masquer les réponses",
|
||||||
"English (auto-generated)": "Anglais (générés automatiquement)",
|
"Show replies": "Afficher les réponses",
|
||||||
"Afrikaans": "Afrikaans",
|
"Incorrect password": "Mot de passe incorrect",
|
||||||
"Albanian": "Albanais",
|
"Quota exceeded, try again in a few hours": "Nombre de tentative de connexion dépassée, réessayez dans quelques heures",
|
||||||
"Amharic": "Amharique",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Impossible de se connecter, si après plusieurs tentative vous ne parvenez toujours pas à vous connecter, assurez-vous que l'authentification à deux facteurs (Authenticator ou SMS) est activée.",
|
||||||
"Arabic": "Arabe",
|
"Invalid TFA code": "Code d'authentification à deux facteurs invalide",
|
||||||
"Armenian": "Arménien",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "La connexion a échoué. Cela peut être dû au fait que l'authentification à deux facteurs n'est pas activée sur votre compte.",
|
||||||
"Azerbaijani": "Azerbaïdjanais",
|
"Wrong answer": "Réponse invalide",
|
||||||
"Bangla": "Bangla",
|
"Erroneous CAPTCHA": "CAPTCHA invalide",
|
||||||
"Basque": "Basque",
|
"CAPTCHA is a required field": "Veuillez entrer un CAPTCHA",
|
||||||
"Belarusian": "Belarusian",
|
"User ID is a required field": "Veuillez entrer un Identifiant Utilisateur",
|
||||||
"Bosnian": "Bosnian",
|
"Password is a required field": "Veuillez entrer un Mot de passe",
|
||||||
"Bulgarian": "Bulgarian",
|
"Wrong username or password": "Nom d'utilisateur ou mot de passe invalide",
|
||||||
"Burmese": "Birman",
|
"Please sign in using 'Log in with Google'": "Veuillez vous connecter en utilisant \"Se connecter avec Google\"",
|
||||||
"Catalan": "Catalan",
|
"Password cannot be empty": "Le mot de passe ne peut pas être vide",
|
||||||
"Cebuano": "Cebuano",
|
"Password cannot be longer than 55 characters": "Le mot de passe ne doit pas comporter plus de 55 caractères",
|
||||||
"Chinese (Simplified)": "Chinois (Simplifié)",
|
"Please log in": "Veuillez vous connecter",
|
||||||
"Chinese (Traditional)": "Chinois (Traditionnel)",
|
"Invidious Private Feed for `x`": "Flux RSS privé pour `x`",
|
||||||
"Corsican": "Corse",
|
"channel:`x`": "chaîne:`x`",
|
||||||
"Croatian": "Croate",
|
"Deleted or invalid channel": "Chaîne supprimée ou invalide",
|
||||||
"Czech": "Tchèque",
|
"This channel does not exist.": "Cette chaine n'existe pas.",
|
||||||
"Danish": "Danois",
|
"Could not get channel info.": "Impossible de charger les informations de cette chaîne.",
|
||||||
"Dutch": "Hollandais",
|
"Could not fetch comments": "Impossible de charger les commentaires",
|
||||||
"Esperanto": "Espéranto",
|
"View `x` replies": "Voir `x` réponses",
|
||||||
"Estonian": "Estonien",
|
"`x` ago": "il y a `x`",
|
||||||
"Filipino": "Philippin",
|
"Load more": "Voir plus",
|
||||||
"Finnish": "Finlandais",
|
"`x` points": "`x` points",
|
||||||
"French": "Français",
|
"Could not create mix.": "Impossible de charger cette liste de lecture.",
|
||||||
"Galician": "Galicien",
|
"Empty playlist": "La liste de lecture est vide",
|
||||||
"Georgian": "Géorgien",
|
"Not a playlist.": "La liste de lecture est invalide.",
|
||||||
"German": "Allemand",
|
"Playlist does not exist.": "La liste de lecture n'existe pas.",
|
||||||
"Greek": "Grec",
|
"Could not pull trending pages.": "Impossible de charger les pages de tendances.",
|
||||||
"Gujarati": "Gujarati",
|
"Hidden field \"challenge\" is a required field": "Le champ masqué \"challenge\" est un champ obligatoire",
|
||||||
"Haitian Creole": "Créole Haïtien",
|
"Hidden field \"token\" is a required field": "Le champ caché \"token\" est requis",
|
||||||
"Hausa": "Haoussa",
|
"Erroneous challenge": "Challenge invalide",
|
||||||
"Hawaiian": "Hawaïen",
|
"Erroneous token": "Token invalide",
|
||||||
"Hebrew": "Hébraïque",
|
"No such user": "Cet utilisateur n'existe pas",
|
||||||
"Hindi": "Hindi",
|
"Token is expired, please try again": "Le token est expiré, veuillez réessayer",
|
||||||
"Hmong": "Hmong",
|
"English": "Anglais",
|
||||||
"Hungarian": "Hongrois",
|
"English (auto-generated)": "Anglais (générés automatiquement)",
|
||||||
"Icelandic": "Islandais",
|
"Afrikaans": "Afrikaans",
|
||||||
"Igbo": "Igbo",
|
"Albanian": "Albanais",
|
||||||
"Indonesian": "Indonésien",
|
"Amharic": "Amharique",
|
||||||
"Irish": "Irlandais",
|
"Arabic": "Arabe",
|
||||||
"Italian": "Italien",
|
"Armenian": "Arménien",
|
||||||
"Japanese": "Japonais",
|
"Azerbaijani": "Azerbaïdjanais",
|
||||||
"Javanese": "Javanais",
|
"Bangla": "Bangla",
|
||||||
"Kannada": "Kannada",
|
"Basque": "Basque",
|
||||||
"Kazakh": "Kazakh",
|
"Belarusian": "Belarusian",
|
||||||
"Khmer": "Khmer",
|
"Bosnian": "Bosnian",
|
||||||
"Korean": "Coréen",
|
"Bulgarian": "Bulgarian",
|
||||||
"Kurdish": "Kurde",
|
"Burmese": "Birman",
|
||||||
"Kyrgyz": "Kirghize",
|
"Catalan": "Catalan",
|
||||||
"Lao": "Lao",
|
"Cebuano": "Cebuano",
|
||||||
"Latin": "Latin",
|
"Chinese (Simplified)": "Chinois (Simplifié)",
|
||||||
"Latvian": "Letton",
|
"Chinese (Traditional)": "Chinois (Traditionnel)",
|
||||||
"Lithuanian": "Lituanien",
|
"Corsican": "Corse",
|
||||||
"Luxembourgish": "Luxembourgeois",
|
"Croatian": "Croate",
|
||||||
"Macedonian": "Macédonien",
|
"Czech": "Tchèque",
|
||||||
"Malagasy": "Malgache",
|
"Danish": "Danois",
|
||||||
"Malay": "Malais",
|
"Dutch": "Hollandais",
|
||||||
"Malayalam": "Malayalam",
|
"Esperanto": "Espéranto",
|
||||||
"Maltese": "Maltais",
|
"Estonian": "Estonien",
|
||||||
"Maori": "Maori",
|
"Filipino": "Philippin",
|
||||||
"Marathi": "Marathi",
|
"Finnish": "Finlandais",
|
||||||
"Mongolian": "Mongol",
|
"French": "Français",
|
||||||
"Nepali": "Népalais",
|
"Galician": "Galicien",
|
||||||
"Norwegian": "Norvégien",
|
"Georgian": "Géorgien",
|
||||||
"Nyanja": "Nyanja",
|
"German": "Allemand",
|
||||||
"Pashto": "Pachtou",
|
"Greek": "Grec",
|
||||||
"Persian": "Persan",
|
"Gujarati": "Gujarati",
|
||||||
"Polish": "Polonais",
|
"Haitian Creole": "Créole Haïtien",
|
||||||
"Portuguese": "Portugais",
|
"Hausa": "Haoussa",
|
||||||
"Punjabi": "Punjabi",
|
"Hawaiian": "Hawaïen",
|
||||||
"Romanian": "Roumain",
|
"Hebrew": "Hébraïque",
|
||||||
"Russian": "Russe",
|
"Hindi": "Hindi",
|
||||||
"Samoan": "Samoan",
|
"Hmong": "Hmong",
|
||||||
"Scottish Gaelic": "Eaélique Ècossais",
|
"Hungarian": "Hongrois",
|
||||||
"Serbian": "Serbe",
|
"Icelandic": "Islandais",
|
||||||
"Shona": "Shona",
|
"Igbo": "Igbo",
|
||||||
"Sindhi": "Sindhi",
|
"Indonesian": "Indonésien",
|
||||||
"Sinhala": "Cinghalais",
|
"Irish": "Irlandais",
|
||||||
"Slovak": "Slovaque",
|
"Italian": "Italien",
|
||||||
"Slovenian": "Slovène",
|
"Japanese": "Japonais",
|
||||||
"Somali": "Somalien",
|
"Javanese": "Javanais",
|
||||||
"Southern Sotho": "Sotho du Sud",
|
"Kannada": "Kannada",
|
||||||
"Spanish": "Espagnol",
|
"Kazakh": "Kazakh",
|
||||||
"Spanish (Latin America)": "Espagnol (Amérique latine)",
|
"Khmer": "Khmer",
|
||||||
"Sundanese": "Sundanais",
|
"Korean": "Coréen",
|
||||||
"Swahili": "Swahili",
|
"Kurdish": "Kurde",
|
||||||
"Swedish": "Suédois",
|
"Kyrgyz": "Kirghize",
|
||||||
"Tajik": "Tajik",
|
"Lao": "Lao",
|
||||||
"Tamil": "Tamil",
|
"Latin": "Latin",
|
||||||
"Telugu": "Telugu",
|
"Latvian": "Letton",
|
||||||
"Thai": "Thaï",
|
"Lithuanian": "Lituanien",
|
||||||
"Turkish": "Turc",
|
"Luxembourgish": "Luxembourgeois",
|
||||||
"Ukrainian": "Ukrainien",
|
"Macedonian": "Macédonien",
|
||||||
"Urdu": "Ourdou",
|
"Malagasy": "Malgache",
|
||||||
"Uzbek": "Ouzbek",
|
"Malay": "Malais",
|
||||||
"Vietnamese": "Vietnamien",
|
"Malayalam": "Malayalam",
|
||||||
"Welsh": "Gallois",
|
"Maltese": "Maltais",
|
||||||
"Western Frisian": "Frison occidental",
|
"Maori": "Maori",
|
||||||
"Xhosa": "Xhosa",
|
"Marathi": "Marathi",
|
||||||
"Yiddish": "Yiddish",
|
"Mongolian": "Mongol",
|
||||||
"Yoruba": "Yoruba",
|
"Nepali": "Népalais",
|
||||||
"Zulu": "Zoulou",
|
"Norwegian Bokmål": "Norvégien",
|
||||||
"`x` years": "`x` ans",
|
"Nyanja": "Nyanja",
|
||||||
"`x` months": "`x` mois",
|
"Pashto": "Pachtou",
|
||||||
"`x` weeks": "`x` semaines",
|
"Persian": "Persan",
|
||||||
"`x` days": "`x` jours",
|
"Polish": "Polonais",
|
||||||
"`x` hours": "`x` heures",
|
"Portuguese": "Portugais",
|
||||||
"`x` minutes": "`x` minutes",
|
"Punjabi": "Punjabi",
|
||||||
"`x` seconds": "`x` secondes",
|
"Romanian": "Roumain",
|
||||||
"Fallback comments: ": "Fallback comments: ",
|
"Russian": "Russe",
|
||||||
"Popular": "Populaire",
|
"Samoan": "Samoan",
|
||||||
"Top": "Top",
|
"Scottish Gaelic": "Eaélique Ècossais",
|
||||||
"About": "A Propos",
|
"Serbian": "Serbe",
|
||||||
"Rating: ": "Évaluation : ",
|
"Shona": "Shona",
|
||||||
"Language: ": "Langue : ",
|
"Sindhi": "Sindhi",
|
||||||
"Default": "Défaut",
|
"Sinhala": "Cinghalais",
|
||||||
"Music": "Musique",
|
"Slovak": "Slovaque",
|
||||||
"Gaming": "Jeux Vidéo",
|
"Slovenian": "Slovène",
|
||||||
"News": "Actualités",
|
"Somali": "Somalien",
|
||||||
"Movies": "Films",
|
"Southern Sotho": "Sotho du Sud",
|
||||||
"Download": "Télécharger",
|
"Spanish": "Espagnol",
|
||||||
"Download as: ": "Télécharger en : ",
|
"Spanish (Latin America)": "Espagnol (Amérique latine)",
|
||||||
"%A %B %-d, %Y": "%A %-d %B %Y",
|
"Sundanese": "Sundanais",
|
||||||
"(edited)": "(modifié)",
|
"Swahili": "Swahili",
|
||||||
"Youtube permalink of the comment": "Lien YouTube permanent vers le commentaire",
|
"Swedish": "Suédois",
|
||||||
"`x` marked it with a ❤": "`x` l'a marqué d'un ❤",
|
"Tajik": "Tajik",
|
||||||
"Audio mode": "Mode Audio",
|
"Tamil": "Tamil",
|
||||||
"Video mode": "Mode Vidéo",
|
"Telugu": "Telugu",
|
||||||
"Videos": "Vidéos",
|
"Thai": "Thaï",
|
||||||
"Playlists": "Liste de lecture",
|
"Turkish": "Turc",
|
||||||
"Current version: ": "Version :"
|
"Ukrainian": "Ukrainien",
|
||||||
|
"Urdu": "Ourdou",
|
||||||
|
"Uzbek": "Ouzbek",
|
||||||
|
"Vietnamese": "Vietnamien",
|
||||||
|
"Welsh": "Gallois",
|
||||||
|
"Western Frisian": "Frison occidental",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Yiddish",
|
||||||
|
"Yoruba": "Yoruba",
|
||||||
|
"Zulu": "Zoulou",
|
||||||
|
"`x` years": "`x` ans",
|
||||||
|
"`x` months": "`x` mois",
|
||||||
|
"`x` weeks": "`x` semaines",
|
||||||
|
"`x` days": "`x` jours",
|
||||||
|
"`x` hours": "`x` heures",
|
||||||
|
"`x` minutes": "`x` minutes",
|
||||||
|
"`x` seconds": "`x` secondes",
|
||||||
|
"Fallback comments: ": "Commentaires alternatifs : ",
|
||||||
|
"Popular": "Populaire",
|
||||||
|
"Top": "Top",
|
||||||
|
"About": "À propos",
|
||||||
|
"Rating: ": "Évaluation : ",
|
||||||
|
"Language: ": "Langue : ",
|
||||||
|
"View as playlist": "Voir en tant que liste de lecture",
|
||||||
|
"Default": "Défaut",
|
||||||
|
"Music": "Musique",
|
||||||
|
"Gaming": "Jeux Vidéo",
|
||||||
|
"News": "Actualités",
|
||||||
|
"Movies": "Films",
|
||||||
|
"Download": "Télécharger",
|
||||||
|
"Download as: ": "Télécharger en : ",
|
||||||
|
"%A %B %-d, %Y": "%A %-d %B %Y",
|
||||||
|
"(edited)": "(modifié)",
|
||||||
|
"YouTube comment permalink": "Lien permanent vers le commentaire sur YouTube",
|
||||||
|
"permalink": "Lien permanent",
|
||||||
|
"`x` marked it with a ❤": "`x` l'a marqué d'un ❤",
|
||||||
|
"Audio mode": "Mode audio",
|
||||||
|
"Video mode": "Mode vidéo",
|
||||||
|
"Videos": "Vidéos",
|
||||||
|
"Playlists": "Listes de lecture",
|
||||||
|
"Community": "Communauté",
|
||||||
|
"Current version: ": "Version actuelle : "
|
||||||
}
|
}
|
||||||
|
|||||||
351
locales/is.json
Normal file
351
locales/is.json
Normal file
@@ -0,0 +1,351 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": "",
|
||||||
|
"`x` videos": "",
|
||||||
|
"`x` playlists": "",
|
||||||
|
"`x` subscribers.": "`x` áskrifandar.",
|
||||||
|
"`x` videos.": "`x` myndbönd.",
|
||||||
|
"LIVE": "BEINT",
|
||||||
|
"Shared `x` ago": "Deilt `x` síðan",
|
||||||
|
"Unsubscribe": "Afskrá",
|
||||||
|
"Subscribe": "Áskrifa",
|
||||||
|
"View channel on YouTube": "Skoða rás á YouTube",
|
||||||
|
"View playlist on YouTube": "Skoða spilunarlisti á YouTube",
|
||||||
|
"newest": "nýjasta",
|
||||||
|
"oldest": "elsta",
|
||||||
|
"popular": "vinsælt",
|
||||||
|
"last": "síðast",
|
||||||
|
"Next page": "Næsta síða",
|
||||||
|
"Previous page": "Fyrri síða",
|
||||||
|
"Clear watch history?": "Hreinsa áhorfssögu?",
|
||||||
|
"New password": "Nýtt lykilorð",
|
||||||
|
"New passwords must match": "Nýtt lykilorð verður að passa",
|
||||||
|
"Cannot change password for Google accounts": "Ekki er hægt að breyta lykilorði fyrir Google reikninga",
|
||||||
|
"Authorize token?": "Leyfa tákn?",
|
||||||
|
"Authorize token for `x`?": "Leyfa tákn fyrir `x`?",
|
||||||
|
"Yes": "Já",
|
||||||
|
"No": "Nei",
|
||||||
|
"Import and Export Data": "Innflutningur og Útflutningur Gagna",
|
||||||
|
"Import": "Flytja inn",
|
||||||
|
"Import Invidious data": "Flytja inn Invidious gögn",
|
||||||
|
"Import YouTube subscriptions": "Flytja inn YouTube áskriftir",
|
||||||
|
"Import FreeTube subscriptions (.db)": "Flytja inn FreeTube áskriftir (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "Flytja inn NewPipe áskriftir (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "Flytja inn NewPipe gögn (.zip)",
|
||||||
|
"Export": "Flytja út",
|
||||||
|
"Export subscriptions as OPML": "Flytja út áskriftir sem OPML",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Flytja út áskriftir sem OPML (fyrir NewPipe & FreeTube)",
|
||||||
|
"Export data as JSON": "Flytja út gögn sem JSON",
|
||||||
|
"Delete account?": "Eyða reikningi?",
|
||||||
|
"History": "Saga",
|
||||||
|
"An alternative front-end to YouTube": "Önnur framhlið fyrir YouTube",
|
||||||
|
"JavaScript license information": "JavaScript leyfi upplýsingar",
|
||||||
|
"source": "uppspretta",
|
||||||
|
"Log in": "Skrá inn",
|
||||||
|
"Log in/register": "Innskráning/nýskráning",
|
||||||
|
"Log in with Google": "Skrá inn með Google",
|
||||||
|
"User ID": "Notandakenni",
|
||||||
|
"Password": "Lykilorð",
|
||||||
|
"Time (h:mm:ss):": "Tími (h:mm: ss):",
|
||||||
|
"Text CAPTCHA": "Texta CAPTCHA",
|
||||||
|
"Image CAPTCHA": "Mynd CAPTCHA",
|
||||||
|
"Sign In": "Skrá inn",
|
||||||
|
"Register": "Nýskrá",
|
||||||
|
"E-mail": "Tölvupóstur",
|
||||||
|
"Google verification code": "Google staðfestingarkóði",
|
||||||
|
"Preferences": "Kjörstillingar",
|
||||||
|
"Player preferences": "Kjörstillingar spilara",
|
||||||
|
"Always loop: ": "Alltaf lykkja: ",
|
||||||
|
"Autoplay: ": "Spila sjálfkrafa: ",
|
||||||
|
"Play next by default: ": "Spila næst sjálfgefið: ",
|
||||||
|
"Autoplay next video: ": "Spila næst sjálfkrafa: ",
|
||||||
|
"Listen by default: ": "Hlusta sjálfgefið: ",
|
||||||
|
"Proxy videos: ": "Proxy myndbönd? ",
|
||||||
|
"Default speed: ": "Sjálfgefinn hraði: ",
|
||||||
|
"Preferred video quality: ": "Æskilegt myndbands gæði: ",
|
||||||
|
"Player volume: ": "Spilara hljóðstyrkur: ",
|
||||||
|
"Default comments: ": "Sjálfgefin ummæli: ",
|
||||||
|
"youtube": "youtube",
|
||||||
|
"reddit": "reddit",
|
||||||
|
"Default captions: ": "Sjálfgefin texti: ",
|
||||||
|
"Fallback captions: ": "Varatextar: ",
|
||||||
|
"Show related videos: ": "Sýna tengd myndbönd? ",
|
||||||
|
"Show annotations by default: ": "Á að sýna glósur sjálfgefið? ",
|
||||||
|
"Visual preferences": "Sjónrænar stillingar",
|
||||||
|
"Player style: ": "",
|
||||||
|
"Dark mode: ": "Myrkur ham: ",
|
||||||
|
"Theme: ": "",
|
||||||
|
"dark": "",
|
||||||
|
"light": "",
|
||||||
|
"Thin mode: ": "Þunnt ham: ",
|
||||||
|
"Subscription preferences": "Áskriftarstillingar",
|
||||||
|
"Show annotations by default for subscribed channels: ": "Á að sýna glósur sjálfgefið fyrir áskriftarrásir? ",
|
||||||
|
"Redirect homepage to feed: ": "Endurbeina heimasíðu að straumi: ",
|
||||||
|
"Number of videos shown in feed: ": "Fjöldi myndbanda sem sýndir eru í straumi: ",
|
||||||
|
"Sort videos by: ": "Raða myndbönd eftir: ",
|
||||||
|
"published": "birt",
|
||||||
|
"published - reverse": "birt - afturábak",
|
||||||
|
"alphabetically": "í stafrófsröð",
|
||||||
|
"alphabetically - reverse": "stafrófsröð - afturábak",
|
||||||
|
"channel name": "heiti rásar",
|
||||||
|
"channel name - reverse": "heiti rásar - afturábak",
|
||||||
|
"Only show latest video from channel: ": "Sýna aðeins nýjasta myndband frá rás: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "Sýna aðeins nýjasta óséð myndband frá rás: ",
|
||||||
|
"Only show unwatched: ": "Sýna aðeins óséð: ",
|
||||||
|
"Only show notifications (if there are any): ": "Sýna aðeins tilkynningar (ef einhverjar eru): ",
|
||||||
|
"Enable web notifications": "Virkja veftilkynningar",
|
||||||
|
"`x` uploaded a video": "`x` hlóð upp myndband",
|
||||||
|
"`x` is live": "`x` er í beinni",
|
||||||
|
"Data preferences": "Gagnastillingar",
|
||||||
|
"Clear watch history": "Hreinsa áhorfssögu",
|
||||||
|
"Import/export data": "Flytja inn/út gögn",
|
||||||
|
"Change password": "Breyta lykilorði",
|
||||||
|
"Manage subscriptions": "Stjórna áskriftum",
|
||||||
|
"Manage tokens": "Stjórna tákn",
|
||||||
|
"Watch history": "Áhorfssögu",
|
||||||
|
"Delete account": "Eyða reikningi",
|
||||||
|
"Administrator preferences": "Kjörstillingar stjórnanda",
|
||||||
|
"Default homepage: ": "Sjálfgefin heimasíða: ",
|
||||||
|
"Feed menu: ": "Straum valmynd: ",
|
||||||
|
"Top enabled: ": "Toppur virkur? ",
|
||||||
|
"CAPTCHA enabled: ": "CAPTCHA virk? ",
|
||||||
|
"Login enabled: ": "Innskráning virk? ",
|
||||||
|
"Registration enabled: ": "Nýskráning virkjuð? ",
|
||||||
|
"Report statistics: ": "Skrá talnagögn? ",
|
||||||
|
"Save preferences": "Vista stillingar",
|
||||||
|
"Subscription manager": "Áskriftarstjóri",
|
||||||
|
"`x` subscriptions": "",
|
||||||
|
"`x` tokens": "",
|
||||||
|
"Token manager": "Táknstjóri",
|
||||||
|
"Token": "Tákn",
|
||||||
|
"`x` subscriptions.": "`x` áskriftir.",
|
||||||
|
"`x` tokens.": "`x` tákn.",
|
||||||
|
"`x` unseen notifications": "",
|
||||||
|
"Import/export": "Flytja inn/út",
|
||||||
|
"unsubscribe": "afskrá",
|
||||||
|
"revoke": "afturkalla",
|
||||||
|
"Subscriptions": "Áskriftir",
|
||||||
|
"`x` unseen notifications.": "`x` óséðar tilkynningar.",
|
||||||
|
"search": "leita",
|
||||||
|
"Log out": "Útskrá",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Útgefið undir AGPLv3 eftir Omar Roth.",
|
||||||
|
"Source available here.": "Frumkóði aðgengilegur hér.",
|
||||||
|
"View JavaScript license information.": "Skoða JavaScript leyfisupplýsingar.",
|
||||||
|
"View privacy policy.": "Skoða meðferð persónuupplýsinga.",
|
||||||
|
"Trending": "Vinsælt",
|
||||||
|
"Public": "",
|
||||||
|
"Unlisted": "Óskráð",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "Horfa á YouTube",
|
||||||
|
"Hide annotations": "Fela glósur",
|
||||||
|
"Show annotations": "Sýna glósur",
|
||||||
|
"Genre: ": "Tegund: ",
|
||||||
|
"License: ": "Notkunarleyfi: ",
|
||||||
|
"Family friendly? ": "Fjölskylduvænt? ",
|
||||||
|
"`x` views": "",
|
||||||
|
"Wilson score: ": "Wilson stig: ",
|
||||||
|
"Engagement: ": "Þátttöku: ",
|
||||||
|
"Whitelisted regions: ": "Svæði á hvítum lista: ",
|
||||||
|
"Blacklisted regions: ": "Svæði á svörtum lista: ",
|
||||||
|
"Shared `x`": "Deilt `x`",
|
||||||
|
"`x` views.": "`x` áhorf.",
|
||||||
|
"Premieres in `x`": "Frumflutt eftir `x`",
|
||||||
|
"Premieres `x`": "Frumflutt `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hæ! Lítur út eins og þú hafir slökkt á JavaScript. Smelltu hér til að skoða ummæli, hafðu í huga að þær geta tekið aðeins lengri tíma að hlaða.",
|
||||||
|
"View YouTube comments": "Skoða YouTube ummæli",
|
||||||
|
"View more comments on Reddit": "Skoða fleiri ummæli á Reddit",
|
||||||
|
"View `x` comments": "Skoða `x` ummæli",
|
||||||
|
"View Reddit comments": "Skoða Reddit ummæli",
|
||||||
|
"Hide replies": "Fela svör",
|
||||||
|
"Show replies": "Sýna svör",
|
||||||
|
"Incorrect password": "Rangt lykilorð",
|
||||||
|
"Quota exceeded, try again in a few hours": "Kvóti fór yfir, reyndu aftur eftir nokkrar klukkustundir",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Ekki er hægt að skrá þig inn, vertu viss um að tvíþætt staðfesting (Authenticator eða SMS) sé kveikt á.",
|
||||||
|
"Invalid TFA code": "Ógildur TFA kóði",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Innskráning mistókst. Þetta gæti verið vegna þess að tvíþátta staðfesting er ekki kveikt á reikningnum þínum.",
|
||||||
|
"Wrong answer": "Rangt svar",
|
||||||
|
"Erroneous CAPTCHA": "Rangt CAPTCHA",
|
||||||
|
"CAPTCHA is a required field": "CAPTCHA er nauðsynlegur reitur",
|
||||||
|
"User ID is a required field": "Notandakenni er nauðsynlegur reitur",
|
||||||
|
"Password is a required field": "Lykilorð er nauðsynlegur reitur",
|
||||||
|
"Wrong username or password": "Rangt notandanafn eða lykilorð",
|
||||||
|
"Please sign in using 'Log in with Google'": "Vinsamlegast skráðu þig inn með því að nota 'Innskráning með Google'",
|
||||||
|
"Password cannot be empty": "Lykilorð má ekki vera autt",
|
||||||
|
"Password cannot be longer than 55 characters": "Lykilorð má ekki vera lengra en 55 stafir",
|
||||||
|
"Please log in": "Vinsamlegast skráðu þig inn",
|
||||||
|
"View `x` replies": "",
|
||||||
|
"Invidious Private Feed for `x`": "Invidious Persónulegur Straumur fyrir `x`",
|
||||||
|
"channel:`x`": "rás:`x`",
|
||||||
|
"`x` points": "",
|
||||||
|
"Deleted or invalid channel": "Eytt eða ógild rás",
|
||||||
|
"This channel does not exist.": "Þessi rás er ekki til.",
|
||||||
|
"Could not get channel info.": "Ekki tókst að fá rásarupplýsingar.",
|
||||||
|
"Could not fetch comments": "Ekki tókst að sækja ummæli",
|
||||||
|
"View `x` replies.": "Skoða `x` svör.",
|
||||||
|
"`x` ago": "`x` síðan",
|
||||||
|
"Load more": "Hlaða meira",
|
||||||
|
"`x` points.": "`x` stig.",
|
||||||
|
"Could not create mix.": "Ekki tókst að búa til blöndu.",
|
||||||
|
"Empty playlist": "Tómur spilunarlisti",
|
||||||
|
"Not a playlist.": "Ekki spilunarlisti.",
|
||||||
|
"Playlist does not exist.": "Spilunarlisti er ekki til.",
|
||||||
|
"Could not pull trending pages.": "Ekki tókst að draga vinsælar síður.",
|
||||||
|
"Hidden field \"challenge\" is a required field": "Falinn reitur \"áskorun\" er nauðsynlegur reitur",
|
||||||
|
"Hidden field \"token\" is a required field": "Falinn reitur \"tákn\" er nauðsynlegur reitur",
|
||||||
|
"Erroneous challenge": "Röng áskorun",
|
||||||
|
"Erroneous token": "Rangt tákn",
|
||||||
|
"No such user": "Enginn slíkur notandi",
|
||||||
|
"Token is expired, please try again": "Tákn er útrunnið, vinsamlegast reyndu aftur",
|
||||||
|
"English": "Enska",
|
||||||
|
"English (auto-generated)": "Enska (sjálfkrafa)",
|
||||||
|
"Afrikaans": "Afríkanska",
|
||||||
|
"Albanian": "Albanska",
|
||||||
|
"Amharic": "Amharíska",
|
||||||
|
"Arabic": "Arabíska",
|
||||||
|
"Armenian": "Armenska",
|
||||||
|
"Azerbaijani": "Aserbaídsjanska",
|
||||||
|
"Bangla": "Bangla",
|
||||||
|
"Basque": "Baskneska",
|
||||||
|
"Belarusian": "Hvítrússneska",
|
||||||
|
"Bosnian": "Bosníska",
|
||||||
|
"Bulgarian": "Búlgarska",
|
||||||
|
"Burmese": "Búrmíska",
|
||||||
|
"Catalan": "Katalónska",
|
||||||
|
"Cebuano": "Cebúanó",
|
||||||
|
"Chinese (Simplified)": "Kínverska (Einfölduð)",
|
||||||
|
"Chinese (Traditional)": "Kínverska (Hefðbundin)",
|
||||||
|
"Corsican": "Korsíska",
|
||||||
|
"Croatian": "Króatíska",
|
||||||
|
"Czech": "Tékkneska",
|
||||||
|
"Danish": "Danska",
|
||||||
|
"Dutch": "Hollenska",
|
||||||
|
"Esperanto": "Esperantó",
|
||||||
|
"Estonian": "Eistneska",
|
||||||
|
"Filipino": "Filippínska",
|
||||||
|
"Finnish": "Finnska",
|
||||||
|
"French": "Franska",
|
||||||
|
"Galician": "Galisíska",
|
||||||
|
"Georgian": "Georgíska",
|
||||||
|
"German": "Þýska",
|
||||||
|
"Greek": "Gríska",
|
||||||
|
"Gujarati": "Gújaratí",
|
||||||
|
"Haitian Creole": "Haítískt Kreólamál",
|
||||||
|
"Hausa": "Hausa",
|
||||||
|
"Hawaiian": "Havaíska",
|
||||||
|
"Hebrew": "Hebreska",
|
||||||
|
"Hindi": "Hindí",
|
||||||
|
"Hmong": "Hmong",
|
||||||
|
"Hungarian": "Ungverska",
|
||||||
|
"Icelandic": "Íslenska",
|
||||||
|
"Igbo": "Igbo",
|
||||||
|
"Indonesian": "Indónesíska",
|
||||||
|
"Irish": "Írska",
|
||||||
|
"Italian": "Ítalska",
|
||||||
|
"Japanese": "Japanska",
|
||||||
|
"Javanese": "Javanska",
|
||||||
|
"Kannada": "Kanaríska",
|
||||||
|
"Kazakh": "Kasakíska",
|
||||||
|
"Khmer": "Khmeríska",
|
||||||
|
"Korean": "Kóreska",
|
||||||
|
"Kurdish": "Kúrdíska",
|
||||||
|
"Kyrgyz": "Kirgisíska",
|
||||||
|
"Lao": "Laó",
|
||||||
|
"Latin": "Latína",
|
||||||
|
"Latvian": "Lettneska",
|
||||||
|
"Lithuanian": "Litháíska",
|
||||||
|
"Luxembourgish": "Lúxemborgíska",
|
||||||
|
"Macedonian": "Makedóníska",
|
||||||
|
"Malagasy": "Malagasíska",
|
||||||
|
"Malay": "Malaíska",
|
||||||
|
"Malayalam": "Malaíalam",
|
||||||
|
"Maltese": "Maltneska",
|
||||||
|
"Maori": "Maórí",
|
||||||
|
"Marathi": "Marathi",
|
||||||
|
"Mongolian": "Mongólska",
|
||||||
|
"Nepali": "Nepalska",
|
||||||
|
"Norwegian Bokmål": "Norskt bókmál",
|
||||||
|
"Nyanja": "Nyanja",
|
||||||
|
"Pashto": "Pashto",
|
||||||
|
"Persian": "Persneska",
|
||||||
|
"Polish": "Pólska",
|
||||||
|
"Portuguese": "Portúgalska",
|
||||||
|
"Punjabi": "Punjabi",
|
||||||
|
"Romanian": "Rúmenska",
|
||||||
|
"Russian": "Rússneska",
|
||||||
|
"Samoan": "Samóíska",
|
||||||
|
"Scottish Gaelic": "Skosk Gelíska",
|
||||||
|
"Serbian": "Serbneska",
|
||||||
|
"Shona": "Shona",
|
||||||
|
"Sindhi": "Sindí",
|
||||||
|
"Sinhala": "Sinhala",
|
||||||
|
"Slovak": "Slóvakíska",
|
||||||
|
"Slovenian": "Slóvenska",
|
||||||
|
"Somali": "Sómalska",
|
||||||
|
"Southern Sotho": "Suður Sótó",
|
||||||
|
"Spanish": "Spænska",
|
||||||
|
"Spanish (Latin America)": "Spænska (Rómönsku Ameríka)",
|
||||||
|
"Sundanese": "Sundaneska",
|
||||||
|
"Swahili": "Svahílí",
|
||||||
|
"Swedish": "Sænska",
|
||||||
|
"Tajik": "Tadsikíska",
|
||||||
|
"Tamil": "Tamílska",
|
||||||
|
"Telugu": "Telúgú",
|
||||||
|
"Thai": "Taílenska",
|
||||||
|
"Turkish": "Tyrkneska",
|
||||||
|
"Ukrainian": "Úkraníska",
|
||||||
|
"Urdu": "Úrdú",
|
||||||
|
"`x` years": "",
|
||||||
|
"`x` months": "",
|
||||||
|
"`x` weeks": "",
|
||||||
|
"`x` days": "",
|
||||||
|
"`x` hours": "",
|
||||||
|
"`x` minutes": "",
|
||||||
|
"`x` seconds": "",
|
||||||
|
"Uzbek": "Úsbekíska",
|
||||||
|
"Vietnamese": "Víetnamska",
|
||||||
|
"Welsh": "Velska",
|
||||||
|
"Western Frisian": "Vestur Frísneska",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Jiddíska",
|
||||||
|
"Yoruba": "Jórúba",
|
||||||
|
"Zulu": "Zúlú",
|
||||||
|
"`x` years.": "`x` ár.",
|
||||||
|
"`x` months.": "`x` mánuði.",
|
||||||
|
"`x` weeks.": "`x` vikur.",
|
||||||
|
"`x` days.": "`x` dagar.",
|
||||||
|
"`x` hours.": "`x` klukkustundir.",
|
||||||
|
"`x` minutes.": "`x` mínútur.",
|
||||||
|
"`x` seconds.": "`x` sekúndur.",
|
||||||
|
"Fallback comments: ": "Vara ummæli: ",
|
||||||
|
"Popular": "Vinsælt",
|
||||||
|
"permalink": "",
|
||||||
|
"Top": "Topp",
|
||||||
|
"About": "Um",
|
||||||
|
"Rating: ": "Einkunn: ",
|
||||||
|
"Language: ": "Tungumál: ",
|
||||||
|
"View as playlist": "Skoða sem spilunarlista",
|
||||||
|
"Community": "",
|
||||||
|
"Default": "Sjálfgefið",
|
||||||
|
"Music": "Tónlist",
|
||||||
|
"Gaming": "Tólvuleikja",
|
||||||
|
"News": "Fréttir",
|
||||||
|
"Movies": "Kvikmyndir",
|
||||||
|
"Download": "Niðurhal",
|
||||||
|
"Download as: ": "Niðurhala sem: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(breytt)",
|
||||||
|
"YouTube comment permalink": "YouTube ummæli varanlegur tengill",
|
||||||
|
"`x` marked it with a ❤": "`x` merkti það með ❤",
|
||||||
|
"Audio mode": "Hljóð ham",
|
||||||
|
"Video mode": "Myndband ham",
|
||||||
|
"Videos": "Myndbönd",
|
||||||
|
"Playlists": "Spilunarlistar",
|
||||||
|
"Current version: ": "Núverandi útgáfa: "
|
||||||
|
}
|
||||||
674
locales/it.json
674
locales/it.json
@@ -1,295 +1,381 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` iscritti",
|
"`x` subscribers": {
|
||||||
"`x` videos": "`x` video",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` iscritto",
|
||||||
"LIVE": "IN DIRETTA",
|
"": "`x` iscritti"
|
||||||
"Shared `x` ago": "Condiviso `x` fa",
|
},
|
||||||
"Unsubscribe": "Disiscriviti",
|
"`x` videos": {
|
||||||
"Subscribe": "Iscriviti",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` video",
|
||||||
"Login to subscribe to `x`": "Accedi per iscriverti a `x`",
|
"": "`x` video"
|
||||||
"View channel on YouTube": "Vedi canale su YouTube",
|
},
|
||||||
"newest": "Data di aggiunta (più recente)",
|
"`x` playlists": "",
|
||||||
"oldest": "Data di aggiunta (più vecchia)",
|
"LIVE": "IN DIRETTA",
|
||||||
"popular": "Tendenze",
|
"Shared `x` ago": "Condiviso `x` fa",
|
||||||
"last": "",
|
"Unsubscribe": "Disiscriviti",
|
||||||
"Next page": "Pagina successiva",
|
"Subscribe": "Iscriviti",
|
||||||
"Previous page": "Pagina precedente",
|
"View channel on YouTube": "Vedi canale su YouTube",
|
||||||
"Clear watch history?": "Sei sicuro di voler cancellare la cronologia dei video guardati?",
|
"View playlist on YouTube": "Vedi playlist su YouTube",
|
||||||
"Yes": "Si",
|
"newest": "più recente",
|
||||||
"No": "No",
|
"oldest": "più vecchio",
|
||||||
"Import and Export Data": "Importazione ed esportazione dati",
|
"popular": "Tendenze",
|
||||||
"Import": "Importa",
|
"last": "durare",
|
||||||
"Import Invidious data": "Importa dati Invidious",
|
"Next page": "Pagina successiva",
|
||||||
"Import YouTube subscriptions": "Importa le iscrizioni da YouTube",
|
"Previous page": "Pagina precedente",
|
||||||
"Import FreeTube subscriptions (.db)": "Importa le iscrizioni da FreeTube (.db)",
|
"Clear watch history?": "Eliminare la cronologia dei video guardati?",
|
||||||
"Import NewPipe subscriptions (.json)": "Importa le iscrizioni da NewPipe (.json)",
|
"New password": "Nuova password",
|
||||||
"Import NewPipe data (.zip)": "Importa i dati di NewPipe (.zip)",
|
"New passwords must match": "Le nuove password devono corrispondere",
|
||||||
"Export": "Esporta",
|
"Cannot change password for Google accounts": "Non è possibile modificare la password per gli account Google",
|
||||||
"Export subscriptions as OPML": "Esporta gli abbonamenti come OPML",
|
"Authorize token?": "Autorizzare gettone?",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Esporta gli abbonamenti come OPML (per NewPipe e FreeTube)",
|
"Authorize token for `x`?": "Autorizzare gettone per `x`?",
|
||||||
"Export data as JSON": "Esporta i dati in formato JSON",
|
"Yes": "Sì",
|
||||||
"Delete account?": "Sei sicuro di voler cancellare l'account?",
|
"No": "No",
|
||||||
"History": "Cronologia",
|
"Import and Export Data": "Importazione ed esportazione dati",
|
||||||
"An alternative front-end to YouTube": "Un'interfaccia alternativa per YouTube",
|
"Import": "Importa",
|
||||||
"JavaScript license information": "Info licenze JavaScript",
|
"Import Invidious data": "Importa dati Invidious",
|
||||||
"source": "sorgente",
|
"Import YouTube subscriptions": "Importa le iscrizioni da YouTube",
|
||||||
"Login": "Entra",
|
"Import FreeTube subscriptions (.db)": "Importa le iscrizioni da FreeTube (.db)",
|
||||||
"Login/Register": "Entra/Registrati",
|
"Import NewPipe subscriptions (.json)": "Importa le iscrizioni da NewPipe (.json)",
|
||||||
"Login to Google": "Entra con Google",
|
"Import NewPipe data (.zip)": "Importa i dati di NewPipe (.zip)",
|
||||||
"User ID:": "ID utente:",
|
"Export": "Esporta",
|
||||||
"Password:": "Password:",
|
"Export subscriptions as OPML": "Esporta gli abbonamenti come OPML",
|
||||||
"Time (h:mm:ss):": "Orario (h:mm:ss):",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Esporta gli abbonamenti come OPML (per NewPipe e FreeTube)",
|
||||||
"Text CAPTCHA": "Testo del CAPTCHA",
|
"Export data as JSON": "Esporta i dati in formato JSON",
|
||||||
"Image CAPTCHA": "Immagine CAPTCHA",
|
"Delete account?": "Eliminare l'account?",
|
||||||
"Sign In": "Entra",
|
"History": "Cronologia",
|
||||||
"Register": "Registrati",
|
"An alternative front-end to YouTube": "Un'interfaccia alternativa per YouTube",
|
||||||
"Email:": "Email:",
|
"JavaScript license information": "Info licenze JavaScript",
|
||||||
"Google verification code:": "Codice di verifica Google:",
|
"source": "sorgente",
|
||||||
"Preferences": "Preferenze",
|
"Log in": "Accedi",
|
||||||
"Player preferences": "Preferenze del riproduttore",
|
"Log in/register": "Accedi/Registrati",
|
||||||
"Always loop: ": "Ripeti sempre: ",
|
"Log in with Google": "Accedi con Google",
|
||||||
"Autoplay: ": "Riproduzione automatica: ",
|
"User ID": "ID utente",
|
||||||
"Autoplay next video: ": "Riproduci automaticamente il prossimo video: ",
|
"Password": "Password",
|
||||||
"Listen by default: ": "Modalità solo audio come predefinita: ",
|
"Time (h:mm:ss):": "Orario (h:mm:ss):",
|
||||||
"Proxy videos? ": "",
|
"Text CAPTCHA": "Testo del CAPTCHA",
|
||||||
"Default speed: ": "Velocità di riproduzione predefinita: ",
|
"Image CAPTCHA": "Immagine CAPTCHA",
|
||||||
"Preferred video quality: ": "Preferenza sulla qualità video: ",
|
"Sign In": "Accedi",
|
||||||
"Player volume: ": "Volume di riproduzione: ",
|
"Register": "Registrati",
|
||||||
"Default comments: ": "Origine dei commenti: ",
|
"E-mail": "Email",
|
||||||
"Default captions: ": "Sottotitoli predefiniti: ",
|
"Google verification code": "Codice di verifica Google",
|
||||||
"Fallback captions: ": "Sottotitoli alternativi: ",
|
"Preferences": "Preferenze",
|
||||||
"Show related videos? ": "Mostra video correlati? ",
|
"Player preferences": "Preferenze del riproduttore",
|
||||||
"Visual preferences": "Preferenze grafiche",
|
"Always loop: ": "Ripeti sempre: ",
|
||||||
"Dark mode: ": "Tema scuro: ",
|
"Autoplay: ": "Riproduzione automatica: ",
|
||||||
"Thin mode: ": "Modalità per connessioni lente: ",
|
"Play next by default: ": "Riproduzione successiva predefinita: ",
|
||||||
"Subscription preferences": "Preferenze iscrizioni",
|
"Autoplay next video: ": "Riproduci automaticamente il video successivo: ",
|
||||||
"Redirect homepage to feed: ": "Reindirizza la pagina principale a quella delle iscrizioni: ",
|
"Listen by default: ": "Modalità solo audio predefinita: ",
|
||||||
"Number of videos shown in feed: ": "Numero di video da mostrare nelle iscrizioni: ",
|
"Proxy videos: ": "Proxy per i video: ",
|
||||||
"Sort videos by: ": "Ordinare i video per: ",
|
"Default speed: ": "Velocità predefinita: ",
|
||||||
"published": "data di pubblicazione",
|
"Preferred video quality: ": "Qualità video preferita: ",
|
||||||
"published - reverse": "data di pubblicazione - decrescente",
|
"Player volume: ": "Volume di riproduzione: ",
|
||||||
"alphabetically": "ordine alfabetico",
|
"Default comments: ": "Origine dei commenti: ",
|
||||||
"alphabetically - reverse": "ordine alfabetico - decrescente",
|
"youtube": "YouTube",
|
||||||
"channel name": "nome del canale",
|
"reddit": "Reddit",
|
||||||
"channel name - reverse": "nome del canale - decrescente",
|
"Default captions: ": "Sottotitoli predefiniti: ",
|
||||||
"Only show latest video from channel: ": "Mostra solo il video più recente del canale: ",
|
"Fallback captions: ": "Sottotitoli alternativi: ",
|
||||||
"Only show latest unwatched video from channel: ": "Mostra solo il video più recente non guardato del canale: ",
|
"Show related videos: ": "Mostra video correlati: ",
|
||||||
"Only show unwatched: ": "Mostra solo i video non guardati: ",
|
"Show annotations by default: ": "Mostra le annotazioni in modo predefinito: ",
|
||||||
"Only show notifications (if there are any): ": "Mostra solo le notifiche (se presenti): ",
|
"Visual preferences": "Preferenze grafiche",
|
||||||
"Data preferences": "Preferenze dati",
|
"Player style: ": "Stile riproduttore",
|
||||||
"Clear watch history": "Cancella la cronologia dei video guardati",
|
"Dark mode: ": "Tema scuro: ",
|
||||||
"Import/Export data": "Importazione/esportazione dati",
|
"Theme: ": "Tema",
|
||||||
"Manage subscriptions": "Gestisci le iscrizioni",
|
"dark": "scuro",
|
||||||
"Watch history": "Cronologia dei video",
|
"light": "chiaro",
|
||||||
"Delete account": "Elimina l'account",
|
"Thin mode: ": "Modalità per connessioni lente: ",
|
||||||
"Administrator preferences": "",
|
"Subscription preferences": "Preferenze iscrizioni",
|
||||||
"Default homepage: ": "",
|
"Show annotations by default for subscribed channels: ": "Mostrare annotazioni in modo predefinito per i canali sottoscritti: ",
|
||||||
"Feed menu: ": "",
|
"Redirect homepage to feed: ": "Reindirizza la pagina principale a quella delle iscrizioni: ",
|
||||||
"Top enabled? ": "",
|
"Number of videos shown in feed: ": "Numero di video da mostrare nelle iscrizioni: ",
|
||||||
"CAPTCHA enabled? ": "",
|
"Sort videos by: ": "Ordina i video per: ",
|
||||||
"Login enabled? ": "",
|
"published": "data di pubblicazione",
|
||||||
"Registration enabled? ": "",
|
"published - reverse": "data di pubblicazione - decrescente",
|
||||||
"Report statistics? ": "",
|
"alphabetically": "ordine alfabetico",
|
||||||
"Save preferences": "Salva le preferenze",
|
"alphabetically - reverse": "ordine alfabetico - decrescente",
|
||||||
"Subscription manager": "Gestisci le iscrizioni",
|
"channel name": "nome del canale",
|
||||||
"`x` subscriptions": "`x` iscrizioni",
|
"channel name - reverse": "nome del canale - decrescente",
|
||||||
"Import/Export": "Importa/esporta",
|
"Only show latest video from channel: ": "Mostra solo il video più recente del canale: ",
|
||||||
"unsubscribe": "disiscriviti",
|
"Only show latest unwatched video from channel: ": "Mostra solo il video più recente non guardato del canale: ",
|
||||||
"Subscriptions": "Iscrizioni",
|
"Only show unwatched: ": "Mostra solo i video non guardati: ",
|
||||||
"`x` unseen notifications": "`x` notifiche non visualizzate",
|
"Only show notifications (if there are any): ": "Mostra solo le notifiche (se presenti): ",
|
||||||
"search": "Cerca",
|
"Enable web notifications": "Attiva le notifiche web",
|
||||||
"Sign out": "Esci",
|
"`x` uploaded a video": "`x` ha caricato un video",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Pubblicato con licenza AGPLv3 da Omar Roth.",
|
"`x` is live": "`x` è in diretta",
|
||||||
"Source available here.": "Codice sorgente.",
|
"Data preferences": "Preferenze dati",
|
||||||
"View JavaScript license information.": "Guarda le informazioni di licenza del codice JavaScript.",
|
"Clear watch history": "Cancella la cronologia dei video guardati",
|
||||||
"View privacy policy.": "",
|
"Import/export data": "Importazione/esportazione dati",
|
||||||
"Trending": "Tendenze",
|
"Change password": "Modifica password",
|
||||||
"Unlisted": "",
|
"Manage subscriptions": "Gestisci le iscrizioni",
|
||||||
"Watch video on Youtube": "Guarda il video su YouTube",
|
"Manage tokens": "Gestisci i gettoni",
|
||||||
"Genre: ": "Genere: ",
|
"Watch history": "Cronologia dei video",
|
||||||
"License: ": "Licenza: ",
|
"Delete account": "Elimina l'account",
|
||||||
"Family friendly? ": "Per tutti? ",
|
"Administrator preferences": "Preferenze amministratore",
|
||||||
"Wilson score: ": "Punteggio di Wilson: ",
|
"Default homepage: ": "Pagina principale predefinita: ",
|
||||||
"Engagement: ": "Tasso di coinvolgimento: ",
|
"Feed menu: ": "Menu iscrizioni: ",
|
||||||
"Whitelisted regions: ": "Regioni nella lista bianca: ",
|
"Top enabled: ": "",
|
||||||
"Blacklisted regions: ": "Regioni nella lista nera: ",
|
"CAPTCHA enabled: ": "CAPTCHA attivati: ",
|
||||||
"Shared `x`": "Condiviso `x`",
|
"Login enabled: ": "Accesso attivato: ",
|
||||||
"Premieres in `x`": "",
|
"Registration enabled: ": "Registrazione attivata: ",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Ciao! Sembra che tu abbia disattivato JavaScript. Clicca qui per visualizzare i commenti. Considera che potrebbe volerci più tempo.",
|
"Report statistics: ": "Resoconto delle statistiche: ",
|
||||||
"View YouTube comments": "Visualizza i commenti da YouTube",
|
"Save preferences": "Salva le preferenze",
|
||||||
"View more comments on Reddit": "Visualizza più commenti su Reddit",
|
"Subscription manager": "Gestione delle iscrizioni",
|
||||||
"View `x` comments": "Visualizza `x` commenti",
|
"Token manager": "Gestione dei gettoni",
|
||||||
"View Reddit comments": "Visualizza i commenti da Reddit",
|
"Token": "Gettone",
|
||||||
"Hide replies": "Nascondi le risposte",
|
"`x` subscriptions": {
|
||||||
"Show replies": "Mostra le risposte",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` iscrizione",
|
||||||
"Incorrect password": "Password sbagliata",
|
"": "`x` iscrizioni"
|
||||||
"Quota exceeded, try again in a few hours": "Limite superato, prova di nuovo fra qualche ora",
|
},
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Impossibile autenticarsi, controlla che l'autenticazione in due passaggi (Authenticator o SMS) sia attiva.",
|
"`x` tokens": {
|
||||||
"Invalid TFA code": "Codice di autenticazione a due fattori non valido",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` gettone",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Login fallito. L'errore potrebbe essere causato dal fatto che la verifica in due passaggi non è attiva sul tuo account.",
|
"": "`x` gettoni"
|
||||||
"Invalid answer": "Risposta errata",
|
},
|
||||||
"Invalid CAPTCHA": "CAPTCHA errato",
|
"Import/export": "Importa/esporta",
|
||||||
"CAPTCHA is a required field": "Il CAPTCHA è un campo obbligatorio",
|
"unsubscribe": "disiscriviti",
|
||||||
"User ID is a required field": "L'ID utente è obbligatorio",
|
"revoke": "revoca",
|
||||||
"Password is a required field": "La password è un campo obbligatorio",
|
"Subscriptions": "Iscrizioni",
|
||||||
"Invalid username or password": "Nome utente o password errati",
|
"`x` unseen notifications": {
|
||||||
"Please sign in using 'Sign in with Google'": "Per favore accedi con \"Entra con Google\"",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` notifica non visualizzata",
|
||||||
"Password cannot be empty": "La password non può essere vuota",
|
"": "`x` notifiche non visualizzate"
|
||||||
"Password cannot be longer than 55 characters": "La password non può contenere più di 55 caratteri",
|
},
|
||||||
"Please sign in": "Per favore, entra",
|
"search": "Cerca",
|
||||||
"Invidious Private Feed for `x`": "Feed privato Invidious per `x`",
|
"Log out": "Esci",
|
||||||
"channel:`x`": "canale:`x`",
|
"Released under the AGPLv3 by Omar Roth.": "Pubblicato con licenza AGPLv3 da Omar Roth.",
|
||||||
"Deleted or invalid channel": "Canale cancellato o invalido",
|
"Source available here.": "Codice sorgente.",
|
||||||
"This channel does not exist.": "Canale inesistente.",
|
"View JavaScript license information.": "Guarda le informazioni di licenza del codice JavaScript.",
|
||||||
"Could not get channel info.": "Impossibile ottenere le informazioni del canale.",
|
"View privacy policy.": "Vedi la politica sulla privacy",
|
||||||
"Could not fetch comments": "Impossibile recuperare i commenti",
|
"Trending": "Tendenze",
|
||||||
"View `x` replies": "Visualizza `x` risposte",
|
"Public": "",
|
||||||
"`x` ago": "`x` fa",
|
"Unlisted": "Non elencati",
|
||||||
"Load more": "Carica altro",
|
"Private": "",
|
||||||
"`x` points": "`x` punti",
|
"View all playlists": "",
|
||||||
"Could not create mix.": "Impossibile creare il mix.",
|
"Updated `x` ago": "",
|
||||||
"Playlist is empty": "Playlist vuota",
|
"Delete playlist `x`?": "",
|
||||||
"Invalid playlist.": "Playlist invalida.",
|
"Delete playlist": "",
|
||||||
"Playlist does not exist.": "Playlist inesistente.",
|
"Create playlist": "",
|
||||||
"Could not pull trending pages.": "Impossibile recuperare le tendenze.",
|
"Title": "",
|
||||||
"Hidden field \"challenge\" is a required field": "Il campo nascosto \"challenge\" è obbligatorio",
|
"Playlist privacy": "",
|
||||||
"Hidden field \"token\" is a required field": "Il campo nascosto \"token\" è obbligatorio",
|
"Editing playlist `x`": "",
|
||||||
"Invalid challenge": "Campo \"challenge\" invalido",
|
"Watch on YouTube": "Guarda su YouTube",
|
||||||
"Invalid token": "Campo \"token\" invalido",
|
"Hide annotations": "Nascondi annotazioni",
|
||||||
"Invalid user": "Utente invalido",
|
"Show annotations": "Mostra annotazioni",
|
||||||
"Token is expired, please try again": "Token scaduto, riprova",
|
"Genre: ": "Genere: ",
|
||||||
"English": "Inglese",
|
"License: ": "Licenza: ",
|
||||||
"English (auto-generated)": "Inglese (generati automaticamente)",
|
"Family friendly? ": "Per tutti? ",
|
||||||
"Afrikaans": "Afrikaans",
|
"Wilson score: ": "Punteggio di Wilson: ",
|
||||||
"Albanian": "Albanese",
|
"Engagement: ": "Tasso di coinvolgimento: ",
|
||||||
"Amharic": "Amarico",
|
"Whitelisted regions: ": "Regioni in lista bianca: ",
|
||||||
"Arabic": "Arabo",
|
"Blacklisted regions: ": "Regioni in lista nera: ",
|
||||||
"Armenian": "Armeno",
|
"Shared `x`": "Condiviso `x`",
|
||||||
"Azerbaijani": "Azero",
|
"`x` views": {
|
||||||
"Bangla": "Bengalese",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` visualizzazione",
|
||||||
"Basque": "Basco",
|
"": "`x` visualizzazioni"
|
||||||
"Belarusian": "Biellorusso",
|
},
|
||||||
"Bosnian": "Bosniaco",
|
"Premieres in `x`": "",
|
||||||
"Bulgarian": "Bulgaro",
|
"Premieres `x`": "",
|
||||||
"Burmese": "Birmano",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Ciao! Sembra che tu abbia disattivato JavaScript. Clicca qui per visualizzare i commenti. Considera che potrebbe volerci più tempo.",
|
||||||
"Catalan": "Catalano",
|
"View YouTube comments": "Visualizza i commenti da YouTube",
|
||||||
"Cebuano": "Sugbuanon",
|
"View more comments on Reddit": "Visualizza più commenti su Reddit",
|
||||||
"Chinese (Simplified)": "Cinese semplifiato",
|
"View `x` comments": "Visualizza `x` commenti",
|
||||||
"Chinese (Traditional)": "Cinese tradizionale",
|
"View Reddit comments": "Visualizza i commenti da Reddit",
|
||||||
"Corsican": "Corso",
|
"Hide replies": "Nascondi le risposte",
|
||||||
"Croatian": "Croato",
|
"Show replies": "Mostra le risposte",
|
||||||
"Czech": "Ceco",
|
"Incorrect password": "Password sbagliata",
|
||||||
"Danish": "Danese",
|
"Quota exceeded, try again in a few hours": "Limite superato, prova di nuovo fra qualche ora",
|
||||||
"Dutch": "Olandese",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Impossibile autenticarsi, controlla che l'autenticazione in due passaggi (Authenticator o SMS) sia attiva.",
|
||||||
"Esperanto": "Esperanto",
|
"Invalid TFA code": "Codice di autenticazione a due fattori non valido",
|
||||||
"Estonian": "Estone",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Login fallito. L'errore potrebbe essere causato dal fatto che la verifica in due passaggi non è attiva sul tuo account.",
|
||||||
"Filipino": "Filippino",
|
"Wrong answer": "Risposta errata",
|
||||||
"Finnish": "Finlandese",
|
"Erroneous CAPTCHA": "CAPTCHA errato",
|
||||||
"French": "Francese",
|
"CAPTCHA is a required field": "Il CAPTCHA è un campo obbligatorio",
|
||||||
"Galician": "Galiziano",
|
"User ID is a required field": "L'ID utente è obbligatorio",
|
||||||
"Georgian": "Georgiano",
|
"Password is a required field": "La password è un campo obbligatorio",
|
||||||
"German": "Tedesco",
|
"Wrong username or password": "Nome utente o password errati",
|
||||||
"Greek": "Greco",
|
"Please sign in using 'Log in with Google'": "Per favore accedi con \"Entra con Google\"",
|
||||||
"Gujarati": "Gujarati",
|
"Password cannot be empty": "La password non può essere vuota",
|
||||||
"Haitian Creole": "Creolo haitiano",
|
"Password cannot be longer than 55 characters": "La password non può contenere più di 55 caratteri",
|
||||||
"Hausa": "Lingua hausa",
|
"Please log in": "Per favore, accedi",
|
||||||
"Hawaiian": "Hawaiano",
|
"Invidious Private Feed for `x`": "Feed privato Invidious per `x`",
|
||||||
"Hebrew": "Ebreo",
|
"channel:`x`": "canale:`x`",
|
||||||
"Hindi": "Hindi",
|
"Deleted or invalid channel": "Canale eliminato o non valido",
|
||||||
"Hmong": "Hmong",
|
"This channel does not exist.": "Questo canale non esiste.",
|
||||||
"Hungarian": "Ungarese",
|
"Could not get channel info.": "Impossibile ottenere le informazioni del canale.",
|
||||||
"Icelandic": "Islandese",
|
"Could not fetch comments": "Impossibile recuperare i commenti",
|
||||||
"Igbo": "Igbo",
|
"View `x` replies": {
|
||||||
"Indonesian": "Indonesiano",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Visualizza `x` risposta",
|
||||||
"Irish": "Irlandese",
|
"": "Visualizza `x` risposte"
|
||||||
"Italian": "Italiano",
|
},
|
||||||
"Japanese": "Giapponese",
|
"`x` ago": "`x` fa",
|
||||||
"Javanese": "Giavanese",
|
"Load more": "Carica altro",
|
||||||
"Kannada": "Kannada",
|
"`x` points": {
|
||||||
"Kazakh": "Kazaco",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` punto",
|
||||||
"Khmer": "Khmer",
|
"": "`x` punti"
|
||||||
"Korean": "Coreano",
|
},
|
||||||
"Kurdish": "Curdo",
|
"Could not create mix.": "Impossibile creare il mix.",
|
||||||
"Kyrgyz": "Kirghize",
|
"Empty playlist": "Playlist vuota",
|
||||||
"Lao": "Lao",
|
"Not a playlist.": "Non è una playlist.",
|
||||||
"Latin": "Latino",
|
"Playlist does not exist.": "La playlist non esiste.",
|
||||||
"Latvian": "Lettone",
|
"Could not pull trending pages.": "Impossibile recuperare le tendenze.",
|
||||||
"Lithuanian": "Lituano",
|
"Hidden field \"challenge\" is a required field": "Il campo nascosto \"challenge\" è obbligatorio",
|
||||||
"Luxembourgish": "Lussemburghese",
|
"Hidden field \"token\" is a required field": "Il campo nascosto \"token\" è obbligatorio",
|
||||||
"Macedonian": "Macedone",
|
"Erroneous challenge": "Campo \"challenge\" non valido",
|
||||||
"Malagasy": "Malgascio",
|
"Erroneous token": "Campo \"token\" non valido",
|
||||||
"Malay": "Malese",
|
"No such user": "Utente non valido",
|
||||||
"Malayalam": "Lingua malayalam",
|
"Token is expired, please try again": "Gettone scaduto, riprova",
|
||||||
"Maltese": "Maltese",
|
"English": "Inglese",
|
||||||
"Maori": "Maori",
|
"English (auto-generated)": "Inglese (generati automaticamente)",
|
||||||
"Marathi": "Marathi",
|
"Afrikaans": "Afrikaans",
|
||||||
"Mongolian": "Mongolo",
|
"Albanian": "Albanese",
|
||||||
"Nepali": "Nepalese",
|
"Amharic": "Amarico",
|
||||||
"Norwegian": "Norvegese",
|
"Arabic": "Arabo",
|
||||||
"Nyanja": "Nyanja",
|
"Armenian": "Armeno",
|
||||||
"Pashto": "Lingua pashtu",
|
"Azerbaijani": "Azero",
|
||||||
"Persian": "Persiano",
|
"Bangla": "Bengalese",
|
||||||
"Polish": "Polacco",
|
"Basque": "Basco",
|
||||||
"Portuguese": "Portoghese",
|
"Belarusian": "Biellorusso",
|
||||||
"Punjabi": "Punjabi",
|
"Bosnian": "Bosniaco",
|
||||||
"Romanian": "Rumeno",
|
"Bulgarian": "Bulgaro",
|
||||||
"Russian": "Russo",
|
"Burmese": "Birmano",
|
||||||
"Samoan": "Samoan",
|
"Catalan": "Catalano",
|
||||||
"Scottish Gaelic": "Gaelico scozzese",
|
"Cebuano": "Sugbuanon",
|
||||||
"Serbian": "Serbo",
|
"Chinese (Simplified)": "Cinese semplifiato",
|
||||||
"Shona": "Shona",
|
"Chinese (Traditional)": "Cinese tradizionale",
|
||||||
"Sindhi": "Sindhi",
|
"Corsican": "Corso",
|
||||||
"Sinhala": "Cingalese",
|
"Croatian": "Croato",
|
||||||
"Slovak": "Slovacco",
|
"Czech": "Ceco",
|
||||||
"Slovenian": "Sloveno",
|
"Danish": "Danese",
|
||||||
"Somali": "Somalo",
|
"Dutch": "Olandese",
|
||||||
"Southern Sotho": "Sotho del Sud",
|
"Esperanto": "Esperanto",
|
||||||
"Spanish": "Spagnolo",
|
"Estonian": "Estone",
|
||||||
"Spanish (Latin America)": "Spagnolo (America latina)",
|
"Filipino": "Filippino",
|
||||||
"Sundanese": "Sudanese",
|
"Finnish": "Finlandese",
|
||||||
"Swahili": "Swahili",
|
"French": "Francese",
|
||||||
"Swedish": "Svedese",
|
"Galician": "Galiziano",
|
||||||
"Tajik": "Tajik",
|
"Georgian": "Georgiano",
|
||||||
"Tamil": "Tamil",
|
"German": "Tedesco",
|
||||||
"Telugu": "Telugu",
|
"Greek": "Greco",
|
||||||
"Thai": "Thaï",
|
"Gujarati": "Gujarati",
|
||||||
"Turkish": "Turco",
|
"Haitian Creole": "Creolo haitiano",
|
||||||
"Ukrainian": "Ucraino",
|
"Hausa": "Lingua hausa",
|
||||||
"Urdu": "Urdu",
|
"Hawaiian": "Hawaiano",
|
||||||
"Uzbek": "Uzbeco",
|
"Hebrew": "Ebreo",
|
||||||
"Vietnamese": "Vietnamese",
|
"Hindi": "Hindi",
|
||||||
"Welsh": "Gallese",
|
"Hmong": "Hmong",
|
||||||
"Western Frisian": "Frisone occidentale",
|
"Hungarian": "Ungarese",
|
||||||
"Xhosa": "Xhosa",
|
"Icelandic": "Islandese",
|
||||||
"Yiddish": "Yiddish",
|
"Igbo": "Igbo",
|
||||||
"Yoruba": "Yoruba",
|
"Indonesian": "Indonesiano",
|
||||||
"Zulu": "Zulu",
|
"Irish": "Irlandese",
|
||||||
"`x` years": "`x` anni",
|
"Italian": "Italiano",
|
||||||
"`x` months": "`x` mesi",
|
"Japanese": "Giapponese",
|
||||||
"`x` weeks": "`x` settimane",
|
"Javanese": "Giavanese",
|
||||||
"`x` days": "`x` giorni",
|
"Kannada": "Kannada",
|
||||||
"`x` hours": "`x` ore",
|
"Kazakh": "Kazaco",
|
||||||
"`x` minutes": "`x` minuti",
|
"Khmer": "Khmer",
|
||||||
"`x` seconds": "`x` secondi",
|
"Korean": "Coreano",
|
||||||
"Fallback comments: ": "Commenti alternativi: ",
|
"Kurdish": "Curdo",
|
||||||
"Popular": "Popolare",
|
"Kyrgyz": "Kirghize",
|
||||||
"Top": "Top",
|
"Lao": "Lao",
|
||||||
"About": "A proposito",
|
"Latin": "Latino",
|
||||||
"Rating: ": "Punteggio: ",
|
"Latvian": "Lettone",
|
||||||
"Language: ": "Lingua: ",
|
"Lithuanian": "Lituano",
|
||||||
"Default": "Predefinito",
|
"Luxembourgish": "Lussemburghese",
|
||||||
"Music": "Musica",
|
"Macedonian": "Macedone",
|
||||||
"Gaming": "Videogiochi",
|
"Malagasy": "Malgascio",
|
||||||
"News": "Notizie",
|
"Malay": "Malese",
|
||||||
"Movies": "Film",
|
"Malayalam": "Lingua malayalam",
|
||||||
"Download": "Scarica",
|
"Maltese": "Maltese",
|
||||||
"Download as: ": "Scarica come: ",
|
"Maori": "Maori",
|
||||||
"%A %B %-d, %Y": "%A %-d %B %Y",
|
"Marathi": "Marathi",
|
||||||
"(edited)": "(modificato)",
|
"Mongolian": "Mongolo",
|
||||||
"Youtube permalink of the comment": "Link permanente al commento di YouTube",
|
"Nepali": "Nepalese",
|
||||||
"`x` marked it with a ❤": "`x` l'ha contrassegnato con un ❤",
|
"Norwegian Bokmål": "Norvegese",
|
||||||
"Audio mode": "Modalità audio",
|
"Nyanja": "Nyanja",
|
||||||
"Video mode": "Modalità video",
|
"Pashto": "Lingua pashtu",
|
||||||
"Videos": "",
|
"Persian": "Persiano",
|
||||||
"Playlists": "",
|
"Polish": "Polacco",
|
||||||
"Current version: ": ""
|
"Portuguese": "Portoghese",
|
||||||
}
|
"Punjabi": "Punjabi",
|
||||||
|
"Romanian": "Rumeno",
|
||||||
|
"Russian": "Russo",
|
||||||
|
"Samoan": "Samoan",
|
||||||
|
"Scottish Gaelic": "Gaelico scozzese",
|
||||||
|
"Serbian": "Serbo",
|
||||||
|
"Shona": "Shona",
|
||||||
|
"Sindhi": "Sindhi",
|
||||||
|
"Sinhala": "Cingalese",
|
||||||
|
"Slovak": "Slovacco",
|
||||||
|
"Slovenian": "Sloveno",
|
||||||
|
"Somali": "Somalo",
|
||||||
|
"Southern Sotho": "Sotho del Sud",
|
||||||
|
"Spanish": "Spagnolo",
|
||||||
|
"Spanish (Latin America)": "Spagnolo (America latina)",
|
||||||
|
"Sundanese": "Sudanese",
|
||||||
|
"Swahili": "Swahili",
|
||||||
|
"Swedish": "Svedese",
|
||||||
|
"Tajik": "Tajik",
|
||||||
|
"Tamil": "Tamil",
|
||||||
|
"Telugu": "Telugu",
|
||||||
|
"Thai": "Thaï",
|
||||||
|
"Turkish": "Turco",
|
||||||
|
"Ukrainian": "Ucraino",
|
||||||
|
"Urdu": "Urdu",
|
||||||
|
"Uzbek": "Uzbeco",
|
||||||
|
"Vietnamese": "Vietnamese",
|
||||||
|
"Welsh": "Gallese",
|
||||||
|
"Western Frisian": "Frisone occidentale",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Yiddish",
|
||||||
|
"Yoruba": "Yoruba",
|
||||||
|
"Zulu": "Zulu",
|
||||||
|
"`x` years": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` anno",
|
||||||
|
"": "`x` anni"
|
||||||
|
},
|
||||||
|
"`x` months": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` mese",
|
||||||
|
"": "`x` mesi"
|
||||||
|
},
|
||||||
|
"`x` weeks": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` settimana",
|
||||||
|
"": "`x` settimane"
|
||||||
|
},
|
||||||
|
"`x` days": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` giorno",
|
||||||
|
"": "`x` giorni"
|
||||||
|
},
|
||||||
|
"`x` hours": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` ora",
|
||||||
|
"": "`x` ore"
|
||||||
|
},
|
||||||
|
"`x` minutes": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` minuto",
|
||||||
|
"": "`x` minuti"
|
||||||
|
},
|
||||||
|
"`x` seconds": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` secondo",
|
||||||
|
"": "`x` secondi"
|
||||||
|
},
|
||||||
|
"Fallback comments: ": "Commenti alternativi: ",
|
||||||
|
"Popular": "Popolare",
|
||||||
|
"Top": "Top",
|
||||||
|
"About": "Al riguardo",
|
||||||
|
"Rating: ": "Punteggio: ",
|
||||||
|
"Language: ": "Lingua: ",
|
||||||
|
"View as playlist": "Vedi come playlist",
|
||||||
|
"Default": "Predefinito",
|
||||||
|
"Music": "Musica",
|
||||||
|
"Gaming": "Videogiochi",
|
||||||
|
"News": "Notizie",
|
||||||
|
"Movies": "Film",
|
||||||
|
"Download": "Scarica",
|
||||||
|
"Download as: ": "Scarica come: ",
|
||||||
|
"%A %B %-d, %Y": "%A %-d %B %Y",
|
||||||
|
"(edited)": "(modificato)",
|
||||||
|
"YouTube comment permalink": "Link permanente al commento di YouTube",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "`x` l'ha contrassegnato con un ❤",
|
||||||
|
"Audio mode": "Modalità audio",
|
||||||
|
"Video mode": "Modalità video",
|
||||||
|
"Videos": "Video",
|
||||||
|
"Playlists": "Playlist",
|
||||||
|
"Community": "Comunità",
|
||||||
|
"Current version: ": "Versione attuale: "
|
||||||
|
}
|
||||||
387
locales/ja.json
Normal file
387
locales/ja.json
Normal file
@@ -0,0 +1,387 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 人の登録者",
|
||||||
|
"": "`x` 人の登録者"
|
||||||
|
},
|
||||||
|
"`x` videos": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個の動画",
|
||||||
|
"": "`x` 個の動画"
|
||||||
|
},
|
||||||
|
"`x` playlists": {
|
||||||
|
"(\\D|^)1(\\D|$)": "`x` 個の再生リスト",
|
||||||
|
"": "`x` 個の再生リスト"
|
||||||
|
},
|
||||||
|
"LIVE": "ライブ",
|
||||||
|
"Shared `x` ago": "`x`前に共有",
|
||||||
|
"Unsubscribe": "登録解除",
|
||||||
|
"Subscribe": "登録",
|
||||||
|
"View channel on YouTube": "YouTube でチャンネルを見る",
|
||||||
|
"View playlist on YouTube": "YouTube で再生リストを見る",
|
||||||
|
"newest": "新しい順",
|
||||||
|
"oldest": "古い順",
|
||||||
|
"popular": "人気順",
|
||||||
|
"last": "追加順",
|
||||||
|
"Next page": "次のページ",
|
||||||
|
"Previous page": "前のページ",
|
||||||
|
"Clear watch history?": "再生履歴を削除しますか?",
|
||||||
|
"New password": "新しいパスワード",
|
||||||
|
"New passwords must match": "新しいパスワードが一致していません",
|
||||||
|
"Cannot change password for Google accounts": "Google アカウントのパスワードは変更できません",
|
||||||
|
"Authorize token?": "トークンを認証しますか?",
|
||||||
|
"Authorize token for `x`?": "トークン `x` を認証しますか?",
|
||||||
|
"Yes": "はい",
|
||||||
|
"No": "いいえ",
|
||||||
|
"Import and Export Data": "データのインポートとエクスポート",
|
||||||
|
"Import": "インポート",
|
||||||
|
"Import Invidious data": "Invidious データをインポート",
|
||||||
|
"Import YouTube subscriptions": "YouTube 登録チャンネルをインポート",
|
||||||
|
"Import FreeTube subscriptions (.db)": "FreeTube 登録チャンネルをインポート (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "NewPipe 登録チャンネルをインポート (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "NewPipe データをインポート (.zip)",
|
||||||
|
"Export": "エクスポート",
|
||||||
|
"Export subscriptions as OPML": "登録チャンネルを OPML でエクスポート",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "登録チャンネルを OPML でエクスポート (NewPipe & FreeTube 用)",
|
||||||
|
"Export data as JSON": "データを JSON でエクスポート",
|
||||||
|
"Delete account?": "アカウントを削除しますか?",
|
||||||
|
"History": "履歴",
|
||||||
|
"An alternative front-end to YouTube": "YouTube の代わりとなる新しいフロントエンド",
|
||||||
|
"JavaScript license information": "JavaScript ライセンス情報",
|
||||||
|
"source": "ソース",
|
||||||
|
"Log in": "ログイン",
|
||||||
|
"Log in/register": "ログイン/登録",
|
||||||
|
"Log in with Google": "Google でログイン",
|
||||||
|
"User ID": "ユーザー ID",
|
||||||
|
"Password": "パスワード",
|
||||||
|
"Time (h:mm:ss):": "時間 (時:分分:秒秒):",
|
||||||
|
"Text CAPTCHA": "テキスト CAPTCHA",
|
||||||
|
"Image CAPTCHA": "画像 CAPTCHA",
|
||||||
|
"Sign In": "サインイン",
|
||||||
|
"Register": "登録",
|
||||||
|
"E-mail": "メールアドレス",
|
||||||
|
"Google verification code": "Google 認証コード",
|
||||||
|
"Preferences": "設定",
|
||||||
|
"Player preferences": "プレイヤー設定",
|
||||||
|
"Always loop: ": "常にループ: ",
|
||||||
|
"Autoplay: ": "自動再生: ",
|
||||||
|
"Play next by default: ": "デフォルトで次を再生: ",
|
||||||
|
"Autoplay next video: ": "次の動画を自動再生: ",
|
||||||
|
"Listen by default: ": "デフォルトでオーディオモードを使用: ",
|
||||||
|
"Proxy videos: ": "動画をプロキシーに通す: ",
|
||||||
|
"Default speed: ": "デフォルトの再生速度: ",
|
||||||
|
"Preferred video quality: ": "優先する画質: ",
|
||||||
|
"Player volume: ": "プレイヤーの音量: ",
|
||||||
|
"Default comments: ": "デフォルトのコメント: ",
|
||||||
|
"youtube": "youtube",
|
||||||
|
"reddit": "reddit",
|
||||||
|
"Default captions: ": "デフォルトの字幕: ",
|
||||||
|
"Fallback captions: ": "フォールバック時の字幕: ",
|
||||||
|
"Show related videos: ": "関連動画を表示: ",
|
||||||
|
"Show annotations by default: ": "デフォルトでアノテーションを表示: ",
|
||||||
|
"Visual preferences": "外観設定",
|
||||||
|
"Player style: ": "プレイヤースタイル: ",
|
||||||
|
"Dark mode: ": "ダークモード: ",
|
||||||
|
"Theme: ": "テーマ: ",
|
||||||
|
"dark": "ダーク",
|
||||||
|
"light": "ライト",
|
||||||
|
"Thin mode: ": "最小モード: ",
|
||||||
|
"Subscription preferences": "登録チャンネル設定",
|
||||||
|
"Show annotations by default for subscribed channels: ": "デフォルトで登録チャンネルのアノテーションを表示しますか? ",
|
||||||
|
"Redirect homepage to feed: ": "ホームからフィードにリダイレクト: ",
|
||||||
|
"Number of videos shown in feed: ": "フィードに表示する動画の量: ",
|
||||||
|
"Sort videos by: ": "動画を並び替え: ",
|
||||||
|
"published": "投稿日",
|
||||||
|
"published - reverse": "投稿日 - 逆順",
|
||||||
|
"alphabetically": "アルファベット",
|
||||||
|
"alphabetically - reverse": "アルファベット - 逆順",
|
||||||
|
"channel name": "チャンネル名",
|
||||||
|
"channel name - reverse": "チャンネル名 - 逆順",
|
||||||
|
"Only show latest video from channel: ": "チャンネルの最新動画のみを表示: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "チャンネルの最新未視聴動画のみを表示: ",
|
||||||
|
"Only show unwatched: ": "未視聴のみを表示: ",
|
||||||
|
"Only show notifications (if there are any): ": "通知のみを表示 (ある場合): ",
|
||||||
|
"Enable web notifications": "ウェブ通知を有効化",
|
||||||
|
"`x` uploaded a video": "`x` が動画を投稿しました",
|
||||||
|
"`x` is live": "`x` がライブ中です",
|
||||||
|
"Data preferences": "データ設定",
|
||||||
|
"Clear watch history": "再生履歴の削除",
|
||||||
|
"Import/export data": "データのインポート/エクスポート",
|
||||||
|
"Change password": "パスワードを変更",
|
||||||
|
"Manage subscriptions": "登録チャンネルを管理",
|
||||||
|
"Manage tokens": "トークンを管理",
|
||||||
|
"Watch history": "再生履歴",
|
||||||
|
"Delete account": "アカウントを削除",
|
||||||
|
"Administrator preferences": "管理者設定",
|
||||||
|
"Default homepage: ": "デフォルトのホーム: ",
|
||||||
|
"Feed menu: ": "フィードメニュー: ",
|
||||||
|
"Top enabled: ": "Top enabled: ",
|
||||||
|
"CAPTCHA enabled: ": "CAPTCHA を有効化: ",
|
||||||
|
"Login enabled: ": "ログインを有効化: ",
|
||||||
|
"Registration enabled: ": "登録を有効化: ",
|
||||||
|
"Report statistics: ": "統計を報告: ",
|
||||||
|
"Save preferences": "設定を保存",
|
||||||
|
"Subscription manager": "登録チャンネルマネージャー",
|
||||||
|
"Token manager": "トークンマネージャー",
|
||||||
|
"Token": "トークン",
|
||||||
|
"`x` subscriptions": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個の登録チャンネル",
|
||||||
|
"": "`x` 個の登録チャンネル"
|
||||||
|
},
|
||||||
|
"`x` tokens": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個のトークン",
|
||||||
|
"": "`x` 個のトークン"
|
||||||
|
},
|
||||||
|
"Import/export": "インポート/エクスポート",
|
||||||
|
"unsubscribe": "登録解除",
|
||||||
|
"revoke": "revoke",
|
||||||
|
"Subscriptions": "登録チャンネル",
|
||||||
|
"`x` unseen notifications": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個の未読通知",
|
||||||
|
"": "`x` 個の未読通知"
|
||||||
|
},
|
||||||
|
"search": "検索",
|
||||||
|
"Log out": "ログアウト",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Omar Roth によって AGPLv3 でリリースされています。",
|
||||||
|
"Source available here.": "ソースはここで閲覧可能です。",
|
||||||
|
"View JavaScript license information.": "JavaScript ライセンス情報を見る。",
|
||||||
|
"View privacy policy.": "プライバシーポリシーを見る。",
|
||||||
|
"Trending": "急上昇",
|
||||||
|
"Public": "公開",
|
||||||
|
"Unlisted": "限定公開",
|
||||||
|
"Private": "非公開",
|
||||||
|
"View all playlists": "再生リストをすべて見る",
|
||||||
|
"Updated `x` ago": "`x`前に更新",
|
||||||
|
"Delete playlist `x`?": "再生リスト `x` を削除しますか?",
|
||||||
|
"Delete playlist": "再生リストを削除",
|
||||||
|
"Create playlist": "再生リストを作成",
|
||||||
|
"Title": "タイトル",
|
||||||
|
"Playlist privacy": "再生リストのプライバシー",
|
||||||
|
"Editing playlist `x`": "再生リスト `x` を編集中",
|
||||||
|
"Watch on YouTube": "YouTube で視聴",
|
||||||
|
"Hide annotations": "アノテーションを隠す",
|
||||||
|
"Show annotations": "アノテーションを表示",
|
||||||
|
"Genre: ": "ジャンル: ",
|
||||||
|
"License: ": "ライセンス: ",
|
||||||
|
"Family friendly? ": "家族向け? ",
|
||||||
|
"Wilson score: ": "ウィルソンスコア: ",
|
||||||
|
"Engagement: ": "エンゲージメント: ",
|
||||||
|
"Whitelisted regions: ": "ホワイトリストの地域: ",
|
||||||
|
"Blacklisted regions: ": "ブラックリストの地域: ",
|
||||||
|
"Shared `x`": "`x`に共有",
|
||||||
|
"`x` views": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 回視聴",
|
||||||
|
"": "`x` 回視聴"
|
||||||
|
},
|
||||||
|
"Premieres in `x`": "Premieres in `x`",
|
||||||
|
"Premieres `x`": "Premieres `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "やあ!君は JavaScript を無効にしているのかな?ここをクリックしてコメントを見れるけど、読み込みには少し時間がかかることがあるのを覚えておいてね。",
|
||||||
|
"View YouTube comments": "YouTube のコメントを見る",
|
||||||
|
"View more comments on Reddit": "Reddit でコメントをもっと見る",
|
||||||
|
"View `x` comments": {
|
||||||
|
"(\\D|^)1(\\D|$)": "`x` 件のコメントを見る",
|
||||||
|
"": "`x` 件のコメントを見る"
|
||||||
|
},
|
||||||
|
"View Reddit comments": "Reddit のコメントを見る",
|
||||||
|
"Hide replies": "返信を非表示",
|
||||||
|
"Show replies": "返信を表示",
|
||||||
|
"Incorrect password": "パスワードが間違っています",
|
||||||
|
"Quota exceeded, try again in a few hours": "試行を制限中です。数時間後にやり直してください",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "ログインできませんでした。2段階認証 (認証アプリまたは SMS) が有効になっていることを確認してください。",
|
||||||
|
"Invalid TFA code": "TFA (2段階認証) コードが無効です",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "ログインに失敗しました。あなたのアカウントで2段階認証が有効になっていない可能性があります。",
|
||||||
|
"Wrong answer": "回答が間違っています",
|
||||||
|
"Erroneous CAPTCHA": "CAPTCHA が間違っています",
|
||||||
|
"CAPTCHA is a required field": "CAPTCHA は必須項目です",
|
||||||
|
"User ID is a required field": "ユーザー ID は必須項目です",
|
||||||
|
"Password is a required field": "パスワードは必須項目です",
|
||||||
|
"Wrong username or password": "ユーザー名またはパスワードが間違っています",
|
||||||
|
"Please sign in using 'Log in with Google'": "'Google でログイン' を使用してログインしてください",
|
||||||
|
"Password cannot be empty": "パスワードを空にすることはできません",
|
||||||
|
"Password cannot be longer than 55 characters": "パスワードは55文字より長くできません",
|
||||||
|
"Please log in": "ログインをしてください",
|
||||||
|
"Invidious Private Feed for `x`": "`x` の Invidious プライベートフィード",
|
||||||
|
"channel:`x`": "チャンネル:`x`",
|
||||||
|
"Deleted or invalid channel": "削除済みまたは無効なチャンネルです",
|
||||||
|
"This channel does not exist.": "このチャンネルは存在していません",
|
||||||
|
"Could not get channel info.": "チャンネル情報を取得できませんでした。",
|
||||||
|
"Could not fetch comments": "コメントを取得できませんでした",
|
||||||
|
"View `x` replies": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 件の返信を見る",
|
||||||
|
"": "`x` 件の返信を見る"
|
||||||
|
},
|
||||||
|
"`x` ago": "`x`前",
|
||||||
|
"Load more": "もっと読み込む",
|
||||||
|
"`x` points": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` ポイント",
|
||||||
|
"": "`x` ポイント"
|
||||||
|
},
|
||||||
|
"Could not create mix.": "ミックスを作成できませんでした。",
|
||||||
|
"Empty playlist": "空の再生リスト",
|
||||||
|
"Not a playlist.": "再生リストではありません。",
|
||||||
|
"Playlist does not exist.": "再生リストが存在していません・",
|
||||||
|
"Could not pull trending pages.": "急上昇ページを取得できませんでした。",
|
||||||
|
"Hidden field \"challenge\" is a required field": "非表示項目 \"challenge\" は必須項目です",
|
||||||
|
"Hidden field \"token\" is a required field": "非表示項目 \"token\" は必須項目です",
|
||||||
|
"Erroneous challenge": "チャレンジが間違っています",
|
||||||
|
"Erroneous token": "トークンが間違っています",
|
||||||
|
"No such user": "ユーザーが存在しません",
|
||||||
|
"Token is expired, please try again": "トークンが期限切れです。再度試してください",
|
||||||
|
"English": "英語",
|
||||||
|
"English (auto-generated)": "英語 (自動生成)",
|
||||||
|
"Afrikaans": "アフリカーンス語",
|
||||||
|
"Albanian": "アルバニア語",
|
||||||
|
"Amharic": "アムハラ語",
|
||||||
|
"Arabic": "アラビア語",
|
||||||
|
"Armenian": "アルメニア語",
|
||||||
|
"Azerbaijani": "アゼルバイジャン語",
|
||||||
|
"Bangla": "ベンガル語",
|
||||||
|
"Basque": "バスク語",
|
||||||
|
"Belarusian": "ベラルーシ語",
|
||||||
|
"Bosnian": "ボスニア語",
|
||||||
|
"Bulgarian": "ブルガリア語",
|
||||||
|
"Burmese": "ビルマ語",
|
||||||
|
"Catalan": "カタルーニャ語",
|
||||||
|
"Cebuano": "セブアノ語",
|
||||||
|
"Chinese (Simplified)": "中国語 (簡体字)",
|
||||||
|
"Chinese (Traditional)": "中国語 (繁体字)",
|
||||||
|
"Corsican": "コルシカ語",
|
||||||
|
"Croatian": "クロアチア語",
|
||||||
|
"Czech": "チェコ語",
|
||||||
|
"Danish": "デンマーク語",
|
||||||
|
"Dutch": "オランダ語",
|
||||||
|
"Esperanto": "エスペラント語",
|
||||||
|
"Estonian": "エストニア語",
|
||||||
|
"Filipino": "フィリピン語",
|
||||||
|
"Finnish": "フィンランド語",
|
||||||
|
"French": "フランス語",
|
||||||
|
"Galician": "ガルシア語",
|
||||||
|
"Georgian": "グルジア語",
|
||||||
|
"German": "ドイツ語",
|
||||||
|
"Greek": "ギリシャ語",
|
||||||
|
"Gujarati": "グジャラート語",
|
||||||
|
"Haitian Creole": "ハイチ語",
|
||||||
|
"Hausa": "ハウサ語",
|
||||||
|
"Hawaiian": "ハワイ語",
|
||||||
|
"Hebrew": "ヘブライ語",
|
||||||
|
"Hindi": "ヒンディー語",
|
||||||
|
"Hmong": "ミャオ語",
|
||||||
|
"Hungarian": "ハンガリー語",
|
||||||
|
"Icelandic": "アイスランド語",
|
||||||
|
"Igbo": "イボ語",
|
||||||
|
"Indonesian": "インドネシア語",
|
||||||
|
"Irish": "アイルランド語",
|
||||||
|
"Italian": "イタリア語",
|
||||||
|
"Japanese": "日本語",
|
||||||
|
"Javanese": "ジャワ語",
|
||||||
|
"Kannada": "カンナダ語",
|
||||||
|
"Kazakh": "カザフ語",
|
||||||
|
"Khmer": "クメール語",
|
||||||
|
"Korean": "韓国語",
|
||||||
|
"Kurdish": "クルド語",
|
||||||
|
"Kyrgyz": "キルギス語",
|
||||||
|
"Lao": "ラーオ語",
|
||||||
|
"Latin": "ラテン語",
|
||||||
|
"Latvian": "ラトビア語",
|
||||||
|
"Lithuanian": "リトアニア語",
|
||||||
|
"Luxembourgish": "ルクセンブルク語",
|
||||||
|
"Macedonian": "マケドニア語",
|
||||||
|
"Malagasy": "マダガスカル語",
|
||||||
|
"Malay": "マレー語",
|
||||||
|
"Malayalam": "マラヤーラム語",
|
||||||
|
"Maltese": "マルタ語",
|
||||||
|
"Maori": "マオリ語",
|
||||||
|
"Marathi": "マラーティー語",
|
||||||
|
"Mongolian": "モンゴル語",
|
||||||
|
"Nepali": "ネパール語",
|
||||||
|
"Norwegian Bokmål": "ノルウェー語",
|
||||||
|
"Nyanja": "チェワ語",
|
||||||
|
"Pashto": "パシュトー語",
|
||||||
|
"Persian": "ペルシア語",
|
||||||
|
"Polish": "ポーランド語",
|
||||||
|
"Portuguese": "ポルトガル語",
|
||||||
|
"Punjabi": "パンジャーブ語",
|
||||||
|
"Romanian": "ルーマニア語",
|
||||||
|
"Russian": "ロシア語",
|
||||||
|
"Samoan": "サモア語",
|
||||||
|
"Scottish Gaelic": "スコットランド・ゲール語",
|
||||||
|
"Serbian": "セルビア語",
|
||||||
|
"Shona": "ショナ語",
|
||||||
|
"Sindhi": "シンド語",
|
||||||
|
"Sinhala": "シンハラ語",
|
||||||
|
"Slovak": "スロバキア語",
|
||||||
|
"Slovenian": "スロベニア語",
|
||||||
|
"Somali": "ソマリ語",
|
||||||
|
"Southern Sotho": "南ソト語",
|
||||||
|
"Spanish": "スペイン語",
|
||||||
|
"Spanish (Latin America)": "スペイン語 (ラテンアメリカ)",
|
||||||
|
"Sundanese": "スンダ語",
|
||||||
|
"Swahili": "スワヒリ語",
|
||||||
|
"Swedish": "スウェーデン語",
|
||||||
|
"Tajik": "タジク語",
|
||||||
|
"Tamil": "タミル語",
|
||||||
|
"Telugu": "テルグ語",
|
||||||
|
"Thai": "タイ語",
|
||||||
|
"Turkish": "トルコ語",
|
||||||
|
"Ukrainian": "ウクライナ語",
|
||||||
|
"Urdu": "ウルドゥー語",
|
||||||
|
"Uzbek": "ウズベク語",
|
||||||
|
"Vietnamese": "ベトナム語",
|
||||||
|
"Welsh": "ウェールズ語",
|
||||||
|
"Western Frisian": "西フリジア語",
|
||||||
|
"Xhosa": "コサ語",
|
||||||
|
"Yiddish": "イディッシュ語",
|
||||||
|
"Yoruba": "ヨルバ語",
|
||||||
|
"Zulu": "ズール語",
|
||||||
|
"`x` years": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`年",
|
||||||
|
"": "`x`年"
|
||||||
|
},
|
||||||
|
"`x` months": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`月",
|
||||||
|
"": "`x`月"
|
||||||
|
},
|
||||||
|
"`x` weeks": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`週",
|
||||||
|
"": "`x`週"
|
||||||
|
},
|
||||||
|
"`x` days": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`日",
|
||||||
|
"": "`x`日"
|
||||||
|
},
|
||||||
|
"`x` hours": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`時間",
|
||||||
|
"": "`x`時間"
|
||||||
|
},
|
||||||
|
"`x` minutes": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`分",
|
||||||
|
"": "`x`分"
|
||||||
|
},
|
||||||
|
"`x` seconds": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x`秒",
|
||||||
|
"": "`x`秒"
|
||||||
|
},
|
||||||
|
"Fallback comments: ": "フォールバック時のコメント: ",
|
||||||
|
"Popular": "人気",
|
||||||
|
"Top": "トップ",
|
||||||
|
"About": "このサービスについて",
|
||||||
|
"Rating: ": "評価: ",
|
||||||
|
"Language: ": "言語: ",
|
||||||
|
"View as playlist": "再生リストで見る",
|
||||||
|
"Default": "デフォルト",
|
||||||
|
"Music": "音楽",
|
||||||
|
"Gaming": "ゲーム",
|
||||||
|
"News": "ニュース",
|
||||||
|
"Movies": "映画",
|
||||||
|
"Download": "ダウンロード",
|
||||||
|
"Download as: ": "ダウンロード: ",
|
||||||
|
"%A %B %-d, %Y": "%Y %B %-d %A",
|
||||||
|
"(edited)": "(編集済み)",
|
||||||
|
"YouTube comment permalink": "YouTube コメントのパーマリンク",
|
||||||
|
"permalink": "パーマリンク",
|
||||||
|
"`x` marked it with a ❤": "`x` が❤を込めてマークしました",
|
||||||
|
"Audio mode": "オーディオモード",
|
||||||
|
"Video mode": "ビデオモード",
|
||||||
|
"Videos": "動画",
|
||||||
|
"Playlists": "プレイリスト",
|
||||||
|
"Community": "コミュニティ",
|
||||||
|
"Current version: ": "現在のバージョン: "
|
||||||
|
}
|
||||||
@@ -1,12 +1,13 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` abonnenter",
|
"`x` subscribers": "`x` abonnenter",
|
||||||
"`x` videos": "`x` videoer",
|
"`x` videos": "`x` videoer",
|
||||||
|
"`x` playlists": "",
|
||||||
"LIVE": "SANNTIDSVISNING",
|
"LIVE": "SANNTIDSVISNING",
|
||||||
"Shared `x` ago": "Delt for `x` siden",
|
"Shared `x` ago": "Delt for `x` siden",
|
||||||
"Unsubscribe": "Opphev abonnement",
|
"Unsubscribe": "Opphev abonnement",
|
||||||
"Subscribe": "Abonner",
|
"Subscribe": "Abonner",
|
||||||
"Login to subscribe to `x`": "Logg inn for å abonnere på `x`",
|
|
||||||
"View channel on YouTube": "Vis kanal på YouTube",
|
"View channel on YouTube": "Vis kanal på YouTube",
|
||||||
|
"View playlist on YouTube": "Vis spilleliste på YouTube",
|
||||||
"newest": "nyeste",
|
"newest": "nyeste",
|
||||||
"oldest": "eldste",
|
"oldest": "eldste",
|
||||||
"popular": "populært",
|
"popular": "populært",
|
||||||
@@ -14,6 +15,11 @@
|
|||||||
"Next page": "Neste side",
|
"Next page": "Neste side",
|
||||||
"Previous page": "Forrige side",
|
"Previous page": "Forrige side",
|
||||||
"Clear watch history?": "Tøm visningshistorikk?",
|
"Clear watch history?": "Tøm visningshistorikk?",
|
||||||
|
"New password": "Nytt passord",
|
||||||
|
"New passwords must match": "Nye passordfelter må stemme overens",
|
||||||
|
"Cannot change password for Google accounts": "Kan ikke endre passord for Google-kontoer",
|
||||||
|
"Authorize token?": "Identitetsbekreft symbol?",
|
||||||
|
"Authorize token for `x`?": "Identitetsbekreft symbol for `x`?",
|
||||||
"Yes": "Ja",
|
"Yes": "Ja",
|
||||||
"No": "Nei",
|
"No": "Nei",
|
||||||
"Import and Export Data": "Importer- og eksporter data",
|
"Import and Export Data": "Importer- og eksporter data",
|
||||||
@@ -32,36 +38,45 @@
|
|||||||
"An alternative front-end to YouTube": "En alternativ grenseflate for YouTube",
|
"An alternative front-end to YouTube": "En alternativ grenseflate for YouTube",
|
||||||
"JavaScript license information": "JavaScript-lisensinformasjon",
|
"JavaScript license information": "JavaScript-lisensinformasjon",
|
||||||
"source": "kilde",
|
"source": "kilde",
|
||||||
"Login": "Logg inn",
|
"Log in": "Logg inn",
|
||||||
"Login/Register": "Logg inn/registrer",
|
"Log in/register": "Logg inn/registrer",
|
||||||
"Login to Google": "Logg inn med Google",
|
"Log in with Google": "Logg inn med Google",
|
||||||
"User ID:": "Bruker-ID:",
|
"User ID": "Bruker-ID",
|
||||||
"Password:": "Passord:",
|
"Password": "Passord",
|
||||||
"Time (h:mm:ss):": "Tid (h:mm:ss):",
|
"Time (h:mm:ss):": "Tid (h:mm:ss):",
|
||||||
"Text CAPTCHA": "Tekst-CAPTCHA",
|
"Text CAPTCHA": "Tekst-CAPTCHA",
|
||||||
"Image CAPTCHA": "Bilde-CAPTCHA",
|
"Image CAPTCHA": "Bilde-CAPTCHA",
|
||||||
"Sign In": "Innlogging",
|
"Sign In": "Innlogging",
|
||||||
"Register": "Registrer",
|
"Register": "Registrer",
|
||||||
"Email:": "E-post:",
|
"E-mail": "E-post",
|
||||||
"Google verification code:": "Google-bekreftelseskode:",
|
"Google verification code": "Google-bekreftelseskode",
|
||||||
"Preferences": "Innstillinger",
|
"Preferences": "Innstillinger",
|
||||||
"Player preferences": "Avspillerinnstillinger",
|
"Player preferences": "Avspillerinnstillinger",
|
||||||
"Always loop: ": "Alltid gjenta: ",
|
"Always loop: ": "Alltid gjenta: ",
|
||||||
"Autoplay: ": "Autoavspilling: ",
|
"Autoplay: ": "Autoavspilling: ",
|
||||||
|
"Play next by default: ": "Spill neste som forvalg: ",
|
||||||
"Autoplay next video: ": "Autospill neste video: ",
|
"Autoplay next video: ": "Autospill neste video: ",
|
||||||
"Listen by default: ": "Lytt som forvalg: ",
|
"Listen by default: ": "Lytt som forvalg: ",
|
||||||
"Proxy videos? ": "Mellomtjen videoer? ",
|
"Proxy videos: ": "Mellomtjen videoer? ",
|
||||||
"Default speed: ": "Forvalgt hastighet: ",
|
"Default speed: ": "Forvalgt hastighet: ",
|
||||||
"Preferred video quality: ": "Foretrukket videokvalitet: ",
|
"Preferred video quality: ": "Foretrukket videokvalitet: ",
|
||||||
"Player volume: ": "Avspillerlydstyrke: ",
|
"Player volume: ": "Avspillerlydstyrke: ",
|
||||||
"Default comments: ": "Forvalgte kommentarer: ",
|
"Default comments: ": "Forvalgte kommentarer: ",
|
||||||
|
"youtube": "YouTube",
|
||||||
|
"reddit": "Reddit",
|
||||||
"Default captions: ": "Forvalgte undertitler: ",
|
"Default captions: ": "Forvalgte undertitler: ",
|
||||||
"Fallback captions: ": "Tilbakefallsundertitler: ",
|
"Fallback captions: ": "Tilbakefallsundertitler: ",
|
||||||
"Show related videos? ": "Vis relaterte videoer? ",
|
"Show related videos: ": "Vis relaterte videoer? ",
|
||||||
|
"Show annotations by default: ": "Vis merknader som forvalg? ",
|
||||||
"Visual preferences": "Visuelle innstillinger",
|
"Visual preferences": "Visuelle innstillinger",
|
||||||
|
"Player style: ": "Avspillerstil: ",
|
||||||
"Dark mode: ": "Mørk drakt: ",
|
"Dark mode: ": "Mørk drakt: ",
|
||||||
|
"Theme: ": "Drakt: ",
|
||||||
|
"dark": "Mørk",
|
||||||
|
"light": "Lys",
|
||||||
"Thin mode: ": "Tynt modus: ",
|
"Thin mode: ": "Tynt modus: ",
|
||||||
"Subscription preferences": "Abonnementsinnstillinger",
|
"Subscription preferences": "Abonnementsinnstillinger",
|
||||||
|
"Show annotations by default for subscribed channels: ": "Vis merknader som forvalg for kanaler det abonneres på? ",
|
||||||
"Redirect homepage to feed: ": "Videresend hjemmeside til flyt: ",
|
"Redirect homepage to feed: ": "Videresend hjemmeside til flyt: ",
|
||||||
"Number of videos shown in feed: ": "Antall videoer å vise i flyt: ",
|
"Number of videos shown in feed: ": "Antall videoer å vise i flyt: ",
|
||||||
"Sort videos by: ": "Sorter videoer etter: ",
|
"Sort videos by: ": "Sorter videoer etter: ",
|
||||||
@@ -75,36 +90,57 @@
|
|||||||
"Only show latest unwatched video from channel: ": "Kun vis siste usette video fra kanal: ",
|
"Only show latest unwatched video from channel: ": "Kun vis siste usette video fra kanal: ",
|
||||||
"Only show unwatched: ": "Kun vis usette: ",
|
"Only show unwatched: ": "Kun vis usette: ",
|
||||||
"Only show notifications (if there are any): ": "Kun vis merknader (hvis det er noen): ",
|
"Only show notifications (if there are any): ": "Kun vis merknader (hvis det er noen): ",
|
||||||
|
"Enable web notifications": "Skru på nettmerknader",
|
||||||
|
"`x` uploaded a video": "`x` lastet opp en video",
|
||||||
|
"`x` is live": "`x` er pålogget",
|
||||||
"Data preferences": "Datainnstillinger",
|
"Data preferences": "Datainnstillinger",
|
||||||
"Clear watch history": "Tøm visningshistorikk",
|
"Clear watch history": "Tøm visningshistorikk",
|
||||||
"Import/Export data": "Importer/eksporter data",
|
"Import/export data": "Importer/eksporter data",
|
||||||
|
"Change password": "Endre passord",
|
||||||
"Manage subscriptions": "Behandle abonnementer",
|
"Manage subscriptions": "Behandle abonnementer",
|
||||||
|
"Manage tokens": "Behandle symboler",
|
||||||
"Watch history": "Visningshistorikk",
|
"Watch history": "Visningshistorikk",
|
||||||
"Delete account": "Slett konto",
|
"Delete account": "Slett konto",
|
||||||
"Administrator preferences": "Administratorinnstillinger",
|
"Administrator preferences": "Administratorinnstillinger",
|
||||||
"Default homepage: ": "Forvalgt hjemmeside: ",
|
"Default homepage: ": "Forvalgt hjemmeside: ",
|
||||||
"Feed menu: ": "Flyt-meny: ",
|
"Feed menu: ": "Flyt-meny: ",
|
||||||
"Top enabled? ": "Topp påskrudd? ",
|
"Top enabled: ": "Topp påskrudd? ",
|
||||||
"CAPTCHA enabled? ": "CAPTCHA påskrudd? ",
|
"CAPTCHA enabled: ": "CAPTCHA påskrudd? ",
|
||||||
"Login enabled? ": "Innlogging påskrudd? ",
|
"Login enabled: ": "Innlogging påskrudd? ",
|
||||||
"Registration enabled? ": "Registrering påskrudd? ",
|
"Registration enabled: ": "Registrering påskrudd? ",
|
||||||
"Report statistics? ": "Innrapporter statistikk? ",
|
"Report statistics: ": "Innrapporter statistikk? ",
|
||||||
"Save preferences": "Lagre innstillinger",
|
"Save preferences": "Lagre innstillinger",
|
||||||
"Subscription manager": "Abonnementsbehandler",
|
"Subscription manager": "Abonnementsbehandler",
|
||||||
|
"Token manager": "Symbolbehandler",
|
||||||
|
"Token": "Symbol",
|
||||||
"`x` subscriptions": "`x` abonnementer",
|
"`x` subscriptions": "`x` abonnementer",
|
||||||
"Import/Export": "Importer/eksporter",
|
"`x` tokens": "`x` symboler",
|
||||||
|
"Import/export": "Importer/eksporter",
|
||||||
"unsubscribe": "opphev abonnement",
|
"unsubscribe": "opphev abonnement",
|
||||||
|
"revoke": "tilbakekall",
|
||||||
"Subscriptions": "Abonnement",
|
"Subscriptions": "Abonnement",
|
||||||
"`x` unseen notifications": "`x` usette merknader",
|
"`x` unseen notifications": "`x` usette merknader",
|
||||||
"search": "søk",
|
"search": "søk",
|
||||||
"Sign out": "Logg ut",
|
"Log out": "Logg ut",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Utgitt med AGPLv3+lisens av Omar Roth.",
|
"Released under the AGPLv3 by Omar Roth.": "Utgitt med AGPLv3+lisens av Omar Roth.",
|
||||||
"Source available here.": "Kildekode tilgjengelig her.",
|
"Source available here.": "Kildekode tilgjengelig her.",
|
||||||
"View JavaScript license information.": "Vis JavaScript-lisensinfo.",
|
"View JavaScript license information.": "Vis JavaScript-lisensinfo.",
|
||||||
"View privacy policy.": "Vis personvernspraksis.",
|
"View privacy policy.": "Vis personvernspraksis.",
|
||||||
"Trending": "Trendsettende",
|
"Trending": "Trendsettende",
|
||||||
|
"Public": "",
|
||||||
"Unlisted": "Ulistet",
|
"Unlisted": "Ulistet",
|
||||||
"Watch video on Youtube": "Vis video på YouTube",
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "Vis video på YouTube",
|
||||||
|
"Hide annotations": "Skjul merknader",
|
||||||
|
"Show annotations": "Vis merknader",
|
||||||
"Genre: ": "Sjanger: ",
|
"Genre: ": "Sjanger: ",
|
||||||
"License: ": "Lisens: ",
|
"License: ": "Lisens: ",
|
||||||
"Family friendly? ": "Familievennlig? ",
|
"Family friendly? ": "Familievennlig? ",
|
||||||
@@ -113,8 +149,10 @@
|
|||||||
"Whitelisted regions: ": "Hvitlistede regioner: ",
|
"Whitelisted regions: ": "Hvitlistede regioner: ",
|
||||||
"Blacklisted regions: ": "Svartelistede regioner: ",
|
"Blacklisted regions: ": "Svartelistede regioner: ",
|
||||||
"Shared `x`": "Delt `x`",
|
"Shared `x`": "Delt `x`",
|
||||||
|
"`x` views": "`x` visninger",
|
||||||
"Premieres in `x`": "Premiere om `x`",
|
"Premieres in `x`": "Premiere om `x`",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hei. Det ser ut til at du har JavaScript avslått. Klikk her for å vise kommentarer, ha i minnet at innlasting tar lengre tid.",
|
"Premieres `x`": "Première `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hei. Det ser ut til at du har JavaScript avslått. Klikk her for å vise kommentarer, ha i minnet at innlasting tar lengre tid.",
|
||||||
"View YouTube comments": "Vis YouTube-kommentarer",
|
"View YouTube comments": "Vis YouTube-kommentarer",
|
||||||
"View more comments on Reddit": "Vis flere kommenterer på Reddit",
|
"View more comments on Reddit": "Vis flere kommenterer på Reddit",
|
||||||
"View `x` comments": "Vis `x` kommentarer",
|
"View `x` comments": "Vis `x` kommentarer",
|
||||||
@@ -123,19 +161,19 @@
|
|||||||
"Show replies": "Vis svar",
|
"Show replies": "Vis svar",
|
||||||
"Incorrect password": "Feil passord",
|
"Incorrect password": "Feil passord",
|
||||||
"Quota exceeded, try again in a few hours": "Kvote overskredet, prøv igjen om et par timer",
|
"Quota exceeded, try again in a few hours": "Kvote overskredet, prøv igjen om et par timer",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Kunne ikke logge inn, forsikre deg om at tofaktor-identitetsbekreftelse (Authenticator eller SMS) er skrudd på.",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Kunne ikke logge inn, forsikre deg om at tofaktor-identitetsbekreftelse (Authenticator eller SMS) er skrudd på.",
|
||||||
"Invalid TFA code": "Ugyldig tofaktorkode",
|
"Invalid TFA code": "Ugyldig tofaktorkode",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Innlogging mislyktes. Dette kan være fordi tofaktor-identitetsbekreftelse er skrudd av på kontoen din.",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Innlogging mislyktes. Dette kan være fordi tofaktor-identitetsbekreftelse er skrudd av på kontoen din.",
|
||||||
"Invalid answer": "Ugyldig svar",
|
"Wrong answer": "Ugyldig svar",
|
||||||
"Invalid CAPTCHA": "Ugyldig CAPTCHA",
|
"Erroneous CAPTCHA": "Ugyldig CAPTCHA",
|
||||||
"CAPTCHA is a required field": "CAPTCHA er et påkrevd felt",
|
"CAPTCHA is a required field": "CAPTCHA er et påkrevd felt",
|
||||||
"User ID is a required field": "Bruker-ID er et påkrevd felt",
|
"User ID is a required field": "Bruker-ID er et påkrevd felt",
|
||||||
"Password is a required field": "Passord er et påkrevd felt",
|
"Password is a required field": "Passord er et påkrevd felt",
|
||||||
"Invalid username or password": "Ugyldig brukernavn eller passord",
|
"Wrong username or password": "Ugyldig brukernavn eller passord",
|
||||||
"Please sign in using 'Sign in with Google'": "Logg inn ved bruk av \"Google-innlogging\"",
|
"Please sign in using 'Log in with Google'": "Logg inn ved bruk av \"Google-innlogging\"",
|
||||||
"Password cannot be empty": "Passordet kan ikke være tomt",
|
"Password cannot be empty": "Passordet kan ikke være tomt",
|
||||||
"Password cannot be longer than 55 characters": "Passordet kan ikke være lengre enn 55 tegn",
|
"Password cannot be longer than 55 characters": "Passordet kan ikke være lengre enn 55 tegn",
|
||||||
"Please sign in": "Logg inn",
|
"Please log in": "Logg inn",
|
||||||
"Invidious Private Feed for `x`": "Ugyldig privat flyt for `x`",
|
"Invidious Private Feed for `x`": "Ugyldig privat flyt for `x`",
|
||||||
"channel:`x`": "kanal `x`",
|
"channel:`x`": "kanal `x`",
|
||||||
"Deleted or invalid channel": "Slettet eller ugyldig kanal",
|
"Deleted or invalid channel": "Slettet eller ugyldig kanal",
|
||||||
@@ -147,15 +185,15 @@
|
|||||||
"Load more": "Last inn flere",
|
"Load more": "Last inn flere",
|
||||||
"`x` points": "`x` poeng",
|
"`x` points": "`x` poeng",
|
||||||
"Could not create mix.": "Kunne ikke opprette miks.",
|
"Could not create mix.": "Kunne ikke opprette miks.",
|
||||||
"Playlist is empty": "Spillelisten er tom",
|
"Empty playlist": "Spillelisten er tom",
|
||||||
"Invalid playlist.": "Ugyldig spilleliste.",
|
"Not a playlist.": "Ugyldig spilleliste.",
|
||||||
"Playlist does not exist.": "Spillelisten finnes ikke.",
|
"Playlist does not exist.": "Spillelisten finnes ikke.",
|
||||||
"Could not pull trending pages.": "Kunne ikke hente trendsettende sider.",
|
"Could not pull trending pages.": "Kunne ikke hente trendsettende sider.",
|
||||||
"Hidden field \"challenge\" is a required field": "Skjult felt \"utfordring\" er et påkrevd felt",
|
"Hidden field \"challenge\" is a required field": "Skjult felt \"utfordring\" er et påkrevd felt",
|
||||||
"Hidden field \"token\" is a required field": "Skjult felt \"symbol\" er et påkrevd felt",
|
"Hidden field \"token\" is a required field": "Skjult felt \"symbol\" er et påkrevd felt",
|
||||||
"Invalid challenge": "Ugyldig utfordring",
|
"Erroneous challenge": "Ugyldig utfordring",
|
||||||
"Invalid token": "Ugyldig symbol",
|
"Erroneous token": "Ugyldig symbol",
|
||||||
"Invalid user": "Ugyldig bruker",
|
"No such user": "Ugyldig bruker",
|
||||||
"Token is expired, please try again": "Symbol utløpt, prøv igjen",
|
"Token is expired, please try again": "Symbol utløpt, prøv igjen",
|
||||||
"English": "Engelsk",
|
"English": "Engelsk",
|
||||||
"English (auto-generated)": "Engelsk (auto-generert)",
|
"English (auto-generated)": "Engelsk (auto-generert)",
|
||||||
@@ -224,7 +262,7 @@
|
|||||||
"Marathi": "",
|
"Marathi": "",
|
||||||
"Mongolian": "",
|
"Mongolian": "",
|
||||||
"Nepali": "",
|
"Nepali": "",
|
||||||
"Norwegian": "Norsk bokmål",
|
"Norwegian Bokmål": "Norsk bokmål",
|
||||||
"Nyanja": "",
|
"Nyanja": "",
|
||||||
"Pashto": "",
|
"Pashto": "",
|
||||||
"Persian": "",
|
"Persian": "",
|
||||||
@@ -276,6 +314,7 @@
|
|||||||
"About": "Om",
|
"About": "Om",
|
||||||
"Rating: ": "Vurdering: ",
|
"Rating: ": "Vurdering: ",
|
||||||
"Language: ": "Språk: ",
|
"Language: ": "Språk: ",
|
||||||
|
"View as playlist": "Vis som spilleliste",
|
||||||
"Default": "Forvalg",
|
"Default": "Forvalg",
|
||||||
"Music": "Musikk",
|
"Music": "Musikk",
|
||||||
"Gaming": "Spill",
|
"Gaming": "Spill",
|
||||||
@@ -285,11 +324,13 @@
|
|||||||
"Download as: ": "Last ned som: ",
|
"Download as: ": "Last ned som: ",
|
||||||
"%A %B %-d, %Y": "",
|
"%A %B %-d, %Y": "",
|
||||||
"(edited)": "(redigert)",
|
"(edited)": "(redigert)",
|
||||||
"Youtube permalink of the comment": "Permanent YouTube-lenke til innholdet",
|
"YouTube comment permalink": "Permanent YouTube-lenke til innholdet",
|
||||||
|
"permalink": "permanent lenke",
|
||||||
"`x` marked it with a ❤": "`x` levnet et ❤",
|
"`x` marked it with a ❤": "`x` levnet et ❤",
|
||||||
"Audio mode": "Lydmodus",
|
"Audio mode": "Lydmodus",
|
||||||
"Video mode": "Video-modus",
|
"Video mode": "Video-modus",
|
||||||
"Videos": "Videoer",
|
"Videos": "Videoer",
|
||||||
"Playlists": "Spillelister",
|
"Playlists": "Spillelister",
|
||||||
|
"Community": "Gemenskap",
|
||||||
"Current version: ": "Nåværende versjon: "
|
"Current version: ": "Nåværende versjon: "
|
||||||
}
|
}
|
||||||
629
locales/nl.json
629
locales/nl.json
@@ -1,295 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` abonnees",
|
"`x` subscribers": "`x` abonnees",
|
||||||
"`x` videos": "`x` videos",
|
"`x` videos": "`x` video's",
|
||||||
"LIVE": "LIVE",
|
"`x` playlists": "",
|
||||||
"Shared `x` ago": "Gedeeld `x` geleden",
|
"LIVE": "LIVE",
|
||||||
"Unsubscribe": "Abonnement opzeggen",
|
"Shared `x` ago": "Gedeeld: `x` geleden",
|
||||||
"Subscribe": "Abonneren",
|
"Unsubscribe": "Deabonneren",
|
||||||
"Login to subscribe to `x`": "Log in om te abonneren op `x`",
|
"Subscribe": "Abonneren",
|
||||||
"View channel on YouTube": "Bekijk kanaal op Youtube",
|
"View channel on YouTube": "Bekijk kanaal op YouTube",
|
||||||
"newest": "nieuwste",
|
"View playlist on YouTube": "Bekijk afspeellijst op YouTube",
|
||||||
"oldest": "oudste",
|
"newest": "nieuwste",
|
||||||
"popular": "populair",
|
"oldest": "oudste",
|
||||||
"last": "",
|
"popular": "populair",
|
||||||
"Next page": "Volgende pagina",
|
"last": "laatste",
|
||||||
"Previous page": "Vorige pagina",
|
"Next page": "Volgende pagina",
|
||||||
"Clear watch history?": "Kijk geschiedenis wissen?",
|
"Previous page": "Vorige pagina",
|
||||||
"Yes": "Ja",
|
"Clear watch history?": "Wil je de kijkgeschiedenis wissen?",
|
||||||
"No": "Nee",
|
"New password": "Nieuw wachtwoord",
|
||||||
"Import and Export Data": "Importeer en Exporteer Gegevens",
|
"New passwords must match": "De nieuwe wachtwoorden moeten overeenkomen",
|
||||||
"Import": "Importeren",
|
"Cannot change password for Google accounts": "Kan het wachtwoord van Google-accounts niet wijzigen",
|
||||||
"Import Invidious data": "Importeer Invidious gegevens",
|
"Authorize token?": "Wil je de toegangssleutel machtigen?",
|
||||||
"Import YouTube subscriptions": "Importeer Youtube abonnees",
|
"Authorize token for `x`?": "Wil je de toegangssleutel machtigen voor `x`?",
|
||||||
"Import FreeTube subscriptions (.db)": "Importeer FreeTube abonnees (.db)",
|
"Yes": "Ja",
|
||||||
"Import NewPipe subscriptions (.json)": "Importeer NewPipe abonnees (.json)",
|
"No": "Nee",
|
||||||
"Import NewPipe data (.zip)": "Importeer NewPipe gegevens (.zip)",
|
"Import and Export Data": "Gegevens im- en exporteren",
|
||||||
"Export": "Exporteren",
|
"Import": "Importeren",
|
||||||
"Export subscriptions as OPML": "Exporteer abonnees als OPML",
|
"Import Invidious data": "Invidious-gegevens importeren",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Exporteer abonnees als OPML (voor NewPipe & FreeTube)",
|
"Import YouTube subscriptions": "YouTube-abonnementen importeren",
|
||||||
"Export data as JSON": "Exporteer gegevens als JSON",
|
"Import FreeTube subscriptions (.db)": "FreeTube-abonnementen importeren (.db)",
|
||||||
"Delete account?": "Verwijder account?",
|
"Import NewPipe subscriptions (.json)": "NewPipe-abonnementen importeren (.json)",
|
||||||
"History": "Geschiedenis",
|
"Import NewPipe data (.zip)": "NewPipe-gegevens importeren (.zip)",
|
||||||
"An alternative front-end to YouTube": "Een alternatieve front-end voor YouTube",
|
"Export": "Exporteren",
|
||||||
"JavaScript license information": "JavaScript licentie informatie",
|
"Export subscriptions as OPML": "Abonnementen exporteren als OPML",
|
||||||
"source": "bron",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonnementen exporteren als OPML (voor NewPipe en FreeTube)",
|
||||||
"Login": "Inloggen",
|
"Export data as JSON": "Gegevens exporteren als JSON",
|
||||||
"Login/Register": "Inloggen/Registreren",
|
"Delete account?": "Wil je je account verwijderen?",
|
||||||
"Login to Google": "Inloggen op Google",
|
"History": "Geschiedenis",
|
||||||
"User ID:": "Gebruiker ID:",
|
"An alternative front-end to YouTube": "Een alternatief front-end voor YouTube",
|
||||||
"Password:": "Wachtwoord:",
|
"JavaScript license information": "JavaScript-licentieinformatie",
|
||||||
"Time (h:mm:ss):": "Tijd (h:mm:ss):",
|
"source": "bron",
|
||||||
"Text CAPTCHA": "Tekst CAPTCHA",
|
"Log in": "Inloggen",
|
||||||
"Image CAPTCHA": "Afbeelding CAPTCHA",
|
"Log in/register": "Inloggen/Registreren",
|
||||||
"Sign In": "Aanmelden",
|
"Log in with Google": "Inloggen met Google",
|
||||||
"Register": "Registreren",
|
"User ID": "Gebruikers-id",
|
||||||
"Email:": "Email:",
|
"Password": "Wachtwoord",
|
||||||
"Google verification code:": "Google verificatie code:",
|
"Time (h:mm:ss):": "Tijd (h:mm:ss):",
|
||||||
"Preferences": "Voorkeuren",
|
"Text CAPTCHA": "Tekst-CAPTCHA",
|
||||||
"Player preferences": "Afspeler voorkeuren",
|
"Image CAPTCHA": "Afbeelding-CAPTCHA",
|
||||||
"Always loop: ": "Altijd herhalen: ",
|
"Sign In": "Inloggen",
|
||||||
"Autoplay: ": "Automatisch afspelen: ",
|
"Register": "Registreren",
|
||||||
"Autoplay next video: ": "Automatisch volgende video afspelen: ",
|
"E-mail": "E-mailadres",
|
||||||
"Listen by default: ": "Standaard luisteren: ",
|
"Google verification code": "Google-verificatiecode",
|
||||||
"Proxy videos? ": "",
|
"Preferences": "Instellingen",
|
||||||
"Default speed: ": "Standaard snelheid: ",
|
"Player preferences": "Spelerinstellingen",
|
||||||
"Preferred video quality: ": "Video kwaliteit voorkeur: ",
|
"Always loop: ": "Altijd herhalen: ",
|
||||||
"Player volume: ": "Afspeler volume: ",
|
"Autoplay: ": "Automatisch afspelen: ",
|
||||||
"Default comments: ": "Standaard reacties: ",
|
"Play next by default: ": "Standaard volgende video afspelen: ",
|
||||||
"Default captions: ": "Standaard ondertitels: ",
|
"Autoplay next video: ": "Volgende video automatisch afspelen: ",
|
||||||
"Fallback captions: ": "Alternatieve ondertitels: ",
|
"Listen by default: ": "Standaard luisteren: ",
|
||||||
"Show related videos? ": "Laat gerelateerde videos zien? ",
|
"Proxy videos: ": "Video's afspelen via proxy? ",
|
||||||
"Visual preferences": "Visuele voorkeuren",
|
"Default speed: ": "Standaard afspeelsnelheid: ",
|
||||||
"Dark mode: ": "Donkere modus: ",
|
"Preferred video quality: ": "Voorkeurskwaliteit: ",
|
||||||
"Thin mode: ": "Smalle modus: ",
|
"Player volume: ": "Spelervolume: ",
|
||||||
"Subscription preferences": "Abonnement voorkeuren",
|
"Default comments: ": "Reacties tonen van: ",
|
||||||
"Redirect homepage to feed: ": "Startpagina omleiden naar feed: ",
|
"youtube": "YouTube",
|
||||||
"Number of videos shown in feed: ": "Aantal videos te zien in feed: ",
|
"reddit": "Reddit",
|
||||||
"Sort videos by: ": "Sorteer videos op: ",
|
"Default captions: ": "Standaard ondertiteling: ",
|
||||||
"published": "gepubliceerd",
|
"Fallback captions: ": "Alternatieve ondertiteling: ",
|
||||||
"published - reverse": "gepubliceerd - omgekeerd",
|
"Show related videos: ": "Gerelateerde video's tonen? ",
|
||||||
"alphabetically": "alfabetische volgorde",
|
"Show annotations by default: ": "Standaard annotaties tonen? ",
|
||||||
"alphabetically - reverse": "alfabetisch - omgekeerd",
|
"Visual preferences": "Visuele instellingen",
|
||||||
"channel name": "kanaal naam",
|
"Player style: ": "",
|
||||||
"channel name - reverse": "kanaal naam - omgekeerd",
|
"Dark mode: ": "Donkere modus: ",
|
||||||
"Only show latest video from channel: ": "Laat alleen laatste video van kanaal zien: ",
|
"Theme: ": "",
|
||||||
"Only show latest unwatched video from channel: ": "Laat alleen de laatste onbekeken video zien van kanaal: ",
|
"dark": "",
|
||||||
"Only show unwatched: ": "Laat alleen onbekeken videos zien: ",
|
"light": "",
|
||||||
"Only show notifications (if there are any): ": "Laat alleen notificaties zien (als die er zijn): ",
|
"Thin mode: ": "Smalle modus: ",
|
||||||
"Data preferences": "Gegevens voorkeuren",
|
"Subscription preferences": "Abonnementsinstellingen",
|
||||||
"Clear watch history": "Kijkgeschiedenis wissen",
|
"Show annotations by default for subscribed channels: ": "Standaard annotaties tonen voor geabonneerde kanalen? ",
|
||||||
"Import/Export data": "Importeer/Exporteer gegevens",
|
"Redirect homepage to feed: ": "Startpagina omleiden naar feed: ",
|
||||||
"Manage subscriptions": "Abonnees beheren",
|
"Number of videos shown in feed: ": "Aantal te tonen video's in feed: ",
|
||||||
"Watch history": "Kijkgeschiedenis",
|
"Sort videos by: ": "Video's sorteren op: ",
|
||||||
"Delete account": "Account verwijderen",
|
"published": "publicatiedatum",
|
||||||
"Administrator preferences": "",
|
"published - reverse": "publicatiedatum - omgekeerd",
|
||||||
"Default homepage: ": "",
|
"alphabetically": "alfabetische volgorde",
|
||||||
"Feed menu: ": "",
|
"alphabetically - reverse": "alfabetische volgorde - omgekeerd",
|
||||||
"Top enabled? ": "",
|
"channel name": "kanaalnaam",
|
||||||
"CAPTCHA enabled? ": "",
|
"channel name - reverse": "kanaalnaam - omgekeerd",
|
||||||
"Login enabled? ": "",
|
"Only show latest video from channel: ": "Alleen nieuwste video van kanaal tonen: ",
|
||||||
"Registration enabled? ": "",
|
"Only show latest unwatched video from channel: ": "Alleen nieuwste niet-bekeken video van kanaal tonen: ",
|
||||||
"Report statistics? ": "",
|
"Only show unwatched: ": "Alleen niet-bekeken videos tonen: ",
|
||||||
"Save preferences": "Opslaan voorkeuren",
|
"Only show notifications (if there are any): ": "Alleen meldingen tonen (als die er zijn): ",
|
||||||
"Subscription manager": "Abonnees beheerder",
|
"Enable web notifications": "Systemmeldingen inschakelen",
|
||||||
"`x` subscriptions": "`x` abonnees",
|
"`x` uploaded a video": "`x` heeft een video geüpload",
|
||||||
"Import/Export": "Importeer/Exporteer",
|
"`x` is live": "`x` zendt nu live uit",
|
||||||
"unsubscribe": "abonnement opzeggen",
|
"Data preferences": "Gegevensinstellingen",
|
||||||
"Subscriptions": "Abonnees",
|
"Clear watch history": "Kijkgeschiedenis wissen",
|
||||||
"`x` unseen notifications": "`x` onbekeken notificaties",
|
"Import/export data": "Gegevens im-/exporteren",
|
||||||
"search": "zoeken",
|
"Change password": "Wachtwoord wijzigen",
|
||||||
"Sign out": "Afmelden",
|
"Manage subscriptions": "Abonnementen beheren",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Uitgegeven onder AGPLv3 door Omar Roth.",
|
"Manage tokens": "Toegangssleutels beheren",
|
||||||
"Source available here.": "Bron beschikbaar hier.",
|
"Watch history": "Kijkgeschiedenis",
|
||||||
"View JavaScript license information.": "Bekijk JavaScript licentie informatie.",
|
"Delete account": "Account verwijderen",
|
||||||
"View privacy policy.": "",
|
"Administrator preferences": "Beheerdersinstellingen",
|
||||||
"Trending": "Trending",
|
"Default homepage: ": "Standaard startpagina: ",
|
||||||
"Unlisted": "",
|
"Feed menu: ": "Feedmenu:",
|
||||||
"Watch video on Youtube": "Bekijk video op Youtube",
|
"Top enabled: ": "Bovenkant inschakelen? ",
|
||||||
"Genre: ": "Genre: ",
|
"CAPTCHA enabled: ": "CAPTCHA gebruiken? ",
|
||||||
"License: ": "Licentie: ",
|
"Login enabled: ": "Inloggen toestaan? ",
|
||||||
"Family friendly? ": "Gezinsvriendelijk? ",
|
"Registration enabled: ": "Registratie toestaan? ",
|
||||||
"Wilson score: ": "Wilson score: ",
|
"Report statistics: ": "Statistieken bijhouden? ",
|
||||||
"Engagement: ": "Betrokkenheid: ",
|
"Save preferences": "Instellingen opslaan",
|
||||||
"Whitelisted regions: ": "Toegestane regio's: ",
|
"Subscription manager": "Abonnementen beheren",
|
||||||
"Blacklisted regions: ": "Geblokkeerde regio's: ",
|
"Token manager": "Toegangssleutels beheren",
|
||||||
"Shared `x`": "`x` gedeeld",
|
"Token": "Toegangssleutel",
|
||||||
"Premieres in `x`": "",
|
"`x` subscriptions": "`x` abonnementen",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Hoi! Het lijkt erop dat je JavaScript uit hebt staan. Klik hier om de reacties te bekijken, hou er rekening mee dat het wat langer duurt om te laden.",
|
"`x` tokens": "`x` toegangssleutels",
|
||||||
"View YouTube comments": "Bekijk YouTube reacties",
|
"Import/export": "Importeren/Exporteren",
|
||||||
"View more comments on Reddit": "Bekijk meer reacties op Reddit",
|
"unsubscribe": "Deabonneren",
|
||||||
"View `x` comments": "`x` reacties zien",
|
"revoke": "Intrekken",
|
||||||
"View Reddit comments": "Bekijk Reddit reacties",
|
"Subscriptions": "Abonnementen",
|
||||||
"Hide replies": "Verberg antwoorden",
|
"`x` unseen notifications": "`x` ongelezen meldingen",
|
||||||
"Show replies": "Laat antwoorden zien",
|
"search": "zoeken",
|
||||||
"Incorrect password": "Onjuist wachtwoord",
|
"Log out": "Uitloggen",
|
||||||
"Quota exceeded, try again in a few hours": "Quota overschreden, probeer het over een paar uur opnieuw",
|
"Released under the AGPLv3 by Omar Roth.": "Uitgebracht onder de AGPLv3-licentie, door Omar Roth.",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Niet in staat om in te loggen, zorg ervoor dat two-factor authentication (Authenticator of SMS) is ingeschakeld.",
|
"Source available here.": "De broncode is hier beschikbaar.",
|
||||||
"Invalid TFA code": "Onjuiste TFA code",
|
"View JavaScript license information.": "JavaScript-licentieinformatie tonen.",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Aanmelden mislukt. Dit kan zijn omdat two-factor authentication niet is ingeschakeld voor uw account.",
|
"View privacy policy.": "Privacybeleid tonen",
|
||||||
"Invalid answer": "Onjuist antwoord",
|
"Trending": "Uitgelicht",
|
||||||
"Invalid CAPTCHA": "Onjuiste CAPTCHA",
|
"Public": "",
|
||||||
"CAPTCHA is a required field": "CAPTCHA is een vereist veld",
|
"Unlisted": "Verborgen",
|
||||||
"User ID is a required field": "Gebruiker ID is een vereist veld",
|
"Private": "",
|
||||||
"Password is a required field": "Wachtwoord is een vereist veld",
|
"View all playlists": "",
|
||||||
"Invalid username or password": "Ongeldige gebruikersnaam of wachtwoord",
|
"Updated `x` ago": "",
|
||||||
"Please sign in using 'Sign in with Google'": "Meld u aan met 'Aanmelden met Google'",
|
"Delete playlist `x`?": "",
|
||||||
"Password cannot be empty": "Wachtwoord mag niet leeg zijn",
|
"Delete playlist": "",
|
||||||
"Password cannot be longer than 55 characters": "Wachtwoord mag niet langer dan 55 tekens zijn",
|
"Create playlist": "",
|
||||||
"Please sign in": "Meld u aan",
|
"Title": "",
|
||||||
"Invidious Private Feed for `x`": "Invidious Privé Feed voor `x`",
|
"Playlist privacy": "",
|
||||||
"channel:`x`": "kanaal:`x`",
|
"Editing playlist `x`": "",
|
||||||
"Deleted or invalid channel": "Verwijderd of ongeldig kanaal",
|
"Watch on YouTube": "Video bekijken op YouTube",
|
||||||
"This channel does not exist.": "Dit kanaal bestaat niet.",
|
"Hide annotations": "Annotaties verbergen",
|
||||||
"Could not get channel info.": "Kan kanaal informatie niet verkrijgen.",
|
"Show annotations": "Annotaties tonen",
|
||||||
"Could not fetch comments": "Kan reacties niet verkrijgen",
|
"Genre: ": "Genre: ",
|
||||||
"View `x` replies": "`x` antwoorden zien",
|
"License: ": "Licentie: ",
|
||||||
"`x` ago": "`x` geleden",
|
"Family friendly? ": "Gezinsvriendelijk? ",
|
||||||
"Load more": "Meer laden",
|
"Wilson score: ": "Wilson-score: ",
|
||||||
"`x` points": "`x` punten",
|
"Engagement: ": "Betrokkenheid: ",
|
||||||
"Could not create mix.": "Kon mix niet maken.",
|
"Whitelisted regions: ": "Toegestane regio's: ",
|
||||||
"Playlist is empty": "Afspeellijst is leeg",
|
"Blacklisted regions: ": "Geblokkeerde regio's: ",
|
||||||
"Invalid playlist.": "Ongeldige afspeellijst.",
|
"Shared `x`": "`x` gedeeld",
|
||||||
"Playlist does not exist.": "Afspeellijst bestaat niet.",
|
"`x` views": "`x` weergaven",
|
||||||
"Could not pull trending pages.": "Kon trending paginas niet verkrijgen.",
|
"Premieres in `x`": "Verschijnt over `x`",
|
||||||
"Hidden field \"challenge\" is a required field": "Verborgen veld \"uitdaging\" is een vereist veld",
|
"Premieres `x`": "",
|
||||||
"Hidden field \"token\" is a required field": "Verborgen veld \"token\" is een vereist veld",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hoi! Het lijkt erop dat je JavaScript hebt uitgeschakeld. Klik hier om de reacties te bekijken. Let op: het laden duurt wat langer.",
|
||||||
"Invalid challenge": "Ongeldige uitdaging",
|
"View YouTube comments": "YouTube-reacties tonen",
|
||||||
"Invalid token": "Ongeldige token",
|
"View more comments on Reddit": "Meer reacties bekijken op Reddit",
|
||||||
"Invalid user": "Ongeldige gebruiker",
|
"View `x` comments": "`x` reacties tonen",
|
||||||
"Token is expired, please try again": "Token is verlopen, probeer het opnieuw",
|
"View Reddit comments": "Reddit-reacties tonen",
|
||||||
"English": "",
|
"Hide replies": "Antwoorden verbergen",
|
||||||
"English (auto-generated)": "",
|
"Show replies": "Antwoorden tonen",
|
||||||
"Afrikaans": "",
|
"Incorrect password": "Wachtwoord is onjuist",
|
||||||
"Albanian": "",
|
"Quota exceeded, try again in a few hours": "Quota overschreden; probeer het over een paar uur opnieuw",
|
||||||
"Amharic": "",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Kan niet inloggen. Zorg ervoor dat authenticatie in twee stappen (Authenticator of sms) is ingeschakeld.",
|
||||||
"Arabic": "",
|
"Invalid TFA code": "Onjuiste TFA-code",
|
||||||
"Armenian": "",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Inloggen mislukt. Wellicht is authenticatie in twee stappen niet ingeschakeld op je account.",
|
||||||
"Azerbaijani": "",
|
"Wrong answer": "Onjuist antwoord",
|
||||||
"Bangla": "",
|
"Erroneous CAPTCHA": "Onjuiste CAPTCHA",
|
||||||
"Basque": "",
|
"CAPTCHA is a required field": "CAPTCHA is vereist",
|
||||||
"Belarusian": "",
|
"User ID is a required field": "Gebruikers-id is vereist",
|
||||||
"Bosnian": "",
|
"Password is a required field": "Wachtwoord is vereist",
|
||||||
"Bulgarian": "",
|
"Wrong username or password": "Onjuiste gebruikersnaam of wachtwoord",
|
||||||
"Burmese": "",
|
"Please sign in using 'Log in with Google'": "Log in via 'Inloggen met Google'",
|
||||||
"Catalan": "",
|
"Password cannot be empty": "Het wachtwoordveld mag niet leeg zijn",
|
||||||
"Cebuano": "",
|
"Password cannot be longer than 55 characters": "Het wachtwoord mag niet langer dan 55 tekens zijn",
|
||||||
"Chinese (Simplified)": "",
|
"Please log in": "Log in",
|
||||||
"Chinese (Traditional)": "",
|
"Invidious Private Feed for `x`": "Invidious-privéfeed van `x`",
|
||||||
"Corsican": "",
|
"channel:`x`": "kanaal:`x`",
|
||||||
"Croatian": "",
|
"Deleted or invalid channel": "Verwijderd of niet-bestaand kanaal",
|
||||||
"Czech": "",
|
"This channel does not exist.": "Dit kanaal bestaat niet.",
|
||||||
"Danish": "",
|
"Could not get channel info.": "Kan geen kanaalinformatie ophalen.",
|
||||||
"Dutch": "",
|
"Could not fetch comments": "Kan reacties niet ophalen",
|
||||||
"Esperanto": "",
|
"View `x` replies": "`x` antwoorden tonen",
|
||||||
"Estonian": "",
|
"`x` ago": "`x` geleden",
|
||||||
"Filipino": "",
|
"Load more": "Meer laden",
|
||||||
"Finnish": "",
|
"`x` points": "`x` punten",
|
||||||
"French": "",
|
"Could not create mix.": "Kan geen mix maken.",
|
||||||
"Galician": "",
|
"Empty playlist": "Lege afspeellijst",
|
||||||
"Georgian": "",
|
"Not a playlist.": "Ongeldige afspeellijst.",
|
||||||
"German": "",
|
"Playlist does not exist.": "Afspeellijst bestaat niet.",
|
||||||
"Greek": "",
|
"Could not pull trending pages.": "Kan uitgelichte pagina's niet ophalen.",
|
||||||
"Gujarati": "",
|
"Hidden field \"challenge\" is a required field": "Verborgen veld \"uitdaging\" is vereist",
|
||||||
"Haitian Creole": "",
|
"Hidden field \"token\" is a required field": "Verborgen veld \"toegangssleutel\" is vereist",
|
||||||
"Hausa": "",
|
"Erroneous challenge": "Ongeldige uitdaging",
|
||||||
"Hawaiian": "",
|
"Erroneous token": "Ongeldige toegangssleutel",
|
||||||
"Hebrew": "",
|
"No such user": "Gebruiker bestaat niet",
|
||||||
"Hindi": "",
|
"Token is expired, please try again": "Toegangssleutel verlopen; probeer het opnieuw",
|
||||||
"Hmong": "",
|
"English": "Engels",
|
||||||
"Hungarian": "",
|
"English (auto-generated)": "Engels (automatisch gegenereerd)",
|
||||||
"Icelandic": "",
|
"Afrikaans": "Afrikaans",
|
||||||
"Igbo": "",
|
"Albanian": "Albanees",
|
||||||
"Indonesian": "",
|
"Amharic": "Amhaars",
|
||||||
"Irish": "",
|
"Arabic": "Arabisch",
|
||||||
"Italian": "",
|
"Armenian": "Armeens",
|
||||||
"Japanese": "",
|
"Azerbaijani": "Azerbeidzjaans",
|
||||||
"Javanese": "",
|
"Bangla": "Bangla",
|
||||||
"Kannada": "",
|
"Basque": "Baskisch",
|
||||||
"Kazakh": "",
|
"Belarusian": "Wit-Rrussisch",
|
||||||
"Khmer": "",
|
"Bosnian": "Bosnisch",
|
||||||
"Korean": "",
|
"Bulgarian": "Bulgaars",
|
||||||
"Kurdish": "",
|
"Burmese": "Birmaans",
|
||||||
"Kyrgyz": "",
|
"Catalan": "Catalaans",
|
||||||
"Lao": "",
|
"Cebuano": "Cebuano",
|
||||||
"Latin": "",
|
"Chinese (Simplified)": "Chinees (Veereenvoudigd)",
|
||||||
"Latvian": "",
|
"Chinese (Traditional)": "Chinees (Traditioneel)",
|
||||||
"Lithuanian": "",
|
"Corsican": "Corsicaans",
|
||||||
"Luxembourgish": "",
|
"Croatian": "Kroatisch",
|
||||||
"Macedonian": "",
|
"Czech": "Tsjechisch",
|
||||||
"Malagasy": "",
|
"Danish": "Deens",
|
||||||
"Malay": "",
|
"Dutch": "Nederlands",
|
||||||
"Malayalam": "",
|
"Esperanto": "Esperanto",
|
||||||
"Maltese": "",
|
"Estonian": "Ests",
|
||||||
"Maori": "",
|
"Filipino": "Filipijns",
|
||||||
"Marathi": "",
|
"Finnish": "Fins",
|
||||||
"Mongolian": "",
|
"French": "Frans",
|
||||||
"Nepali": "",
|
"Galician": "Galicisch",
|
||||||
"Norwegian": "",
|
"Georgian": "Georgisch",
|
||||||
"Nyanja": "",
|
"German": "Duits",
|
||||||
"Pashto": "",
|
"Greek": "Grieks",
|
||||||
"Persian": "",
|
"Gujarati": "Gujarati",
|
||||||
"Polish": "",
|
"Haitian Creole": "Creools",
|
||||||
"Portuguese": "",
|
"Hausa": "Hausa",
|
||||||
"Punjabi": "",
|
"Hawaiian": "Hawaïaans",
|
||||||
"Romanian": "",
|
"Hebrew": "Heebreeuws",
|
||||||
"Russian": "",
|
"Hindi": "Hindi",
|
||||||
"Samoan": "",
|
"Hmong": "Hmong",
|
||||||
"Scottish Gaelic": "",
|
"Hungarian": "Hongaars",
|
||||||
"Serbian": "",
|
"Icelandic": "IJslands",
|
||||||
"Shona": "",
|
"Igbo": "Igbo",
|
||||||
"Sindhi": "",
|
"Indonesian": "Indonesisch",
|
||||||
"Sinhala": "",
|
"Irish": "Iers",
|
||||||
"Slovak": "",
|
"Italian": "Italiaans",
|
||||||
"Slovenian": "",
|
"Japanese": "Japans",
|
||||||
"Somali": "",
|
"Javanese": "Javaans",
|
||||||
"Southern Sotho": "",
|
"Kannada": "Kannada",
|
||||||
"Spanish": "",
|
"Kazakh": "Kazachs",
|
||||||
"Spanish (Latin America)": "",
|
"Khmer": "Khmer",
|
||||||
"Sundanese": "",
|
"Korean": "Koreaans",
|
||||||
"Swahili": "",
|
"Kurdish": "Koerdisch",
|
||||||
"Swedish": "",
|
"Kyrgyz": "Kirgizisch",
|
||||||
"Tajik": "",
|
"Lao": "Laotiaans",
|
||||||
"Tamil": "",
|
"Latin": "Latijns",
|
||||||
"Telugu": "",
|
"Latvian": "Lets",
|
||||||
"Thai": "",
|
"Lithuanian": "Litouws",
|
||||||
"Turkish": "",
|
"Luxembourgish": "Luxemburgs",
|
||||||
"Ukrainian": "",
|
"Macedonian": "Macedonisch",
|
||||||
"Urdu": "",
|
"Malagasy": "Malagassisch",
|
||||||
"Uzbek": "",
|
"Malay": "Maleisisch",
|
||||||
"Vietnamese": "",
|
"Malayalam": "Malayalam",
|
||||||
"Welsh": "",
|
"Maltese": "Maltees",
|
||||||
"Western Frisian": "",
|
"Maori": "Maorisch",
|
||||||
"Xhosa": "",
|
"Marathi": "Marathi",
|
||||||
"Yiddish": "",
|
"Mongolian": "Mongools",
|
||||||
"Yoruba": "",
|
"Nepali": "Nepalees",
|
||||||
"Zulu": "",
|
"Norwegian Bokmål": "Noors (Bokmål)",
|
||||||
"`x` years": "`x` jaar",
|
"Nyanja": "Nyanja",
|
||||||
"`x` months": "`x` maanden",
|
"Pashto": "Pashto",
|
||||||
"`x` weeks": "`x` weken",
|
"Persian": "Perzisch",
|
||||||
"`x` days": "`x` dagen",
|
"Polish": "Pools",
|
||||||
"`x` hours": "`x` uur",
|
"Portuguese": "Portugees",
|
||||||
"`x` minutes": "`x` minuten",
|
"Punjabi": "Punjabi",
|
||||||
"`x` seconds": "`x` seconden",
|
"Romanian": "Roemeens",
|
||||||
"Fallback comments: ": "",
|
"Russian": "Russisch",
|
||||||
"Popular": "",
|
"Samoan": "Samoaans",
|
||||||
"Top": "",
|
"Scottish Gaelic": "Schots-Gaelisch",
|
||||||
"About": "",
|
"Serbian": "Servisch",
|
||||||
"Rating: ": "",
|
"Shona": "Shona",
|
||||||
"Language: ": "",
|
"Sindhi": "Sindhi",
|
||||||
"Default": "",
|
"Sinhala": "Sinhala",
|
||||||
"Music": "",
|
"Slovak": "Slowaaks",
|
||||||
"Gaming": "",
|
"Slovenian": "Sloveens",
|
||||||
"News": "",
|
"Somali": "Somalisch",
|
||||||
"Movies": "",
|
"Southern Sotho": "Zuid-Sotho",
|
||||||
"Download": "",
|
"Spanish": "Spaans",
|
||||||
"Download as: ": "",
|
"Spanish (Latin America)": "Spaans (Latijns-Amerika)",
|
||||||
"%A %B %-d, %Y": "",
|
"Sundanese": "Soedanees",
|
||||||
"(edited)": "",
|
"Swahili": "Swahili",
|
||||||
"Youtube permalink of the comment": "",
|
"Swedish": "Zweeds",
|
||||||
"`x` marked it with a ❤": "",
|
"Tajik": "Tajik",
|
||||||
"Audio mode": "",
|
"Tamil": "Tamil",
|
||||||
"Video mode": "",
|
"Telugu": "Telugu",
|
||||||
"Videos": "",
|
"Thai": "Thaïs",
|
||||||
"Playlists": "",
|
"Turkish": "Turks",
|
||||||
"Current version: ": ""
|
"Ukrainian": "Oekraïens",
|
||||||
}
|
"Urdu": "Urdu",
|
||||||
|
"Uzbek": "Oezbeeks",
|
||||||
|
"Vietnamese": "Vietnamees",
|
||||||
|
"Welsh": "Welsh",
|
||||||
|
"Western Frisian": "Fries",
|
||||||
|
"Xhosa": "Xhosa",
|
||||||
|
"Yiddish": "Joods",
|
||||||
|
"Yoruba": "Yoruba",
|
||||||
|
"Zulu": "Zulu",
|
||||||
|
"`x` years": "`x` jaar",
|
||||||
|
"`x` months": "`x` maanden",
|
||||||
|
"`x` weeks": "`x` weken",
|
||||||
|
"`x` days": "`x` dagen",
|
||||||
|
"`x` hours": "`x` uur",
|
||||||
|
"`x` minutes": "`x` minuten",
|
||||||
|
"`x` seconds": "`x` seconden",
|
||||||
|
"Fallback comments: ": "Terugvallen op",
|
||||||
|
"Popular": "Populair",
|
||||||
|
"Top": "Top",
|
||||||
|
"About": "Over",
|
||||||
|
"Rating: ": "Waardering",
|
||||||
|
"Language: ": "Taal",
|
||||||
|
"View as playlist": "Tonen als afspeellijst",
|
||||||
|
"Default": "Standaard",
|
||||||
|
"Music": "Muziek",
|
||||||
|
"Gaming": "Gaming",
|
||||||
|
"News": "Nieuws",
|
||||||
|
"Movies": "Films",
|
||||||
|
"Download": "Downloaden",
|
||||||
|
"Download as: ": "Downloaden als: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(bewerkt)",
|
||||||
|
"YouTube comment permalink": "Link naar YouTube-reactie",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "`x` heeft dit gemarkeerd met ❤",
|
||||||
|
"Audio mode": "Audiomodus",
|
||||||
|
"Video mode": "Videomodus",
|
||||||
|
"Videos": "Video's",
|
||||||
|
"Playlists": "Afspeellijsten",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": "Huidige versie: "
|
||||||
|
}
|
||||||
629
locales/pl.json
629
locales/pl.json
@@ -1,295 +1,336 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` subskrybcji",
|
"`x` subscribers": "`x` subskrybcji",
|
||||||
"`x` videos": "`x` filmów",
|
"`x` videos": "`x` filmów",
|
||||||
"LIVE": "NA ŻYWO",
|
"`x` playlists": "",
|
||||||
"Shared `x` ago": "Udostępniono `x` temu",
|
"LIVE": "NA ŻYWO",
|
||||||
"Unsubscribe": "Odsubskrybuj",
|
"Shared `x` ago": "Udostępniono `x` temu",
|
||||||
"Subscribe": "Subskrybuj",
|
"Unsubscribe": "Odsubskrybuj",
|
||||||
"Login to subscribe to `x`": "Zaloguj się, aby subskrybować `x`",
|
"Subscribe": "Subskrybuj",
|
||||||
"View channel on YouTube": "Wyświetl kanał na YouTube",
|
"View channel on YouTube": "Wyświetl kanał na YouTube",
|
||||||
"newest": "najnowsze",
|
"View playlist on YouTube": "",
|
||||||
"oldest": "najstarsze",
|
"newest": "najnowsze",
|
||||||
"popular": "popularne",
|
"oldest": "najstarsze",
|
||||||
"last": "ostatnie",
|
"popular": "popularne",
|
||||||
"Next page": "Następna strona",
|
"last": "ostatnie",
|
||||||
"Previous page": "Poprzednia strona",
|
"Next page": "Następna strona",
|
||||||
"Clear watch history?": "Wyczyścić historię?",
|
"Previous page": "Poprzednia strona",
|
||||||
"Yes": "Tak",
|
"Clear watch history?": "Wyczyścić historię?",
|
||||||
"No": "Nie",
|
"New password": "",
|
||||||
"Import and Export Data": "Import i eksport danych",
|
"New passwords must match": "",
|
||||||
"Import": "Import",
|
"Cannot change password for Google accounts": "",
|
||||||
"Import Invidious data": "Importuj dane Invidious",
|
"Authorize token?": "",
|
||||||
"Import YouTube subscriptions": "Importuj subskrybcje z YouTube",
|
"Authorize token for `x`?": "",
|
||||||
"Import FreeTube subscriptions (.db)": "Importuj subskrybcje z FreeTube (.db)",
|
"Yes": "Tak",
|
||||||
"Import NewPipe subscriptions (.json)": "Importuj subskrybcje z NewPipe (.json)",
|
"No": "Nie",
|
||||||
"Import NewPipe data (.zip)": "Importuj dane NewPipe (.zip)",
|
"Import and Export Data": "Import i eksport danych",
|
||||||
"Export": "Eksport",
|
"Import": "Import",
|
||||||
"Export subscriptions as OPML": "Eksportuj subskrybcje jako OPML",
|
"Import Invidious data": "Importuj dane Invidious",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksportuj subskrybcje jako OPML (dla NewPipe i FreeTube)",
|
"Import YouTube subscriptions": "Importuj subskrybcje z YouTube",
|
||||||
"Export data as JSON": "Eksportuj dane jako JSON",
|
"Import FreeTube subscriptions (.db)": "Importuj subskrybcje z FreeTube (.db)",
|
||||||
"Delete account?": "Usunąć konto?",
|
"Import NewPipe subscriptions (.json)": "Importuj subskrybcje z NewPipe (.json)",
|
||||||
"History": "Historia",
|
"Import NewPipe data (.zip)": "Importuj dane NewPipe (.zip)",
|
||||||
"An alternative front-end to YouTube": "Alternatywny front-end dla YouTube",
|
"Export": "Eksport",
|
||||||
"JavaScript license information": "Informacja o licencji JavaScript",
|
"Export subscriptions as OPML": "Eksportuj subskrybcje jako OPML",
|
||||||
"source": "źródło",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksportuj subskrybcje jako OPML (dla NewPipe i FreeTube)",
|
||||||
"Login": "Zaloguj",
|
"Export data as JSON": "Eksportuj dane jako JSON",
|
||||||
"Login/Register": "Zaloguj/Zarejestruj",
|
"Delete account?": "Usunąć konto?",
|
||||||
"Login to Google": "Zaloguj do Google",
|
"History": "Historia",
|
||||||
"User ID:": "ID użytkownika:",
|
"An alternative front-end to YouTube": "Alternatywny front-end dla YouTube",
|
||||||
"Password:": "Hasło:",
|
"JavaScript license information": "Informacja o licencji JavaScript",
|
||||||
"Time (h:mm:ss):": "Godzina (h:mm:ss):",
|
"source": "źródło",
|
||||||
"Text CAPTCHA": "Tekst CAPTCHA",
|
"Log in": "Zaloguj",
|
||||||
"Image CAPTCHA": "Obraz CAPTCHA",
|
"Log in/register": "Zaloguj/Zarejestruj",
|
||||||
"Sign In": "Zaloguj się",
|
"Log in with Google": "Zaloguj do Google",
|
||||||
"Register": "Zarejestruj się",
|
"User ID": "ID użytkownika",
|
||||||
"Email:": "Email:",
|
"Password": "Hasło",
|
||||||
"Google verification code:": "Kod weryfikacyjny Google:",
|
"Time (h:mm:ss):": "Godzina (h:mm:ss):",
|
||||||
"Preferences": "Preferencje",
|
"Text CAPTCHA": "Tekst CAPTCHA",
|
||||||
"Player preferences": "Ustawienia odtwarzacza",
|
"Image CAPTCHA": "Obraz CAPTCHA",
|
||||||
"Always loop: ": "Zawsze zapętlaj: ",
|
"Sign In": "Zaloguj się",
|
||||||
"Autoplay: ": "Autoodtwarzanie: ",
|
"Register": "Zarejestruj się",
|
||||||
"Autoplay next video: ": "Odtwórz następny film: ",
|
"E-mail": "Email",
|
||||||
"Listen by default: ": "Tryb dźwiękowy: ",
|
"Google verification code": "Kod weryfikacyjny Google",
|
||||||
"Proxy videos? ": "Filmy przez proxy? ",
|
"Preferences": "Preferencje",
|
||||||
"Default speed: ": "Domyślna prędkość: ",
|
"Player preferences": "Ustawienia odtwarzacza",
|
||||||
"Preferred video quality: ": "Preferowana jakość filmów: ",
|
"Always loop: ": "Zawsze zapętlaj: ",
|
||||||
"Player volume: ": "Głośność odtwarzacza: ",
|
"Autoplay: ": "Autoodtwarzanie: ",
|
||||||
"Default comments: ": "Domyślne komentarze: ",
|
"Play next by default: ": "",
|
||||||
"Default captions: ": "Domyślne napisy: ",
|
"Autoplay next video: ": "Odtwórz następny film: ",
|
||||||
"Fallback captions: ": "Zastępcze napisy: ",
|
"Listen by default: ": "Tryb dźwiękowy: ",
|
||||||
"Show related videos? ": "Pokaż powiązane filmy? ",
|
"Proxy videos: ": "Filmy przez proxy? ",
|
||||||
"Visual preferences": "Preferencje Wizualne",
|
"Default speed: ": "Domyślna prędkość: ",
|
||||||
"Dark mode: ": "Ciemny motyw: ",
|
"Preferred video quality: ": "Preferowana jakość filmów: ",
|
||||||
"Thin mode: ": "Tryb minimalny: ",
|
"Player volume: ": "Głośność odtwarzacza: ",
|
||||||
"Subscription preferences": "Preferencje subskrybcji",
|
"Default comments: ": "Domyślne komentarze: ",
|
||||||
"Redirect homepage to feed: ": "Przekieruj stronę główną do subskrybcji: ",
|
"youtube": "",
|
||||||
"Number of videos shown in feed: ": "Liczba filmów widoczna na stronie subskrybcji: ",
|
"reddit": "",
|
||||||
"Sort videos by: ": "Sortuj filmy: ",
|
"Default captions: ": "Domyślne napisy: ",
|
||||||
"published": "po czasie publikacji",
|
"Fallback captions: ": "Zastępcze napisy: ",
|
||||||
"published - reverse": "po czasie publikacji od najstarszych",
|
"Show related videos: ": "Pokaż powiązane filmy? ",
|
||||||
"alphabetically": "alfabetycznie",
|
"Show annotations by default: ": "",
|
||||||
"alphabetically - reverse": "alfabetycznie od tyłu",
|
"Visual preferences": "Preferencje Wizualne",
|
||||||
"channel name": "po nazwie kanału",
|
"Player style: ": "",
|
||||||
"channel name - reverse": "po nazwie kanału od tyłu",
|
"Dark mode: ": "Ciemny motyw: ",
|
||||||
"Only show latest video from channel: ": "Pokazuj tylko najnowszy film z kanału: ",
|
"Theme: ": "",
|
||||||
"Only show latest unwatched video from channel: ": "Pokazuj tylko najnowszy nie obejrzany film z kanału: ",
|
"dark": "",
|
||||||
"Only show unwatched: ": "Pokazuj tylko nie obejrzane: ",
|
"light": "",
|
||||||
"Only show notifications (if there are any): ": "Pokazuj tylko powiadomienia (jeśli są): ",
|
"Thin mode: ": "Tryb minimalny: ",
|
||||||
"Data preferences": "Preferencje danych",
|
"Subscription preferences": "Preferencje subskrybcji",
|
||||||
"Clear watch history": "Wyczyść historię",
|
"Show annotations by default for subscribed channels: ": "",
|
||||||
"Import/Export data": "Import/Eksport danych",
|
"Redirect homepage to feed: ": "Przekieruj stronę główną do subskrybcji: ",
|
||||||
"Manage subscriptions": "Organizuj subskrybcje",
|
"Number of videos shown in feed: ": "Liczba filmów widoczna na stronie subskrybcji: ",
|
||||||
"Watch history": "Historia",
|
"Sort videos by: ": "Sortuj filmy: ",
|
||||||
"Delete account": "Usuń konto",
|
"published": "po czasie publikacji",
|
||||||
"Administrator preferences": "Preferencje administratora",
|
"published - reverse": "po czasie publikacji od najstarszych",
|
||||||
"Default homepage: ": "Domyślna strona główna: ",
|
"alphabetically": "alfabetycznie",
|
||||||
"Feed menu: ": "",
|
"alphabetically - reverse": "alfabetycznie od tyłu",
|
||||||
"Top enabled? ": "",
|
"channel name": "po nazwie kanału",
|
||||||
"CAPTCHA enabled? ": "CAPTCHA aktywna? ",
|
"channel name - reverse": "po nazwie kanału od tyłu",
|
||||||
"Login enabled? ": "Logowanie włączone? ",
|
"Only show latest video from channel: ": "Pokazuj tylko najnowszy film z kanału: ",
|
||||||
"Registration enabled? ": "Rejestracja włączona? ",
|
"Only show latest unwatched video from channel: ": "Pokazuj tylko najnowszy nie obejrzany film z kanału: ",
|
||||||
"Report statistics? ": "Raportować statystyki? ",
|
"Only show unwatched: ": "Pokazuj tylko nie obejrzane: ",
|
||||||
"Save preferences": "Zapisz preferencje",
|
"Only show notifications (if there are any): ": "Pokazuj tylko powiadomienia (jeśli są): ",
|
||||||
"Subscription manager": "Manager subskrybcji",
|
"Enable web notifications": "",
|
||||||
"`x` subscriptions": "`x` subskrybcji",
|
"`x` uploaded a video": "",
|
||||||
"Import/Export": "Import/Eksport",
|
"`x` is live": "",
|
||||||
"unsubscribe": "odsubskrybuj",
|
"Data preferences": "Preferencje danych",
|
||||||
"Subscriptions": "Subskrybcje",
|
"Clear watch history": "Wyczyść historię",
|
||||||
"`x` unseen notifications": "`x` niewidzianych powiadomień",
|
"Import/export data": "Import/Eksport danych",
|
||||||
"search": "szukaj",
|
"Change password": "",
|
||||||
"Sign out": "Wyloguj",
|
"Manage subscriptions": "Organizuj subskrybcje",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Wydano na licencji AGPLv3 przez Omar Roth.",
|
"Manage tokens": "",
|
||||||
"Source available here.": "Kod źródłowy dostępny tutaj.",
|
"Watch history": "Historia",
|
||||||
"View JavaScript license information.": "Wyświetl informację o licencji JavaScript.",
|
"Delete account": "Usuń konto",
|
||||||
"View privacy policy.": "Polityka prywatności.",
|
"Administrator preferences": "Preferencje administratora",
|
||||||
"Trending": "Na czasie",
|
"Default homepage: ": "Domyślna strona główna: ",
|
||||||
"Unlisted": "",
|
"Feed menu: ": "",
|
||||||
"Watch video on Youtube": "Zobacz film na YouTube",
|
"Top enabled: ": "",
|
||||||
"Genre: ": "Gatunek: ",
|
"CAPTCHA enabled: ": "CAPTCHA aktywna? ",
|
||||||
"License: ": "Licencja: ",
|
"Login enabled: ": "Logowanie włączone? ",
|
||||||
"Family friendly? ": "Przyjazny rodzinie? ",
|
"Registration enabled: ": "Rejestracja włączona? ",
|
||||||
"Wilson score: ": "Punktacja Wilsona: ",
|
"Report statistics: ": "Raportować statystyki? ",
|
||||||
"Engagement: ": "Zaangażowanie: ",
|
"Save preferences": "Zapisz preferencje",
|
||||||
"Whitelisted regions: ": "Dostępny na obszarach: ",
|
"Subscription manager": "Manager subskrybcji",
|
||||||
"Blacklisted regions: ": "Niedostępny na obszarach: ",
|
"Token manager": "",
|
||||||
"Shared `x`": "Udostępniono `x`",
|
"Token": "",
|
||||||
"Premieres in `x`": "",
|
"`x` subscriptions": "`x` subskrybcji",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Cześć! Wygląda na to, że masz wyłączoną obsługę JavaScriptu. Kliknij tutaj, żeby zobaczyć komentarze. Pamiętaj, że wczytywanie może potrwać dłużej.",
|
"`x` tokens": "",
|
||||||
"View YouTube comments": "Wyświetl komentarze z YouTube",
|
"Import/export": "Import/Eksport",
|
||||||
"View more comments on Reddit": "Wyświetl więcej komentarzy na Reddicie",
|
"unsubscribe": "odsubskrybuj",
|
||||||
"View `x` comments": "Wyświetl `x` komentarzy",
|
"revoke": "",
|
||||||
"View Reddit comments": "Wyświetl komentarze z Redditta",
|
"Subscriptions": "Subskrybcje",
|
||||||
"Hide replies": "Ukryj odpowiedzi",
|
"`x` unseen notifications": "`x` nowych powiadomień",
|
||||||
"Show replies": "Pokaż odpowiedzi",
|
"search": "szukaj",
|
||||||
"Incorrect password": "Niepoprawne hasło",
|
"Log out": "Wyloguj",
|
||||||
"Quota exceeded, try again in a few hours": "Przekroczony limit zapytań, spróbuj ponownie za kilka godzin",
|
"Released under the AGPLv3 by Omar Roth.": "Wydano na licencji AGPLv3 przez Omar Roth.",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Nie udało się zalogować, upewnij się, że dwuetapowe uwierzytelnianie (Autentykator lub SMS) jest aktywne.",
|
"Source available here.": "Kod źródłowy dostępny tutaj.",
|
||||||
"Invalid TFA code": "Niepoprawny kod TFA",
|
"View JavaScript license information.": "Wyświetl informację o licencji JavaScript.",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Nie udało się zalogować. To może być spowodowane wyłączoną dwustopniową autoryzacją na twoim koncie.",
|
"View privacy policy.": "Polityka prywatności.",
|
||||||
"Invalid answer": "Niepoprawna odpowiedź",
|
"Trending": "Na czasie",
|
||||||
"Invalid CAPTCHA": "CAPTCHA wykonane błędnie",
|
"Public": "",
|
||||||
"CAPTCHA is a required field": "CAPTCHA jest polem wymaganym",
|
"Unlisted": "",
|
||||||
"User ID is a required field": "ID użytkownika jest polem wymaganym",
|
"Private": "",
|
||||||
"Password is a required field": "Hasło jest polem wymaganym",
|
"View all playlists": "",
|
||||||
"Invalid username or password": "Niepoprawny login lub hasło",
|
"Updated `x` ago": "",
|
||||||
"Please sign in using 'Sign in with Google'": "Zaloguj się używając \"Zaloguj się przez Google\"",
|
"Delete playlist `x`?": "",
|
||||||
"Password cannot be empty": "Hasło nie może być puste",
|
"Delete playlist": "",
|
||||||
"Password cannot be longer than 55 characters": "Hasło nie może być dłuższe niż 55 znaków",
|
"Create playlist": "",
|
||||||
"Please sign in": "Proszę się zalogować",
|
"Title": "",
|
||||||
"Invidious Private Feed for `x`": "",
|
"Playlist privacy": "",
|
||||||
"channel:`x`": "kanał:`x",
|
"Editing playlist `x`": "",
|
||||||
"Deleted or invalid channel": "Usunięty lub niepoprawny kanał",
|
"Watch on YouTube": "Zobacz film na YouTube",
|
||||||
"This channel does not exist.": "Ten kanał nie istnieje.",
|
"Hide annotations": "",
|
||||||
"Could not get channel info.": "Nie udało się uzyskać informacji o kanale.",
|
"Show annotations": "",
|
||||||
"Could not fetch comments": "Nie udało się pobrać komentarzy",
|
"Genre: ": "Gatunek: ",
|
||||||
"View `x` replies": "Wyświetl `x` odpowiedzi",
|
"License: ": "Licencja: ",
|
||||||
"`x` ago": "`x` temu",
|
"Family friendly? ": "Przyjazny rodzinie? ",
|
||||||
"Load more": "Wczytaj więcej",
|
"Wilson score: ": "Punktacja Wilsona: ",
|
||||||
"`x` points": "`x` punktów",
|
"Engagement: ": "Zaangażowanie: ",
|
||||||
"Could not create mix.": "Nie udało się utworzyć miksu.",
|
"Whitelisted regions: ": "Dostępny na obszarach: ",
|
||||||
"Playlist is empty": "Lista odtwarzania jest pusta",
|
"Blacklisted regions: ": "Niedostępny na obszarach: ",
|
||||||
"Invalid playlist.": "Niepoprawna lista.",
|
"Shared `x`": "Udostępniono `x`",
|
||||||
"Playlist does not exist.": "Lista odtwarzania nie istnieje.",
|
"`x` views": "`x` wyświetleń",
|
||||||
"Could not pull trending pages.": "Nie udało się pobrać strony na czasie.",
|
"Premieres in `x`": "Publikacja za `x`",
|
||||||
"Hidden field \"challenge\" is a required field": "Ukryte pole \"wyzwanie\" jest polem wymaganym",
|
"Premieres `x`": "",
|
||||||
"Hidden field \"token\" is a required field": "Ukryte pole \"token\" jest polem wymaganym",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Cześć! Wygląda na to, że masz wyłączoną obsługę JavaScriptu. Kliknij tutaj, żeby zobaczyć komentarze. Pamiętaj, że wczytywanie może potrwać dłużej.",
|
||||||
"Invalid challenge": "Niepoprawne wyzwanie",
|
"View YouTube comments": "Wyświetl komentarze z YouTube",
|
||||||
"Invalid token": "Niepoprawny token",
|
"View more comments on Reddit": "Wyświetl więcej komentarzy na Reddicie",
|
||||||
"Invalid user": "Niepoprawny użytkownik",
|
"View `x` comments": "Wyświetl `x` komentarzy",
|
||||||
"Token is expired, please try again": "Token wygasł, spróbuj ponownie",
|
"View Reddit comments": "Wyświetl komentarze z Redditta",
|
||||||
"English": "angielski",
|
"Hide replies": "Ukryj odpowiedzi",
|
||||||
"English (auto-generated)": "angielski (automatycznie generowane)",
|
"Show replies": "Pokaż odpowiedzi",
|
||||||
"Afrikaans": "afrykanerski",
|
"Incorrect password": "Niepoprawne hasło",
|
||||||
"Albanian": "albański",
|
"Quota exceeded, try again in a few hours": "Przekroczony limit zapytań, spróbuj ponownie za kilka godzin",
|
||||||
"Amharic": "amharski",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Nie udało się zalogować, upewnij się, że dwuetapowe uwierzytelnianie (Autentykator lub SMS) jest aktywne.",
|
||||||
"Arabic": "arabski",
|
"Invalid TFA code": "Niepoprawny kod TFA",
|
||||||
"Armenian": "armeński",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Nie udało się zalogować. To może być spowodowane wyłączoną dwustopniową autoryzacją na twoim koncie.",
|
||||||
"Azerbaijani": "azerski",
|
"Wrong answer": "Niepoprawna odpowiedź",
|
||||||
"Bangla": "bengalski",
|
"Erroneous CAPTCHA": "CAPTCHA wykonane błędnie",
|
||||||
"Basque": "baskijski",
|
"CAPTCHA is a required field": "CAPTCHA jest polem wymaganym",
|
||||||
"Belarusian": "białoruski",
|
"User ID is a required field": "ID użytkownika jest polem wymaganym",
|
||||||
"Bosnian": "bośniacki",
|
"Password is a required field": "Hasło jest polem wymaganym",
|
||||||
"Bulgarian": "bułgarski",
|
"Wrong username or password": "Niepoprawny login lub hasło",
|
||||||
"Burmese": "birmański",
|
"Please sign in using 'Log in with Google'": "Zaloguj się używając \"Zaloguj się przez Google\"",
|
||||||
"Catalan": "kataloński",
|
"Password cannot be empty": "Hasło nie może być puste",
|
||||||
"Cebuano": "cebuański",
|
"Password cannot be longer than 55 characters": "Hasło nie może być dłuższe niż 55 znaków",
|
||||||
"Chinese (Simplified)": "chiński (uproszczony)",
|
"Please log in": "Proszę się zalogować",
|
||||||
"Chinese (Traditional)": "chiński (tradycyjny)",
|
"Invidious Private Feed for `x`": "",
|
||||||
"Corsican": "korsykański",
|
"channel:`x`": "kanał:`x",
|
||||||
"Croatian": "chorwacki",
|
"Deleted or invalid channel": "Usunięty lub niepoprawny kanał",
|
||||||
"Czech": "czeski",
|
"This channel does not exist.": "Ten kanał nie istnieje.",
|
||||||
"Danish": "duński",
|
"Could not get channel info.": "Nie udało się uzyskać informacji o kanale.",
|
||||||
"Dutch": "holenderski",
|
"Could not fetch comments": "Nie udało się pobrać komentarzy",
|
||||||
"Esperanto": "esperanto",
|
"View `x` replies": "Wyświetl `x` odpowiedzi",
|
||||||
"Estonian": "estoński",
|
"`x` ago": "`x` temu",
|
||||||
"Filipino": "filipiński",
|
"Load more": "Wczytaj więcej",
|
||||||
"Finnish": "fiński",
|
"`x` points": "`x` punktów",
|
||||||
"French": "francuski",
|
"Could not create mix.": "Nie udało się utworzyć miksu.",
|
||||||
"Galician": "galicyjski",
|
"Empty playlist": "Lista odtwarzania jest pusta",
|
||||||
"Georgian": "gruziński",
|
"Not a playlist.": "Niepoprawna lista.",
|
||||||
"German": "niemiecki",
|
"Playlist does not exist.": "Lista odtwarzania nie istnieje.",
|
||||||
"Greek": "grecki",
|
"Could not pull trending pages.": "Nie udało się pobrać strony na czasie.",
|
||||||
"Gujarati": "gudźarati",
|
"Hidden field \"challenge\" is a required field": "Ukryte pole \"wyzwanie\" jest polem wymaganym",
|
||||||
"Haitian Creole": "kreolski haitański",
|
"Hidden field \"token\" is a required field": "Ukryte pole \"token\" jest polem wymaganym",
|
||||||
"Hausa": "hausa",
|
"Erroneous challenge": "Niepoprawne wyzwanie",
|
||||||
"Hawaiian": "hawajski",
|
"Erroneous token": "Niepoprawny token",
|
||||||
"Hebrew": "hebrajski",
|
"No such user": "Niepoprawny użytkownik",
|
||||||
"Hindi": "hindi",
|
"Token is expired, please try again": "Token wygasł, spróbuj ponownie",
|
||||||
"Hmong": "hmong",
|
"English": "angielski",
|
||||||
"Hungarian": "węgierski",
|
"English (auto-generated)": "angielski (automatycznie generowane)",
|
||||||
"Icelandic": "islandzki",
|
"Afrikaans": "afrykanerski",
|
||||||
"Igbo": "ibo",
|
"Albanian": "albański",
|
||||||
"Indonesian": "indonezyjski",
|
"Amharic": "amharski",
|
||||||
"Irish": "irlandzki",
|
"Arabic": "arabski",
|
||||||
"Italian": "włoski",
|
"Armenian": "armeński",
|
||||||
"Japanese": "japoński",
|
"Azerbaijani": "azerski",
|
||||||
"Javanese": "jawajski",
|
"Bangla": "bengalski",
|
||||||
"Kannada": "kannada",
|
"Basque": "baskijski",
|
||||||
"Kazakh": "kazachski",
|
"Belarusian": "białoruski",
|
||||||
"Khmer": "khmerski",
|
"Bosnian": "bośniacki",
|
||||||
"Korean": "koreański",
|
"Bulgarian": "bułgarski",
|
||||||
"Kurdish": "kurdyjski",
|
"Burmese": "birmański",
|
||||||
"Kyrgyz": "kirgiski",
|
"Catalan": "kataloński",
|
||||||
"Lao": "laotański",
|
"Cebuano": "cebuański",
|
||||||
"Latin": "łaciński",
|
"Chinese (Simplified)": "chiński (uproszczony)",
|
||||||
"Latvian": "łotewski",
|
"Chinese (Traditional)": "chiński (tradycyjny)",
|
||||||
"Lithuanian": "litewski",
|
"Corsican": "korsykański",
|
||||||
"Luxembourgish": "luksemburski",
|
"Croatian": "chorwacki",
|
||||||
"Macedonian": "macedoński",
|
"Czech": "czeski",
|
||||||
"Malagasy": "malgaski",
|
"Danish": "duński",
|
||||||
"Malay": "malajski",
|
"Dutch": "holenderski",
|
||||||
"Malayalam": "malajalam",
|
"Esperanto": "esperanto",
|
||||||
"Maltese": "maltański",
|
"Estonian": "estoński",
|
||||||
"Maori": "maoryski",
|
"Filipino": "filipiński",
|
||||||
"Marathi": "marathi",
|
"Finnish": "fiński",
|
||||||
"Mongolian": "mongolski",
|
"French": "francuski",
|
||||||
"Nepali": "nepalski",
|
"Galician": "galicyjski",
|
||||||
"Norwegian": "norweski",
|
"Georgian": "gruziński",
|
||||||
"Nyanja": "njandża",
|
"German": "niemiecki",
|
||||||
"Pashto": "paszto",
|
"Greek": "grecki",
|
||||||
"Persian": "perski",
|
"Gujarati": "gudźarati",
|
||||||
"Polish": "polski",
|
"Haitian Creole": "kreolski haitański",
|
||||||
"Portuguese": "portugalski",
|
"Hausa": "hausa",
|
||||||
"Punjabi": "pendżabski",
|
"Hawaiian": "hawajski",
|
||||||
"Romanian": "rumuński",
|
"Hebrew": "hebrajski",
|
||||||
"Russian": "rosyjski",
|
"Hindi": "hindi",
|
||||||
"Samoan": "samoański",
|
"Hmong": "hmong",
|
||||||
"Scottish Gaelic": "gaelicki szkocki",
|
"Hungarian": "węgierski",
|
||||||
"Serbian": "serbski",
|
"Icelandic": "islandzki",
|
||||||
"Shona": "shona",
|
"Igbo": "ibo",
|
||||||
"Sindhi": "sindhi",
|
"Indonesian": "indonezyjski",
|
||||||
"Sinhala": "syngaleski",
|
"Irish": "irlandzki",
|
||||||
"Slovak": "słowacki",
|
"Italian": "włoski",
|
||||||
"Slovenian": "słoweński",
|
"Japanese": "japoński",
|
||||||
"Somali": "somalijski",
|
"Javanese": "jawajski",
|
||||||
"Southern Sotho": "sotho południowy",
|
"Kannada": "kannada",
|
||||||
"Spanish": "hiszpański",
|
"Kazakh": "kazachski",
|
||||||
"Spanish (Latin America)": "hiszpański (ameryka łacińska)",
|
"Khmer": "khmerski",
|
||||||
"Sundanese": "sundajski",
|
"Korean": "koreański",
|
||||||
"Swahili": "suahili",
|
"Kurdish": "kurdyjski",
|
||||||
"Swedish": "szwedzki",
|
"Kyrgyz": "kirgiski",
|
||||||
"Tajik": "tadżycki",
|
"Lao": "laotański",
|
||||||
"Tamil": "tamilski",
|
"Latin": "łaciński",
|
||||||
"Telugu": "telugu",
|
"Latvian": "łotewski",
|
||||||
"Thai": "tajski",
|
"Lithuanian": "litewski",
|
||||||
"Turkish": "turecki",
|
"Luxembourgish": "luksemburski",
|
||||||
"Ukrainian": "ukraiński",
|
"Macedonian": "macedoński",
|
||||||
"Urdu": "urdu",
|
"Malagasy": "malgaski",
|
||||||
"Uzbek": "uzbecki",
|
"Malay": "malajski",
|
||||||
"Vietnamese": "wietnamski",
|
"Malayalam": "malajalam",
|
||||||
"Welsh": "walijski",
|
"Maltese": "maltański",
|
||||||
"Western Frisian": "zachodniofryzyjski",
|
"Maori": "maoryski",
|
||||||
"Xhosa": "xhosa",
|
"Marathi": "marathi",
|
||||||
"Yiddish": "jidysz",
|
"Mongolian": "mongolski",
|
||||||
"Yoruba": "joruba",
|
"Nepali": "nepalski",
|
||||||
"Zulu": "zuluski",
|
"Norwegian Bokmål": "norweski",
|
||||||
"`x` years": "`x` lat",
|
"Nyanja": "njandża",
|
||||||
"`x` months": "`x` miesięcy",
|
"Pashto": "paszto",
|
||||||
"`x` weeks": "`x` tygodni",
|
"Persian": "perski",
|
||||||
"`x` days": "`x` dni",
|
"Polish": "polski",
|
||||||
"`x` hours": "`x` godzin",
|
"Portuguese": "portugalski",
|
||||||
"`x` minutes": "`x` minut",
|
"Punjabi": "pendżabski",
|
||||||
"`x` seconds": "`x` sekund",
|
"Romanian": "rumuński",
|
||||||
"Fallback comments: ": "Zastępcze komentarze: ",
|
"Russian": "rosyjski",
|
||||||
"Popular": "Popularne",
|
"Samoan": "samoański",
|
||||||
"Top": "Najczęściej oglądane",
|
"Scottish Gaelic": "gaelicki szkocki",
|
||||||
"About": "Informacje",
|
"Serbian": "serbski",
|
||||||
"Rating: ": "Ocena: ",
|
"Shona": "shona",
|
||||||
"Language: ": "Język: ",
|
"Sindhi": "sindhi",
|
||||||
"Default": "Domyślnie",
|
"Sinhala": "syngaleski",
|
||||||
"Music": "Muzyka",
|
"Slovak": "słowacki",
|
||||||
"Gaming": "Gry",
|
"Slovenian": "słoweński",
|
||||||
"News": "Wiadomości",
|
"Somali": "somalijski",
|
||||||
"Movies": "Filmy",
|
"Southern Sotho": "sotho południowy",
|
||||||
"Download": "Pobierz",
|
"Spanish": "hiszpański",
|
||||||
"Download as: ": "Pobierz jako: ",
|
"Spanish (Latin America)": "hiszpański (ameryka łacińska)",
|
||||||
"%A %B %-d, %Y": "",
|
"Sundanese": "sundajski",
|
||||||
"(edited)": "(edytowany)",
|
"Swahili": "suahili",
|
||||||
"Youtube permalink of the comment": "Odnośnik bezpośredni do komentarza na YouTube",
|
"Swedish": "szwedzki",
|
||||||
"`x` marked it with a ❤": "'x' oznaczonych ❤",
|
"Tajik": "tadżycki",
|
||||||
"Audio mode": "Tryb audio",
|
"Tamil": "tamilski",
|
||||||
"Video mode": "Tryb wideo",
|
"Telugu": "telugu",
|
||||||
"Videos": "Filmy",
|
"Thai": "tajski",
|
||||||
"Playlists": "Playlisty",
|
"Turkish": "turecki",
|
||||||
"Current version: ": "Aktualna wersja: "
|
"Ukrainian": "ukraiński",
|
||||||
}
|
"Urdu": "urdu",
|
||||||
|
"Uzbek": "uzbecki",
|
||||||
|
"Vietnamese": "wietnamski",
|
||||||
|
"Welsh": "walijski",
|
||||||
|
"Western Frisian": "zachodniofryzyjski",
|
||||||
|
"Xhosa": "xhosa",
|
||||||
|
"Yiddish": "jidysz",
|
||||||
|
"Yoruba": "joruba",
|
||||||
|
"Zulu": "zuluski",
|
||||||
|
"`x` years": "`x` lat",
|
||||||
|
"`x` months": "`x` miesięcy",
|
||||||
|
"`x` weeks": "`x` tygodni",
|
||||||
|
"`x` days": "`x` dni",
|
||||||
|
"`x` hours": "`x` godzin",
|
||||||
|
"`x` minutes": "`x` minut",
|
||||||
|
"`x` seconds": "`x` sekund",
|
||||||
|
"Fallback comments: ": "Zastępcze komentarze: ",
|
||||||
|
"Popular": "Popularne",
|
||||||
|
"Top": "Najczęściej oglądane",
|
||||||
|
"About": "Informacje",
|
||||||
|
"Rating: ": "Ocena: ",
|
||||||
|
"Language: ": "Język: ",
|
||||||
|
"View as playlist": "Obejrzyj w playliście",
|
||||||
|
"Default": "Domyślnie",
|
||||||
|
"Music": "Muzyka",
|
||||||
|
"Gaming": "Gry",
|
||||||
|
"News": "Wiadomości",
|
||||||
|
"Movies": "Filmy",
|
||||||
|
"Download": "Pobierz",
|
||||||
|
"Download as: ": "Pobierz jako: ",
|
||||||
|
"%A %B %-d, %Y": "",
|
||||||
|
"(edited)": "(edytowany)",
|
||||||
|
"YouTube comment permalink": "Odnośnik bezpośredni do komentarza na YouTube",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "`x` oznaczonych ❤",
|
||||||
|
"Audio mode": "Tryb audio",
|
||||||
|
"Video mode": "Tryb wideo",
|
||||||
|
"Videos": "Filmy",
|
||||||
|
"Playlists": "Playlisty",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": "Aktualna wersja: "
|
||||||
|
}
|
||||||
233
locales/ru.json
233
locales/ru.json
@@ -1,164 +1,200 @@
|
|||||||
{
|
{
|
||||||
"`x` subscribers": "`x` подписчиков",
|
"`x` subscribers": "`x` подписчиков",
|
||||||
"`x` videos": "`x` видео",
|
"`x` videos": "`x` видео",
|
||||||
|
"`x` playlists": "",
|
||||||
"LIVE": "ПРЯМОЙ ЭФИР",
|
"LIVE": "ПРЯМОЙ ЭФИР",
|
||||||
"Shared `x` ago": "Опубликовано `x` назад",
|
"Shared `x` ago": "Опубликовано `x` назад",
|
||||||
"Unsubscribe": "Отписаться",
|
"Unsubscribe": "Отписаться",
|
||||||
"Subscribe": "Подписаться",
|
"Subscribe": "Подписаться",
|
||||||
"Login to subscribe to `x`": "Войти, чтобы подписаться на `x`",
|
"View channel on YouTube": "Смотреть канал на YouTube",
|
||||||
"View channel on YouTube": "Канал на YouTube",
|
"View playlist on YouTube": "Посмотреть плейлист на YouTube",
|
||||||
"newest": "новые",
|
"newest": "самые свежие",
|
||||||
"oldest": "старые",
|
"oldest": "самые старые",
|
||||||
"popular": "популярные",
|
"popular": "популярные",
|
||||||
"last": "недавно обновленные",
|
"last": "недавние",
|
||||||
"Next page": "Следующая страница",
|
"Next page": "Следующая страница",
|
||||||
"Previous page": "Предыдущая страница",
|
"Previous page": "Предыдущая страница",
|
||||||
"Clear watch history?": "Очистить историю просмотров?",
|
"Clear watch history?": "Очистить историю просмотров?",
|
||||||
|
"New password": "Новый пароль",
|
||||||
|
"New passwords must match": "Новые пароли не совпадают",
|
||||||
|
"Cannot change password for Google accounts": "Изменить пароль аккаунта Google невозможно",
|
||||||
|
"Authorize token?": "Авторизовать токен?",
|
||||||
|
"Authorize token for `x`?": "Авторизовать токен для `x`?",
|
||||||
"Yes": "Да",
|
"Yes": "Да",
|
||||||
"No": "Нет",
|
"No": "Нет",
|
||||||
"Import and Export Data": "Импорт и экспорт данных",
|
"Import and Export Data": "Импорт и экспорт данных",
|
||||||
"Import": "Импорт",
|
"Import": "Импорт",
|
||||||
"Import Invidious data": "Импортировать данные Invidious",
|
"Import Invidious data": "Импортировать данные Invidious",
|
||||||
"Import YouTube subscriptions": "Импортировать YouTube подписки",
|
"Import YouTube subscriptions": "Импортировать подписки из YouTube",
|
||||||
"Import FreeTube subscriptions (.db)": "Импортировать FreeTube подписки (.db)",
|
"Import FreeTube subscriptions (.db)": "Импортировать подписки из FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Импортировать NewPipe подписки (.json)",
|
"Import NewPipe subscriptions (.json)": "Импортировать подписки из NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Импортировать данные NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Импортировать данные из NewPipe (.zip)",
|
||||||
"Export": "Экспорт",
|
"Export": "Экспорт",
|
||||||
"Export subscriptions as OPML": "Экспортировать подписки в OPML",
|
"Export subscriptions as OPML": "Экспортировать подписки в формате OPML",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Экспортировать подписки в OPML (для NewPipe и FreeTube)",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Экспортировать подписки в формате OPML (для NewPipe и FreeTube)",
|
||||||
"Export data as JSON": "Экспортировать данные в JSON",
|
"Export data as JSON": "Экспортировать данные в формате JSON",
|
||||||
"Delete account?": "Удалить аккаунт?",
|
"Delete account?": "Удалить аккаунт?",
|
||||||
"History": "История",
|
"History": "История",
|
||||||
"An alternative front-end to YouTube": "Альтернативный фронтенд для YouTube",
|
"An alternative front-end to YouTube": "Альтернативный фронтенд для YouTube",
|
||||||
"JavaScript license information": "Лицензии JavaScript",
|
"JavaScript license information": "Информация о лицензиях JavaScript",
|
||||||
"source": "источник",
|
"source": "источник",
|
||||||
"Login": "Войти",
|
"Log in": "Войти",
|
||||||
"Login/Register": "Войти/Регистрация",
|
"Log in/register": "Войти или зарегистрироваться",
|
||||||
"Login to Google": "Войти через Google",
|
"Log in with Google": "Войти через Google",
|
||||||
"User ID:": "ID пользователя:",
|
"User ID": "ID пользователя",
|
||||||
"Password:": "Пароль:",
|
"Password": "Пароль",
|
||||||
"Time (h:mm:ss):": "Время (ч:мм:сс):",
|
"Time (h:mm:ss):": "Время (ч:мм:сс):",
|
||||||
"Text CAPTCHA": "Текст капчи",
|
"Text CAPTCHA": "Текст капчи",
|
||||||
"Image CAPTCHA": "Изображение капчи",
|
"Image CAPTCHA": "Изображение капчи",
|
||||||
"Sign In": "Войти",
|
"Sign In": "Войти",
|
||||||
"Register": "Регистрация",
|
"Register": "Зарегистрироваться",
|
||||||
"Email:": "Эл. почта:",
|
"E-mail": "Электронная почта",
|
||||||
"Google verification code:": "Код подтверждения Google:",
|
"Google verification code": "Код подтверждения Google",
|
||||||
"Preferences": "Настройки",
|
"Preferences": "Настройки",
|
||||||
"Player preferences": "Настройки проигрывателя",
|
"Player preferences": "Настройки проигрывателя",
|
||||||
"Always loop: ": "Всегда повторять: ",
|
"Always loop: ": "Всегда повторять: ",
|
||||||
"Autoplay: ": "Автовоспроизведение: ",
|
"Autoplay: ": "Автовоспроизведение: ",
|
||||||
"Autoplay next video: ": "Автовоспроизведение следующего видео: ",
|
"Play next by default: ": "Всегда включать следующее видео? ",
|
||||||
"Listen by default: ": "Режим \"только аудио\" по-умолчанию: ",
|
"Autoplay next video: ": "Автопроигрывание следующего видео: ",
|
||||||
"Proxy videos? ": "Проксировать видео? ",
|
"Listen by default: ": "Режим «только аудио» по умолчанию: ",
|
||||||
"Default speed: ": "Скорость по-умолчанию: ",
|
"Proxy videos: ": "Проигрывать видео через прокси? ",
|
||||||
|
"Default speed: ": "Скорость видео по умолчанию: ",
|
||||||
"Preferred video quality: ": "Предпочтительное качество видео: ",
|
"Preferred video quality: ": "Предпочтительное качество видео: ",
|
||||||
"Player volume: ": "Громкость воспроизведения: ",
|
"Player volume: ": "Громкость видео: ",
|
||||||
"Default comments: ": "Источник комментариев: ",
|
"Default comments: ": "Источник комментариев: ",
|
||||||
"youtube": "YouTube",
|
"youtube": "YouTube",
|
||||||
"reddit": "Reddit",
|
"reddit": "Reddit",
|
||||||
"Default captions: ": "Субтитры по-умолчанию: ",
|
"Default captions: ": "Основной язык субтитров: ",
|
||||||
"Fallback captions: ": "Резервные субтитры: ",
|
"Fallback captions: ": "Дополнительный язык субтитров: ",
|
||||||
"Show related videos? ": "Показывать похожие видео? ",
|
"Show related videos: ": "Показывать похожие видео? ",
|
||||||
"Visual preferences": "Визуальные настройки",
|
"Show annotations by default: ": "Всегда показывать аннотации? ",
|
||||||
"Dark mode: ": "Темная тема: ",
|
"Visual preferences": "Настройки сайта",
|
||||||
"Thin mode: ": "Облегченный режим: ",
|
"Player style: ": "",
|
||||||
|
"Dark mode: ": "Тёмное оформление: ",
|
||||||
|
"Theme: ": "",
|
||||||
|
"dark": "",
|
||||||
|
"light": "",
|
||||||
|
"Thin mode: ": "Облегчённое оформление: ",
|
||||||
"Subscription preferences": "Настройки подписок",
|
"Subscription preferences": "Настройки подписок",
|
||||||
"Redirect homepage to feed: ": "Отображать ленту вместо главной страницы: ",
|
"Show annotations by default for subscribed channels: ": "Всегда показывать аннотации в видео каналов, на которые вы подписаны? ",
|
||||||
"Number of videos shown in feed: ": "Число видео в ленте: ",
|
"Redirect homepage to feed: ": "Отображать видео с каналов, на которые вы подписаны, как главную страницу: ",
|
||||||
"Sort videos by: ": "Сортировать видео по: ",
|
"Number of videos shown in feed: ": "Число видео, на которые вы подписаны, в ленте: ",
|
||||||
"published": "дате публикации",
|
"Sort videos by: ": "Сортировать видео: ",
|
||||||
"published - reverse": "дате - обратный порядок",
|
"published": "по дате публикации",
|
||||||
"alphabetically": "алфавиту",
|
"published - reverse": "по дате публикации в обратном порядке",
|
||||||
"alphabetically - reverse": "алфавиту - обратный порядок",
|
"alphabetically": "по алфавиту",
|
||||||
"channel name": "имени канала",
|
"alphabetically - reverse": "по алфавиту в обратном порядке",
|
||||||
"channel name - reverse": "имени канала - обратный порядок",
|
"channel name": "по названию канала",
|
||||||
"Only show latest video from channel: ": "Отображать только последние видео с каждого канала: ",
|
"channel name - reverse": "по названию канала в обратном порядке",
|
||||||
"Only show latest unwatched video from channel: ": "Отображать только непросмотренные видео с каждого канала: ",
|
"Only show latest video from channel: ": "Показывать только последние видео с каналов: ",
|
||||||
"Only show unwatched: ": "Отображать только непросмотренные видео: ",
|
"Only show latest unwatched video from channel: ": "Показывать только непросмотренные видео с каналов: ",
|
||||||
"Only show notifications (if there are any): ": "Отображать только оповещения (если есть): ",
|
"Only show unwatched: ": "Показывать только непросмотренные видео: ",
|
||||||
|
"Only show notifications (if there are any): ": "Показывать только оповещения, если они есть: ",
|
||||||
|
"Enable web notifications": "Включить уведомления в браузере",
|
||||||
|
"`x` uploaded a video": "`x` разместил видео",
|
||||||
|
"`x` is live": "`x` в прямом эфире",
|
||||||
"Data preferences": "Настройки данных",
|
"Data preferences": "Настройки данных",
|
||||||
"Clear watch history": "Очистить историю просмотра",
|
"Clear watch history": "Очистить историю просмотров",
|
||||||
"Import/Export data": "Импорт/Экспорт данных",
|
"Import/export data": "Импорт/Экспорт данных",
|
||||||
"Manage subscriptions": "Управление подписками",
|
"Change password": "Изменить пароль",
|
||||||
|
"Manage subscriptions": "Управлять подписками",
|
||||||
|
"Manage tokens": "Управлять токенами",
|
||||||
"Watch history": "История просмотров",
|
"Watch history": "История просмотров",
|
||||||
"Delete account": "Удалить аккаунт",
|
"Delete account": "Удалить аккаунт",
|
||||||
"Administrator preferences": "Настройки администратора",
|
"Administrator preferences": "Администраторские настройки",
|
||||||
"Default homepage: ": "Главная страница по умолчанию: ",
|
"Default homepage: ": "Главная страница по умолчанию: ",
|
||||||
"Feed menu: ": "Меню ленты: ",
|
"Feed menu: ": "Меню ленты видео: ",
|
||||||
"Top enabled? ": "Включить ТОП? ",
|
"Top enabled: ": "Включить топ видео? ",
|
||||||
"CAPTCHA enabled? ": "Включить капчу? ",
|
"CAPTCHA enabled: ": "Включить капчу? ",
|
||||||
"Login enabled? ": "Включить логин? ",
|
"Login enabled: ": "Включить авторизацию? ",
|
||||||
"Registration enabled? ": "Включить регистрацию? ",
|
"Registration enabled: ": "Включить регистрацию? ",
|
||||||
"Report statistics? ": "Отображать статистику? ",
|
"Report statistics: ": "Сообщать статистику? ",
|
||||||
"Save preferences": "Сохранить настройки",
|
"Save preferences": "Сохранить настройки",
|
||||||
"Subscription manager": "Менеджер подписок",
|
"Subscription manager": "Менеджер подписок",
|
||||||
|
"Token manager": "Менеджер токенов",
|
||||||
|
"Token": "Токен",
|
||||||
"`x` subscriptions": "`x` подписок",
|
"`x` subscriptions": "`x` подписок",
|
||||||
"Import/Export": "Импорт/Экспорт",
|
"`x` tokens": "`x` токенов",
|
||||||
|
"Import/export": "Импорт и экспорт",
|
||||||
"unsubscribe": "отписаться",
|
"unsubscribe": "отписаться",
|
||||||
|
"revoke": "отозвать",
|
||||||
"Subscriptions": "Подписки",
|
"Subscriptions": "Подписки",
|
||||||
"`x` unseen notifications": "`x` новых оповещений",
|
"`x` unseen notifications": "`x` непросмотренных оповещений",
|
||||||
"search": "поиск",
|
"search": "поиск",
|
||||||
"Sign out": "Выйти",
|
"Log out": "Выйти",
|
||||||
"Released under the AGPLv3 by Omar Roth.": "Распространяется Omar Roth по AGPLv3.",
|
"Released under the AGPLv3 by Omar Roth.": "Реализовано Омаром Ротом по лицензии AGPLv3.",
|
||||||
"Source available here.": "Исходный код доступен здесь.",
|
"Source available here.": "Исходный код доступен здесь.",
|
||||||
"View JavaScript license information.": "Посмотреть лицензии JavaScript кода.",
|
"View JavaScript license information.": "Посмотреть информацию по лицензии JavaScript.",
|
||||||
"View privacy policy.": "См. политику конфиденциальности.",
|
"View privacy policy.": "Посмотреть политику конфиденциальности.",
|
||||||
"Trending": "В тренде",
|
"Trending": "В тренде",
|
||||||
"Unlisted": "Доступно по ссылке",
|
"Public": "",
|
||||||
"Watch video on Youtube": "Смотреть на YouTube",
|
"Unlisted": "Нет в списке",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "Смотреть на YouTube",
|
||||||
|
"Hide annotations": "Скрыть аннотации",
|
||||||
|
"Show annotations": "Показать аннотации",
|
||||||
"Genre: ": "Жанр: ",
|
"Genre: ": "Жанр: ",
|
||||||
"License: ": "Лицензия: ",
|
"License: ": "Лицензия: ",
|
||||||
"Family friendly? ": "Семейный просмотр: ",
|
"Family friendly? ": "Семейный просмотр: ",
|
||||||
"Wilson score: ": "Рейтинг Вильсона: ",
|
"Wilson score: ": "Рейтинг Уилсона: ",
|
||||||
"Engagement: ": "Вовлеченность: ",
|
"Engagement: ": "Вовлечённость: ",
|
||||||
"Whitelisted regions: ": "Доступно для: ",
|
"Whitelisted regions: ": "Доступно в регионах: ",
|
||||||
"Blacklisted regions: ": "Недоступно для: ",
|
"Blacklisted regions: ": "Недоступно в регионах: ",
|
||||||
"Shared `x`": "Опубликовано `x`",
|
"Shared `x`": "Опубликовано `x`",
|
||||||
|
"`x` views": "`x` просмотров",
|
||||||
"Premieres in `x`": "Премьера через `x`",
|
"Premieres in `x`": "Премьера через `x`",
|
||||||
"Hi! Looks like you have JavaScript disabled. Click here to view comments, keep in mind it may take a bit longer to load.": "Похоже, что у Вас отключен JavaScript. Нажмите сюда, чтобы увидеть комментарии (учтите, что они могут загружаться дольше).",
|
"Premieres `x`": "Премьера `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Похоже, у вас отключён JavaScript. Чтобы увидить комментарии, нажмите сюда, но учтите: они могут загружаться немного медленнее.",
|
||||||
"View YouTube comments": "Смотреть комментарии с YouTube",
|
"View YouTube comments": "Смотреть комментарии с YouTube",
|
||||||
"View more comments on Reddit": "Больше комментариев на Reddit",
|
"View more comments on Reddit": "Посмотреть больше комментариев на Reddit",
|
||||||
"View `x` comments": "Показать `x` комментариев",
|
"View `x` comments": "Показать `x` комментариев",
|
||||||
"View Reddit comments": "Смотреть комментарии с Reddit",
|
"View Reddit comments": "Смотреть комментарии с Reddit",
|
||||||
"Hide replies": "Скрыть ответы",
|
"Hide replies": "Скрыть ответы",
|
||||||
"Show replies": "Показать ответы",
|
"Show replies": "Показать ответы",
|
||||||
"Incorrect password": "Неправильный пароль",
|
"Incorrect password": "Неправильный пароль",
|
||||||
"Quota exceeded, try again in a few hours": "Превышена квота, попробуйте снова через несколько часов",
|
"Quota exceeded, try again in a few hours": "Лимит превышен, попробуйте снова через несколько часов",
|
||||||
"Unable to login, make sure two-factor authentication (Authenticator or SMS) is enabled.": "Вход не выполнен, проверьте, не включена ли двухфакторная аутентификация.",
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Войти не удаётся. Проверьте, не включена ли двухфакторная аутентификация (по коду или смс).",
|
||||||
"Invalid TFA code": "Неправильный TFA код",
|
"Invalid TFA code": "Неправильный код двухфакторной аутентификации",
|
||||||
"Login failed. This may be because two-factor authentication is not enabled on your account.": "Не удалось войти. Это может быть из-за того, что в вашем аккаунте не включена двухфакторная аутентификация.",
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Не удаётся войти. Это может быть из-за того, что в вашем аккаунте не включена двухфакторная аутентификация.",
|
||||||
"Invalid answer": "Неверный ответ",
|
"Wrong answer": "Неправильный ответ",
|
||||||
"Invalid CAPTCHA": "Неверная капча",
|
"Erroneous CAPTCHA": "Неправильная капча",
|
||||||
"CAPTCHA is a required field": "Необходимо ввести капчу",
|
"CAPTCHA is a required field": "Необходимо пройти капчу",
|
||||||
"User ID is a required field": "Необходимо ввести идентификатор пользователя",
|
"User ID is a required field": "Необходимо ввести ID пользователя",
|
||||||
"Password is a required field": "Необходимо ввести пароль",
|
"Password is a required field": "Необходимо ввести пароль",
|
||||||
"Invalid username or password": "Недопустимый пароль или имя пользователя",
|
"Wrong username or password": "Неправильный логин или пароль",
|
||||||
"Please sign in using 'Sign in with Google'": "Пожалуйста войдите через Google",
|
"Please sign in using 'Log in with Google'": "Пожалуйста, нажмите «Войти через Google»",
|
||||||
"Password cannot be empty": "Пароль не может быть пустым",
|
"Password cannot be empty": "Пароль не может быть пустым",
|
||||||
"Password cannot be longer than 55 characters": "Пароль не может быть длиннее 55 символов",
|
"Password cannot be longer than 55 characters": "Пароль не может быть длиннее 55 символов",
|
||||||
"Please sign in": "Пожалуйста, войдите",
|
"Please log in": "Пожалуйста, войдите",
|
||||||
"Invidious Private Feed for `x`": "Приватная лента Invidious для `x`",
|
"Invidious Private Feed for `x`": "Приватная лента Invidious для `x`",
|
||||||
"channel:`x`": "канал: `x`",
|
"channel:`x`": "канал: `x`",
|
||||||
"Deleted or invalid channel": "Канал удален или не найден",
|
"Deleted or invalid channel": "Канал удалён или не найден",
|
||||||
"This channel does not exist.": "Такой канал не существует.",
|
"This channel does not exist.": "Такого канала не существует.",
|
||||||
"Could not get channel info.": "Невозможно получить информацию о канале.",
|
"Could not get channel info.": "Не удаётся получить информацию об этом канале.",
|
||||||
"Could not fetch comments": "Невозможно получить комментарии",
|
"Could not fetch comments": "Не удаётся загрузить комментарии",
|
||||||
"View `x` replies": "Показать `x` ответов",
|
"View `x` replies": "Показать `x` ответов",
|
||||||
"`x` ago": "`x` назад",
|
"`x` ago": "`x` назад",
|
||||||
"Load more": "Загрузить больше",
|
"Load more": "Загрузить больше",
|
||||||
"`x` points": "`x` очков",
|
"`x` points": "`x` очков",
|
||||||
"Could not create mix.": "Невозможно создать \"микс\".",
|
"Could not create mix.": "Не удаётся создать микс.",
|
||||||
"Playlist is empty": "Плейлист пуст",
|
"Empty playlist": "Плейлист пуст",
|
||||||
"Invalid playlist.": "Некорректный плейлист.",
|
"Not a playlist.": "Некорректный плейлист.",
|
||||||
"Playlist does not exist.": "Плейлист не существует.",
|
"Playlist does not exist.": "Плейлист не существует.",
|
||||||
"Could not pull trending pages.": "Невозможно получить страницы \"в тренде\".",
|
"Could not pull trending pages.": "Не удаётся загрузить страницы «в тренде».",
|
||||||
"Hidden field \"challenge\" is a required field": "Необходимо заполнить скрытое поле \"challenge\"",
|
"Hidden field \"challenge\" is a required field": "Необходимо заполнить скрытое поле «challenge»",
|
||||||
"Hidden field \"token\" is a required field": "Необходимо заполнить скрытое поле \"токен\"",
|
"Hidden field \"token\" is a required field": "Необходимо заполнить скрытое поле «токен»",
|
||||||
"Invalid challenge": "Неправильный ответ в \"challenge\"",
|
"Erroneous challenge": "Неправильный ответ в «challenge»",
|
||||||
"Invalid token": "Неправильный токен",
|
"Erroneous token": "Неправильный токен",
|
||||||
"Invalid user": "Недопустимое имя пользователя",
|
"No such user": "Недопустимое имя пользователя",
|
||||||
"Token is expired, please try again": "Срок действия токена истек, попробуйте позже",
|
"Token is expired, please try again": "Срок действия токена истёк, попробуйте позже",
|
||||||
"English": "Английский",
|
"English": "Английский",
|
||||||
"English (auto-generated)": "Английский (созданы автоматически)",
|
"English (auto-generated)": "Английский (созданы автоматически)",
|
||||||
"Afrikaans": "Африкаанс",
|
"Afrikaans": "Африкаанс",
|
||||||
@@ -226,7 +262,7 @@
|
|||||||
"Marathi": "Маратхи",
|
"Marathi": "Маратхи",
|
||||||
"Mongolian": "Монгольская",
|
"Mongolian": "Монгольская",
|
||||||
"Nepali": "Непальский",
|
"Nepali": "Непальский",
|
||||||
"Norwegian": "Норвежский",
|
"Norwegian Bokmål": "Норвежский",
|
||||||
"Nyanja": "Ньянджа",
|
"Nyanja": "Ньянджа",
|
||||||
"Pashto": "Пушту",
|
"Pashto": "Пушту",
|
||||||
"Persian": "Персидский",
|
"Persian": "Персидский",
|
||||||
@@ -278,6 +314,7 @@
|
|||||||
"About": "О сайте",
|
"About": "О сайте",
|
||||||
"Rating: ": "Рейтинг: ",
|
"Rating: ": "Рейтинг: ",
|
||||||
"Language: ": "Язык: ",
|
"Language: ": "Язык: ",
|
||||||
|
"View as playlist": "Смотреть как плейлист",
|
||||||
"Default": "По-умолчанию",
|
"Default": "По-умолчанию",
|
||||||
"Music": "Музыка",
|
"Music": "Музыка",
|
||||||
"Gaming": "Игры",
|
"Gaming": "Игры",
|
||||||
@@ -287,11 +324,13 @@
|
|||||||
"Download as: ": "Скачать как: ",
|
"Download as: ": "Скачать как: ",
|
||||||
"%A %B %-d, %Y": "%-d %B %Y, %A",
|
"%A %B %-d, %Y": "%-d %B %Y, %A",
|
||||||
"(edited)": "(изменено)",
|
"(edited)": "(изменено)",
|
||||||
"Youtube permalink of the comment": "Прямая ссылка на YouTube",
|
"YouTube comment permalink": "Прямая ссылка на YouTube",
|
||||||
|
"permalink": "",
|
||||||
"`x` marked it with a ❤": "❤ от автора канала \"`x`\"",
|
"`x` marked it with a ❤": "❤ от автора канала \"`x`\"",
|
||||||
"Audio mode": "Аудио режим",
|
"Audio mode": "Аудио режим",
|
||||||
"Video mode": "Видео режим",
|
"Video mode": "Видео режим",
|
||||||
"Videos": "Видео",
|
"Videos": "Видео",
|
||||||
"Playlists": "Плейлисты",
|
"Playlists": "Плейлисты",
|
||||||
|
"Community": "",
|
||||||
"Current version: ": "Текущая версия: "
|
"Current version: ": "Текущая версия: "
|
||||||
}
|
}
|
||||||
344
locales/tr.json
Normal file
344
locales/tr.json
Normal file
@@ -0,0 +1,344 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": "",
|
||||||
|
"`x` videos": "",
|
||||||
|
"`x` playlists": "",
|
||||||
|
"`x` subscribers.": "`x` abone.",
|
||||||
|
"`x` videos.": "`x` video.",
|
||||||
|
"LIVE": "CANLI",
|
||||||
|
"Shared `x` ago": "`x` önce paylaşıldı",
|
||||||
|
"Unsubscribe": "Abonelikten çık",
|
||||||
|
"Subscribe": "Abone ol",
|
||||||
|
"View channel on YouTube": "Kanalı YouTube'da görüntüle",
|
||||||
|
"View playlist on YouTube": "Çalma listesini YouTube'da görüntüle",
|
||||||
|
"newest": "en yeni",
|
||||||
|
"oldest": "en eski",
|
||||||
|
"popular": "popüler",
|
||||||
|
"last": "son",
|
||||||
|
"Next page": "Sonraki sayfa",
|
||||||
|
"Previous page": "Önceki sayfa",
|
||||||
|
"Clear watch history?": "İzleme geçmisini temizle?",
|
||||||
|
"New password": "Yeni parola",
|
||||||
|
"New passwords must match": "Yeni parolalar eşleşmek zorunda",
|
||||||
|
"Cannot change password for Google accounts": "Google hesapları için parola değiştirilemez",
|
||||||
|
"Authorize token?": "Jetonu yetkilendir?",
|
||||||
|
"Authorize token for `x`?": "`x` için jetonu yetkilendir?",
|
||||||
|
"Yes": "Evet",
|
||||||
|
"No": "Hayır",
|
||||||
|
"Import and Export Data": "Verileri İçe ve Dışa Aktar",
|
||||||
|
"Import": "İçe aktar",
|
||||||
|
"Import Invidious data": "İnvidious verilerini içe aktar",
|
||||||
|
"Import YouTube subscriptions": "YouTube aboneliklerini içe aktar",
|
||||||
|
"Import FreeTube subscriptions (.db)": "FreeTube aboneliklerini içe aktar (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "NewPipe aboneliklerini içe aktar (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "NewPipe verilerini içe aktar (.zip)",
|
||||||
|
"Export": "Dışa aktar",
|
||||||
|
"Export subscriptions as OPML": "Abonelikleri OPML olarak dışa aktar",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Abonelikleri OPML olarak dışa aktar (NewPipe ve FreeTube için)",
|
||||||
|
"Export data as JSON": "Verileri JSON olarak dışa aktar",
|
||||||
|
"Delete account?": "Hesabı sil?",
|
||||||
|
"History": "Geçmiş",
|
||||||
|
"An alternative front-end to YouTube": "YouTube için alternatif bir ön-yüz",
|
||||||
|
"JavaScript license information": "JavaScript lisans bilgileri",
|
||||||
|
"source": "kaynak",
|
||||||
|
"Log in": "Oturum aç",
|
||||||
|
"Log in/register": "Oturum aç/kayıt ol",
|
||||||
|
"Log in with Google": "Google ile oturum aç",
|
||||||
|
"User ID": "Kullanıcı kimliği",
|
||||||
|
"Password": "Parola",
|
||||||
|
"Time (h:mm:ss):": "Zaman (h:mm:ss):",
|
||||||
|
"Text CAPTCHA": "Metin CAPTCHA",
|
||||||
|
"Image CAPTCHA": "Resim CAPTCHA",
|
||||||
|
"Sign In": "Oturum Aç",
|
||||||
|
"Register": "Kayıt Ol",
|
||||||
|
"E-mail": "E-posta",
|
||||||
|
"Google verification code": "Google doğrulama kodu",
|
||||||
|
"Preferences": "Tercihler",
|
||||||
|
"Player preferences": "Oynatıcı tercihleri",
|
||||||
|
"Always loop: ": "Sürekli döngü: ",
|
||||||
|
"Autoplay: ": "Otomatik oynat: ",
|
||||||
|
"Play next by default: ": "Varsayılan olarak sonrakini oynat: ",
|
||||||
|
"Autoplay next video: ": "Sonraki videoyu otomatik oynat: ",
|
||||||
|
"Listen by default: ": "Varsayılan olarak dinle: ",
|
||||||
|
"Proxy videos: ": "Videoları proxy'le: ",
|
||||||
|
"Default speed: ": "Varsayılan hız: ",
|
||||||
|
"Preferred video quality: ": "Tercih edilen video kalitesi: ",
|
||||||
|
"Player volume: ": "Oynatıcı ses seviyesi: ",
|
||||||
|
"Default comments: ": "Varsayılan yorumlar: ",
|
||||||
|
"youtube": "youtube",
|
||||||
|
"reddit": "reddit",
|
||||||
|
"Default captions: ": "Varsayılan altyazılar: ",
|
||||||
|
"Fallback captions: ": "Yedek altyazılar: ",
|
||||||
|
"Show related videos: ": "İlgili videoları göster: ",
|
||||||
|
"Show annotations by default: ": "Varsayılan olarak ek açıklamaları göster: ",
|
||||||
|
"Visual preferences": "Görsel tercihler",
|
||||||
|
"Player style: ": "Oynatıcı biçimi: ",
|
||||||
|
"Dark mode: ": "Karanlık mod: ",
|
||||||
|
"Theme: ": "Tema: ",
|
||||||
|
"dark": "karanlık",
|
||||||
|
"light": "aydınlık",
|
||||||
|
"Thin mode: ": "İnce mod: ",
|
||||||
|
"Subscription preferences": "Abonelik tercihleri",
|
||||||
|
"Show annotations by default for subscribed channels: ": "Abone olunan kanallar için ek açıklamaları varsayılan olarak göster: ",
|
||||||
|
"Redirect homepage to feed: ": "Ana sayfayı akışa yönlendir: ",
|
||||||
|
"Number of videos shown in feed: ": "Akışta gösterilen video sayısı: ",
|
||||||
|
"Sort videos by: ": "Videoları sıralama kriteri: ",
|
||||||
|
"published": "yayınlandı",
|
||||||
|
"published - reverse": "yayınlandı - ters",
|
||||||
|
"alphabetically": "alfabetik olarak",
|
||||||
|
"alphabetically - reverse": "alfabetik olarak - ters",
|
||||||
|
"channel name": "kanal adı",
|
||||||
|
"channel name - reverse": "kanal adı - ters",
|
||||||
|
"Only show latest video from channel: ": "Sadece kanaldaki en son videoyu göster: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "Sadece kanaldaki en son izlenmemiş videoyu göster: ",
|
||||||
|
"Only show unwatched: ": "Sadece izlenmemişleri göster: ",
|
||||||
|
"Only show notifications (if there are any): ": "Sadece bildirimleri göster (eğer varsa): ",
|
||||||
|
"Enable web notifications": "Ağ bildirimlerini etkinleştir",
|
||||||
|
"`x` uploaded a video": "`x` bir video yükledi",
|
||||||
|
"`x` is live": "`x` canlı yayında",
|
||||||
|
"Data preferences": "Veri tercihleri",
|
||||||
|
"Clear watch history": "İzleme geçmişini temizle",
|
||||||
|
"Import/export data": "Verileri içe/dışa aktar",
|
||||||
|
"Change password": "Parolayı değiştir",
|
||||||
|
"Manage subscriptions": "Abonelikleri yönet",
|
||||||
|
"Manage tokens": "Jetonları yönet",
|
||||||
|
"Watch history": "İzleme geçmişi",
|
||||||
|
"Delete account": "Hesap silme",
|
||||||
|
"Administrator preferences": "Yönetici tercihleri",
|
||||||
|
"Default homepage: ": "Varsayılan ana sayfa: ",
|
||||||
|
"Feed menu: ": "Akış menüsü: ",
|
||||||
|
"Top enabled: ": "Top etkin: ",
|
||||||
|
"CAPTCHA enabled: ": "CAPTCHA etkin: ",
|
||||||
|
"Login enabled: ": "Oturum açma etkin: ",
|
||||||
|
"Registration enabled: ": "Kayıt olma etkin: ",
|
||||||
|
"Report statistics: ": "Rapor istatistikleri: ",
|
||||||
|
"Save preferences": "Tercihleri kaydet",
|
||||||
|
"Subscription manager": "Abonelik yöneticisi",
|
||||||
|
"`x` subscriptions": "",
|
||||||
|
"`x` tokens": "",
|
||||||
|
"Token manager": "Jeton yöneticisi",
|
||||||
|
"Token": "Jeton",
|
||||||
|
"`x` subscriptions.": "`x` abonelik.",
|
||||||
|
"`x` tokens.": "`x` jeton.",
|
||||||
|
"`x` unseen notifications": "",
|
||||||
|
"Import/export": "İçe/dışa aktar",
|
||||||
|
"unsubscribe": "abonelikten çık",
|
||||||
|
"revoke": "geri al",
|
||||||
|
"Subscriptions": "Abonelikler",
|
||||||
|
"`x` unseen notifications.": "`x` okunmamış bildirim.",
|
||||||
|
"search": "ara",
|
||||||
|
"Log out": "Çıkış yap",
|
||||||
|
"Public": "",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Omar Roth tarafından AGPLv3 altında yayımlandı.",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Source available here.": "Kaynak kodu burada mevcut.",
|
||||||
|
"View JavaScript license information.": "JavaScript lisans bilgilerini görüntüle.",
|
||||||
|
"View privacy policy.": "Gizlilik politikasını görüntüle.",
|
||||||
|
"Trending": "Trendler",
|
||||||
|
"Unlisted": "Listelenmemiş",
|
||||||
|
"Watch on YouTube": "YouTube'da izle",
|
||||||
|
"Hide annotations": "Ek açıklamaları gizle",
|
||||||
|
"Show annotations": "Ek açıklamaları göster",
|
||||||
|
"Genre: ": "Tür: ",
|
||||||
|
"License: ": "Lisans: ",
|
||||||
|
"Family friendly? ": "Aile için uygun? ",
|
||||||
|
"`x` views": "",
|
||||||
|
"Wilson score: ": "Wilson puanı: ",
|
||||||
|
"Engagement: ": "İzleyenlerin oy verme oranı: ",
|
||||||
|
"Whitelisted regions: ": "Beyaz listeye alınan bölgeler: ",
|
||||||
|
"Blacklisted regions: ": "Kara listeye alınan bölgeler: ",
|
||||||
|
"Shared `x`": "`x` paylaşıldı",
|
||||||
|
"`x` views.": "`x` izlenme.",
|
||||||
|
"Premieres in `x`": "`x`içinde ilk gösterim",
|
||||||
|
"Premieres `x`": "`x` ilk gösterim",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Merhaba! JavaScript'i kapatmış gibi görünüyorsun. Yorumları görüntülemek için buraya tıkla, yüklenmelerinin biraz uzun sürebileceğini unutma.",
|
||||||
|
"View YouTube comments": "YouTube yorumlarını görüntüle",
|
||||||
|
"View more comments on Reddit": "Reddit'te daha fazla yorum görüntüle",
|
||||||
|
"View `x` comments": "`x` yorum görüntüle",
|
||||||
|
"View Reddit comments": "Reddit yorumlarını görüntüle",
|
||||||
|
"Hide replies": "Cevapları gizle",
|
||||||
|
"Show replies": "Cevapları göster",
|
||||||
|
"Incorrect password": "Yanlış parola",
|
||||||
|
"Quota exceeded, try again in a few hours": "Kota aşıldı, birkaç saat içinde tekrar deneyin",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Oturum açılamadı, iki faktörlü kimlik doğrulamanın (Authenticator ya da SMS) açık olduğundan emin olun.",
|
||||||
|
"Invalid TFA code": "Geçersiz TFA kodu",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Giriş başarısız. Bunun nedeni, hesabınız için iki faktörlü kimlik doğrulamanın açık olmaması olabilir.",
|
||||||
|
"Wrong answer": "Yanlış cevap",
|
||||||
|
"Erroneous CAPTCHA": "Hatalı CAPTCHA",
|
||||||
|
"CAPTCHA is a required field": "CAPTCHA zorunlu bir alandır",
|
||||||
|
"User ID is a required field": "Kullanıcı kimliği zorunlu bir alandır",
|
||||||
|
"Password is a required field": "Parola zorunlu bir alandır",
|
||||||
|
"Wrong username or password": "Yanlış kullanıcı adı ya da parola",
|
||||||
|
"Please sign in using 'Log in with Google'": "Lütfen 'Google ile giriş yap' seçeneğini kullanarak oturum açın",
|
||||||
|
"Password cannot be empty": "Parola boş olamaz",
|
||||||
|
"Password cannot be longer than 55 characters": "Parola 55 karakterden uzun olamaz",
|
||||||
|
"Please log in": "Lütfen oturum açın",
|
||||||
|
"View `x` replies": "",
|
||||||
|
"Invidious Private Feed for `x`": "`x` için İnvidious Özel Akışı",
|
||||||
|
"channel:`x`": "kanal:`x`",
|
||||||
|
"`x` points": "",
|
||||||
|
"Deleted or invalid channel": "Silinmiş ya da geçersiz kanal",
|
||||||
|
"This channel does not exist.": "Bu kanal mevcut değil.",
|
||||||
|
"Could not get channel info.": "Kanal bilgisi alınamadı.",
|
||||||
|
"Could not fetch comments": "Yorumlar alınamadı",
|
||||||
|
"View `x` replies.": "`x` yanıtı görüntüle.",
|
||||||
|
"`x` ago": "`x` önce",
|
||||||
|
"Load more": "Daha fazla yükle",
|
||||||
|
"`x` points.": "`x` puan.",
|
||||||
|
"Could not create mix.": "Mix oluşturulamadı.",
|
||||||
|
"Empty playlist": "Boş oynatma listesi",
|
||||||
|
"Not a playlist.": "Oynatma listesi değil.",
|
||||||
|
"Playlist does not exist.": "Oynatma listesi mevcut değil.",
|
||||||
|
"Could not pull trending pages.": "Trend sayfaları alınamıyor.",
|
||||||
|
"Hidden field \"challenge\" is a required field": "Gizli alan \"challenge\" zorunlu bir alandır",
|
||||||
|
"Hidden field \"token\" is a required field": "Gizli alan \"jeton\" zorunlu bir alandır",
|
||||||
|
"Erroneous challenge": "Hatalı challenge",
|
||||||
|
"Erroneous token": "Hatalı jeton",
|
||||||
|
"No such user": "Böyle bir kullanıcı yok",
|
||||||
|
"Token is expired, please try again": "Jetonun süresi doldu, lütfen tekrar deneyin",
|
||||||
|
"English": "İngilizce",
|
||||||
|
"English (auto-generated)": "İngilizce (otomatik oluşturuldu)",
|
||||||
|
"Afrikaans": "Afrikanca",
|
||||||
|
"Albanian": "Arnavutça",
|
||||||
|
"Amharic": "Amharca",
|
||||||
|
"Arabic": "Arapça",
|
||||||
|
"Armenian": "Ermenice",
|
||||||
|
"Azerbaijani": "Azerice",
|
||||||
|
"Bangla": "Bengalce",
|
||||||
|
"Basque": "Baskça",
|
||||||
|
"Belarusian": "Belarusça",
|
||||||
|
"Bosnian": "Boşnakça",
|
||||||
|
"Bulgarian": "Bulgarca",
|
||||||
|
"Burmese": "Birmanca",
|
||||||
|
"Catalan": "Katalanca",
|
||||||
|
"Cebuano": "Sebuanca",
|
||||||
|
"Chinese (Simplified)": "Çince (Basitleştirilmiş)",
|
||||||
|
"Chinese (Traditional)": "Çince (Geleneksel)",
|
||||||
|
"Corsican": "Korsikaca",
|
||||||
|
"Croatian": "Hırvatça",
|
||||||
|
"Czech": "Çekçe",
|
||||||
|
"Danish": "Danca",
|
||||||
|
"Dutch": "Flemenkçe",
|
||||||
|
"Esperanto": "Esperanto",
|
||||||
|
"Estonian": "Estonca",
|
||||||
|
"Filipino": "Filipince",
|
||||||
|
"Finnish": "Fince",
|
||||||
|
"French": "Fransızca",
|
||||||
|
"Galician": "Galiçyaca",
|
||||||
|
"Georgian": "Gürcüce",
|
||||||
|
"German": "Almanca",
|
||||||
|
"Greek": "Yunanca",
|
||||||
|
"Gujarati": "Guceratça",
|
||||||
|
"Haitian Creole": "Haiti Creole dili",
|
||||||
|
"Hausa": "Hausaca",
|
||||||
|
"Hawaiian": "Hawaii dili",
|
||||||
|
"Hebrew": "İbranice",
|
||||||
|
"Hindi": "Hintçe",
|
||||||
|
"Hmong": "Hmong",
|
||||||
|
"Hungarian": "Macarca",
|
||||||
|
"Icelandic": "İzlandaca",
|
||||||
|
"Igbo": "İgbo",
|
||||||
|
"Indonesian": "Endonezce",
|
||||||
|
"Irish": "İrlandaca",
|
||||||
|
"Italian": "İtalyanca",
|
||||||
|
"Japanese": "Japonca",
|
||||||
|
"Javanese": "Cava dili",
|
||||||
|
"Kannada": "Kannada dili",
|
||||||
|
"Kazakh": "Kazakça",
|
||||||
|
"Khmer": "Kmerce",
|
||||||
|
"Korean": "Korece",
|
||||||
|
"Kurdish": "Kürtçe",
|
||||||
|
"Kyrgyz": "Kırgızca",
|
||||||
|
"Lao": "Laoca",
|
||||||
|
"Latin": "Latince",
|
||||||
|
"Latvian": "Letonca",
|
||||||
|
"Lithuanian": "Litvanyaca",
|
||||||
|
"Luxembourgish": "Lüksemburgca",
|
||||||
|
"Macedonian": "Makedonca",
|
||||||
|
"Malagasy": "Malgaşça",
|
||||||
|
"Malay": "Malayca",
|
||||||
|
"Malayalam": "Malayalam dili",
|
||||||
|
"Maltese": "Maltaca",
|
||||||
|
"Maori": "Maori dili",
|
||||||
|
"Marathi": "Marati dili",
|
||||||
|
"Mongolian": "Moğolca",
|
||||||
|
"Nepali": "Nepalce",
|
||||||
|
"Norwegian Bokmål": "Norveççe Bokmål",
|
||||||
|
"Nyanja": "Çevaca",
|
||||||
|
"Pashto": "Peştuca",
|
||||||
|
"Persian": "Farsça",
|
||||||
|
"Polish": "Lehçe",
|
||||||
|
"Portuguese": "Portekizce",
|
||||||
|
"Punjabi": "Pencap dili",
|
||||||
|
"Romanian": "Rumence",
|
||||||
|
"Russian": "Rusça",
|
||||||
|
"Samoan": "Samoa dili",
|
||||||
|
"Scottish Gaelic": "İskoç Galcesi",
|
||||||
|
"Serbian": "Sırpça",
|
||||||
|
"Shona": "Şona dili",
|
||||||
|
"Sindhi": "Sintçe",
|
||||||
|
"Sinhala": "Seylanca",
|
||||||
|
"Slovak": "Slovakça",
|
||||||
|
"Slovenian": "Slovence",
|
||||||
|
"Somali": "Somalice",
|
||||||
|
"Southern Sotho": "Güney Sotho dili",
|
||||||
|
"Spanish": "İspanyolca",
|
||||||
|
"Spanish (Latin America)": "İspanyolca (Latin Amerika)",
|
||||||
|
"Sundanese": "Sundaca",
|
||||||
|
"Swahili": "Svahili dili",
|
||||||
|
"Swedish": "İsveççe",
|
||||||
|
"Tajik": "Tacikçe",
|
||||||
|
"Tamil": "Tamilce",
|
||||||
|
"Telugu": "Telugu dili",
|
||||||
|
"Thai": "Tayca",
|
||||||
|
"Turkish": "Türkçe",
|
||||||
|
"Ukrainian": "Ukraynaca",
|
||||||
|
"Urdu": "Urduca",
|
||||||
|
"Uzbek": "Özbekçe",
|
||||||
|
"Vietnamese": "Vietnamca",
|
||||||
|
"Welsh": "Galce",
|
||||||
|
"Western Frisian": "Batı Frizcesi",
|
||||||
|
"Xhosa": "Xhosa dili",
|
||||||
|
"Yiddish": "Yiddiş",
|
||||||
|
"Yoruba": "Yoruba dili",
|
||||||
|
"Zulu": "Zuluca",
|
||||||
|
"`x` years": "`x` yıl",
|
||||||
|
"`x` months": "`x` ay",
|
||||||
|
"`x` weeks": "`x` hafta",
|
||||||
|
"`x` days": "`x` gün",
|
||||||
|
"`x` hours": "`x` saat",
|
||||||
|
"`x` minutes": "`x` dakika",
|
||||||
|
"`x` seconds": "`x` saniye",
|
||||||
|
"Fallback comments: ": "Yedek yorumlar: ",
|
||||||
|
"Popular": "Popüler",
|
||||||
|
"Top": "Enler",
|
||||||
|
"About": "Hakkında",
|
||||||
|
"Rating: ": "Değerlendirme: ",
|
||||||
|
"Language: ": "Dil: ",
|
||||||
|
"View as playlist": "Oynatma listesi olarak görüntüle",
|
||||||
|
"Default": "Varsayılan",
|
||||||
|
"Music": "Müzik",
|
||||||
|
"Gaming": "Oyun",
|
||||||
|
"News": "Haberler",
|
||||||
|
"Movies": "Filmler",
|
||||||
|
"Download": "İndir",
|
||||||
|
"Download as: ": "Şu şekilde indir: ",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(düzenlendi)",
|
||||||
|
"YouTube comment permalink": "YouTube yorumu kalıcı linki",
|
||||||
|
"permalink": "kalıcı link",
|
||||||
|
"`x` marked it with a ❤": "`x` ❤ ile işaretlendi",
|
||||||
|
"Audio mode": "Ses modu",
|
||||||
|
"Video mode": "Video modu",
|
||||||
|
"Videos": "Videolar",
|
||||||
|
"Playlists": "Oynatma listeleri",
|
||||||
|
"Community": "Topluluk",
|
||||||
|
"Current version: ": "Şu anki versiyon: "
|
||||||
|
}
|
||||||
336
locales/uk.json
Normal file
336
locales/uk.json
Normal file
@@ -0,0 +1,336 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": "`x` підписників",
|
||||||
|
"`x` videos": "`x` відео",
|
||||||
|
"`x` playlists": "",
|
||||||
|
"LIVE": "ПРЯМИЙ ЕФІР",
|
||||||
|
"Shared `x` ago": "Розміщено `x` назад",
|
||||||
|
"Unsubscribe": "Відписатися",
|
||||||
|
"Subscribe": "Підписатися",
|
||||||
|
"View channel on YouTube": "Подивитися канал на YouTube",
|
||||||
|
"View playlist on YouTube": "Подивитися плейлист на YouTube",
|
||||||
|
"newest": "найновіше",
|
||||||
|
"oldest": "найстаріше",
|
||||||
|
"popular": "популярне",
|
||||||
|
"last": "останнє",
|
||||||
|
"Next page": "Наступна сторінка",
|
||||||
|
"Previous page": "Попередня сторінка",
|
||||||
|
"Clear watch history?": "Очистити історію переглядів?",
|
||||||
|
"New password": "Новий пароль",
|
||||||
|
"New passwords must match": "Нові паролі не співпадають",
|
||||||
|
"Cannot change password for Google accounts": "Змінити пароль обліківки Google неможливо",
|
||||||
|
"Authorize token?": "Авторизувати токен?",
|
||||||
|
"Authorize token for `x`?": "Авторизувати токен для `x`?",
|
||||||
|
"Yes": "Так",
|
||||||
|
"No": "Ні",
|
||||||
|
"Import and Export Data": "Імпорт і експорт даних",
|
||||||
|
"Import": "Імпорт",
|
||||||
|
"Import Invidious data": "Імпортувати дані Invidious",
|
||||||
|
"Import YouTube subscriptions": "Імпортувати підписки з YouTube",
|
||||||
|
"Import FreeTube subscriptions (.db)": "Імпортувати підписки з FreeTube (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "Імпортувати підписки з NewPipe (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "Імпортувати дані з NewPipe (.zip)",
|
||||||
|
"Export": "Експорт",
|
||||||
|
"Export subscriptions as OPML": "Експортувати підписки у форматі OPML",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Експортувати підписки у форматі OPML (для NewPipe та FreeTube)",
|
||||||
|
"Export data as JSON": "Експортувати дані у форматі JSON",
|
||||||
|
"Delete account?": "Видалити обліківку?",
|
||||||
|
"History": "Історія",
|
||||||
|
"An alternative front-end to YouTube": "Альтернативний фронтенд до YouTube",
|
||||||
|
"JavaScript license information": "Інформація щодо ліцензій JavaScript",
|
||||||
|
"source": "джерело",
|
||||||
|
"Log in": "Увійти",
|
||||||
|
"Log in/register": "Увійти або зареєструватися",
|
||||||
|
"Log in with Google": "Увійти через Google",
|
||||||
|
"User ID": "ID користувача",
|
||||||
|
"Password": "Пароль",
|
||||||
|
"Time (h:mm:ss):": "Час (г:мм:сс):",
|
||||||
|
"Text CAPTCHA": "Текст капчі",
|
||||||
|
"Image CAPTCHA": "Зображення капчі",
|
||||||
|
"Sign In": "Увійти",
|
||||||
|
"Register": "Зареєструватися",
|
||||||
|
"E-mail": "Електронна пошта",
|
||||||
|
"Google verification code": "Код підтвердження Google",
|
||||||
|
"Preferences": "Налаштування",
|
||||||
|
"Player preferences": "Налаштування програвача",
|
||||||
|
"Always loop: ": "Завжди повторювати: ",
|
||||||
|
"Autoplay: ": "Автовідтворення: ",
|
||||||
|
"Play next by default: ": "Завжди вмикати наступне відео: ",
|
||||||
|
"Autoplay next video: ": "Автовідтворення наступного відео: ",
|
||||||
|
"Listen by default: ": "Режим «тільки звук» як усталений: ",
|
||||||
|
"Proxy videos: ": "Програвати відео через проксі? ",
|
||||||
|
"Default speed: ": "Усталена швидкість відео: ",
|
||||||
|
"Preferred video quality: ": "Пріорітетна якість відео: ",
|
||||||
|
"Player volume: ": "Гучність відео: ",
|
||||||
|
"Default comments: ": "Джерело коментарів: ",
|
||||||
|
"youtube": "YouTube",
|
||||||
|
"reddit": "Reddit",
|
||||||
|
"Default captions: ": "Основна мова субтитрів: ",
|
||||||
|
"Fallback captions: ": "Запасна мова субтитрів: ",
|
||||||
|
"Show related videos: ": "Показувати схожі відео? ",
|
||||||
|
"Show annotations by default: ": "Завжди показувати анотації? ",
|
||||||
|
"Visual preferences": "Налаштування сайту",
|
||||||
|
"Player style: ": "",
|
||||||
|
"Dark mode: ": "Темне оформлення: ",
|
||||||
|
"Theme: ": "",
|
||||||
|
"dark": "",
|
||||||
|
"light": "",
|
||||||
|
"Thin mode: ": "Полегшене оформлення: ",
|
||||||
|
"Subscription preferences": "Налаштування підписок",
|
||||||
|
"Show annotations by default for subscribed channels: ": "Завжди показувати анотації у відео каналів, на які ви підписані? ",
|
||||||
|
"Redirect homepage to feed: ": "Показувати відео з каналів, на які підписані, як головну сторінку: ",
|
||||||
|
"Number of videos shown in feed: ": "Кількість відео з каналів, на які підписані, у потоці: ",
|
||||||
|
"Sort videos by: ": "Сортувати відео: ",
|
||||||
|
"published": "за датою розміщення",
|
||||||
|
"published - reverse": "за датою розміщення в зворотному порядку",
|
||||||
|
"alphabetically": "за абеткою",
|
||||||
|
"alphabetically - reverse": "за абеткою в зворотному порядку",
|
||||||
|
"channel name": "за назвою каналу",
|
||||||
|
"channel name - reverse": "за назвою каналу в зворотному порядку",
|
||||||
|
"Only show latest video from channel: ": "Показувати тільки останнє відео з каналів: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "Показувати тільки непереглянуті відео з каналів: ",
|
||||||
|
"Only show unwatched: ": "Показувати тільки непереглянуті відео: ",
|
||||||
|
"Only show notifications (if there are any): ": "Показувати лише сповіщення, якщо вони є: ",
|
||||||
|
"Enable web notifications": "Ввімкнути сповіщення в браузері",
|
||||||
|
"`x` uploaded a video": "`x` розмістив відео",
|
||||||
|
"`x` is live": "`x` у прямому ефірі",
|
||||||
|
"Data preferences": "Налаштування даних",
|
||||||
|
"Clear watch history": "Очистити історію переглядів",
|
||||||
|
"Import/export data": "Імпорт і експорт даних",
|
||||||
|
"Change password": "Змінити пароль",
|
||||||
|
"Manage subscriptions": "Керування підписками",
|
||||||
|
"Manage tokens": "Керувати токенами",
|
||||||
|
"Watch history": "Історія переглядів",
|
||||||
|
"Delete account": "Видалити обліківку",
|
||||||
|
"Administrator preferences": "Адміністраторські налаштування",
|
||||||
|
"Default homepage: ": "Усталена домашня сторінка: ",
|
||||||
|
"Feed menu: ": "Меню потоку з відео: ",
|
||||||
|
"Top enabled: ": "Увімкнути топ відео? ",
|
||||||
|
"CAPTCHA enabled: ": "Увімкнути капчу? ",
|
||||||
|
"Login enabled: ": "Увімкнути авторизацію? ",
|
||||||
|
"Registration enabled: ": "Увімкнути реєстрацію? ",
|
||||||
|
"Report statistics: ": "Повідомляти статистику? ",
|
||||||
|
"Save preferences": "Зберегти налаштування",
|
||||||
|
"Subscription manager": "Менеджер підписок",
|
||||||
|
"Token manager": "Менеджер токенів",
|
||||||
|
"Token": "Токен",
|
||||||
|
"`x` subscriptions": "`x` підписка / підписок / підписки",
|
||||||
|
"`x` tokens": "`x` токенів",
|
||||||
|
"Import/export": "Імпорт і експорт",
|
||||||
|
"unsubscribe": "відписатися",
|
||||||
|
"revoke": "скасувати",
|
||||||
|
"Subscriptions": "Підписки",
|
||||||
|
"`x` unseen notifications": "`x` непереглянуте сповіщення / непереглянутих сповіщень / непереглянутих сповіщення",
|
||||||
|
"search": "пошук",
|
||||||
|
"Log out": "Вийти",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Реалізовано Омаром Ротом за ліцензією AGPLv3.",
|
||||||
|
"Source available here.": "Програмний код доступний тут.",
|
||||||
|
"View JavaScript license information.": "Переглянути інформацію щодо ліцензії JavaScript.",
|
||||||
|
"View privacy policy.": "Переглянути політику приватності.",
|
||||||
|
"Trending": "У тренді",
|
||||||
|
"Public": "",
|
||||||
|
"Unlisted": "Немає в списку",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "Дивитися на YouTube",
|
||||||
|
"Hide annotations": "Приховати анотації",
|
||||||
|
"Show annotations": "Показати анотації",
|
||||||
|
"Genre: ": "Жанр: ",
|
||||||
|
"License: ": "Ліцензія: ",
|
||||||
|
"Family friendly? ": "Перегляд із родиною? ",
|
||||||
|
"Wilson score: ": "Рейтинг Вілсона: ",
|
||||||
|
"Engagement: ": "Залученість: ",
|
||||||
|
"Whitelisted regions: ": "Доступно у регіонах: ",
|
||||||
|
"Blacklisted regions: ": "Недоступно у регіонах: ",
|
||||||
|
"Shared `x`": "Розміщено `x`",
|
||||||
|
"`x` views": "`x` переглядів",
|
||||||
|
"Premieres in `x`": "Прем’єра через `x`",
|
||||||
|
"Premieres `x`": "Прем’єра `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Схоже, у вас відключений JavaScript. Щоб побачити коментарі, натисніть сюда, але майте на увазі, що вони можуть завантажуватися трохи довше.",
|
||||||
|
"View YouTube comments": "Переглянути коментарі з YouTube",
|
||||||
|
"View more comments on Reddit": "Переглянути більше коментарів на Reddit",
|
||||||
|
"View `x` comments": "Переглянути `x` коментар / коментарів / коментаря",
|
||||||
|
"View Reddit comments": "Переглянути коментарі з Reddit",
|
||||||
|
"Hide replies": "Сховати відповіді",
|
||||||
|
"Show replies": "Показати відповіді",
|
||||||
|
"Incorrect password": "Неправильний пароль",
|
||||||
|
"Quota exceeded, try again in a few hours": "Ліміт перевищено, спробуйте знову за декілька годин",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "Не вдається увійти. Перевірте, чи не ввімкнена двофакторна аутентифікація (за кодом чи смс).",
|
||||||
|
"Invalid TFA code": "Неправильний код двофакторної аутентифікації",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "Не вдається увійти. Це може бути через те, що у вашій обліківці не ввімкнена двофакторна аутентифікація.",
|
||||||
|
"Wrong answer": "Неправильна відповідь",
|
||||||
|
"Erroneous CAPTCHA": "Неправильна капча",
|
||||||
|
"CAPTCHA is a required field": "Необхідно пройти капчу",
|
||||||
|
"User ID is a required field": "Необхідно ввести ID користувача",
|
||||||
|
"Password is a required field": "Необхідно ввести пароль",
|
||||||
|
"Wrong username or password": "Неправильний логін чи пароль",
|
||||||
|
"Please sign in using 'Log in with Google'": "Будь ласка, натисніть «Увійти через Google»",
|
||||||
|
"Password cannot be empty": "Пароль не може бути порожнім",
|
||||||
|
"Password cannot be longer than 55 characters": "Пароль не може бути довшим за 55 знаків",
|
||||||
|
"Please log in": "Будь ласка, увійдіть",
|
||||||
|
"Invidious Private Feed for `x`": "Приватний поток відео Invidious для `x`",
|
||||||
|
"channel:`x`": "канал: `x`",
|
||||||
|
"Deleted or invalid channel": "Канал видалено або не знайдено",
|
||||||
|
"This channel does not exist.": "Такого каналу не існує.",
|
||||||
|
"Could not get channel info.": "Не вдається отримати інформацію щодо цього каналу.",
|
||||||
|
"Could not fetch comments": "Не вдається завантажити коментарі",
|
||||||
|
"View `x` replies": "Переглянути `x` відповідь / відповідей / відповіді",
|
||||||
|
"`x` ago": "`x` тому",
|
||||||
|
"Load more": "Завантажити більше",
|
||||||
|
"`x` points": "`x` очко / очок / очка",
|
||||||
|
"Could not create mix.": "Не вдається створити мікс.",
|
||||||
|
"Empty playlist": "Плейлист порожній",
|
||||||
|
"Not a playlist.": "Недійсний плейлист.",
|
||||||
|
"Playlist does not exist.": "Плейлист не існує.",
|
||||||
|
"Could not pull trending pages.": "Не вдається завантажити сторінки «у тренді».",
|
||||||
|
"Hidden field \"challenge\" is a required field": "Необхідно заповнити приховане поле «challenge»",
|
||||||
|
"Hidden field \"token\" is a required field": "Необхідно заповнити приховане поле «token»",
|
||||||
|
"Erroneous challenge": "Неправильна відповідь у «challenge»",
|
||||||
|
"Erroneous token": "Недійсний токен",
|
||||||
|
"No such user": "Недопустиме ім’я користувача",
|
||||||
|
"Token is expired, please try again": "Термін дії токена закінчився, спробуйте пізніше",
|
||||||
|
"English": "Англійська",
|
||||||
|
"English (auto-generated)": "Англійська (сгенеровано автоматично)",
|
||||||
|
"Afrikaans": "Африкаанс",
|
||||||
|
"Albanian": "Албанська",
|
||||||
|
"Amharic": "Амхарська",
|
||||||
|
"Arabic": "Арабська",
|
||||||
|
"Armenian": "Вірменська",
|
||||||
|
"Azerbaijani": "Азербайджанська",
|
||||||
|
"Bangla": "Бенгальска",
|
||||||
|
"Basque": "Баскська",
|
||||||
|
"Belarusian": "Білоруська",
|
||||||
|
"Bosnian": "Боснійська",
|
||||||
|
"Bulgarian": "Болгарська",
|
||||||
|
"Burmese": "Бірманська",
|
||||||
|
"Catalan": "Каталонська",
|
||||||
|
"Cebuano": "Себуанська",
|
||||||
|
"Chinese (Simplified)": "Китайська (спрощена)",
|
||||||
|
"Chinese (Traditional)": "Китайська (традиційна)",
|
||||||
|
"Corsican": "Корсиканська",
|
||||||
|
"Croatian": "Хорватська",
|
||||||
|
"Czech": "Чеська",
|
||||||
|
"Danish": "Данська",
|
||||||
|
"Dutch": "Нідерландська",
|
||||||
|
"Esperanto": "Есперанто",
|
||||||
|
"Estonian": "Естонська",
|
||||||
|
"Filipino": "Філіппінська",
|
||||||
|
"Finnish": "Фінська",
|
||||||
|
"French": "Французька",
|
||||||
|
"Galician": "Галісійська",
|
||||||
|
"Georgian": "Грузинська",
|
||||||
|
"German": "Німецька",
|
||||||
|
"Greek": "Грецька",
|
||||||
|
"Gujarati": "Гуджаратська",
|
||||||
|
"Haitian Creole": "Гаїтянська креольська",
|
||||||
|
"Hausa": "Хауса",
|
||||||
|
"Hawaiian": "Гавайська",
|
||||||
|
"Hebrew": "Іврит",
|
||||||
|
"Hindi": "Гінді",
|
||||||
|
"Hmong": "Хмонгська",
|
||||||
|
"Hungarian": "Угорська",
|
||||||
|
"Icelandic": "Ісландська",
|
||||||
|
"Igbo": "Ігбо",
|
||||||
|
"Indonesian": "Індонезійська",
|
||||||
|
"Irish": "Ірландська",
|
||||||
|
"Italian": "Італійська",
|
||||||
|
"Japanese": "Японська",
|
||||||
|
"Javanese": "Яванська",
|
||||||
|
"Kannada": "Каннада",
|
||||||
|
"Kazakh": "Казахська",
|
||||||
|
"Khmer": "Кхмерська",
|
||||||
|
"Korean": "Корейська",
|
||||||
|
"Kurdish": "Курдська",
|
||||||
|
"Kyrgyz": "Киргизька",
|
||||||
|
"Lao": "Лаоська",
|
||||||
|
"Latin": "Латинська",
|
||||||
|
"Latvian": "Латиська",
|
||||||
|
"Lithuanian": "Литовська",
|
||||||
|
"Luxembourgish": "Люксембурзька",
|
||||||
|
"Macedonian": "Македонська",
|
||||||
|
"Malagasy": "Малагасійська",
|
||||||
|
"Malay": "Малайська",
|
||||||
|
"Malayalam": "Малаялам",
|
||||||
|
"Maltese": "Мальтійська",
|
||||||
|
"Maori": "Маорі",
|
||||||
|
"Marathi": "Маратхі",
|
||||||
|
"Mongolian": "Монгольська",
|
||||||
|
"Nepali": "Непальська",
|
||||||
|
"Norwegian Bokmål": "Норвезька",
|
||||||
|
"Nyanja": "Ньянджа",
|
||||||
|
"Pashto": "Пушту",
|
||||||
|
"Persian": "Перська",
|
||||||
|
"Polish": "Польська",
|
||||||
|
"Portuguese": "Португальська",
|
||||||
|
"Punjabi": "Пенджабська",
|
||||||
|
"Romanian": "Румунська",
|
||||||
|
"Russian": "Російська",
|
||||||
|
"Samoan": "Самоанська",
|
||||||
|
"Scottish Gaelic": "Шотландська ґельська",
|
||||||
|
"Serbian": "Сербська",
|
||||||
|
"Shona": "Шона",
|
||||||
|
"Sindhi": "Сіндгі",
|
||||||
|
"Sinhala": "Сингальська",
|
||||||
|
"Slovak": "Словацька",
|
||||||
|
"Slovenian": "Словенська",
|
||||||
|
"Somali": "Сомалійська",
|
||||||
|
"Southern Sotho": "Сесото (південна сото)",
|
||||||
|
"Spanish": "Іспанська",
|
||||||
|
"Spanish (Latin America)": "Испанська (Латинська Америка)",
|
||||||
|
"Sundanese": "Сунданська",
|
||||||
|
"Swahili": "Суахілі",
|
||||||
|
"Swedish": "Шведська",
|
||||||
|
"Tajik": "Таджицька",
|
||||||
|
"Tamil": "Тамільська",
|
||||||
|
"Telugu": "Телугу",
|
||||||
|
"Thai": "Тайська",
|
||||||
|
"Turkish": "Турецька",
|
||||||
|
"Ukrainian": "Українська",
|
||||||
|
"Urdu": "Урду",
|
||||||
|
"Uzbek": "Узбецька",
|
||||||
|
"Vietnamese": "В’єтнамська",
|
||||||
|
"Welsh": "Валлійська",
|
||||||
|
"Western Frisian": "Західнофризька",
|
||||||
|
"Xhosa": "Коса",
|
||||||
|
"Yiddish": "Їдиш",
|
||||||
|
"Yoruba": "Йоруба",
|
||||||
|
"Zulu": "Зулу",
|
||||||
|
"`x` years": "`x` років",
|
||||||
|
"`x` months": "`x` місяців",
|
||||||
|
"`x` weeks": "`x` тижнів",
|
||||||
|
"`x` days": "`x` днів",
|
||||||
|
"`x` hours": "`x` годин",
|
||||||
|
"`x` minutes": "`x` хвилин",
|
||||||
|
"`x` seconds": "`x` секунд",
|
||||||
|
"Fallback comments: ": "Резервні коментарі: ",
|
||||||
|
"Popular": "Популярне",
|
||||||
|
"Top": "Топ",
|
||||||
|
"About": "Про сайт",
|
||||||
|
"Rating: ": "Рейтинг: ",
|
||||||
|
"Language: ": "Мова: ",
|
||||||
|
"View as playlist": "Дивитися як плейлист",
|
||||||
|
"Default": "Усталено",
|
||||||
|
"Music": "Музика",
|
||||||
|
"Gaming": "Ігри",
|
||||||
|
"News": "Новини",
|
||||||
|
"Movies": "Фільми",
|
||||||
|
"Download": "Завантажити",
|
||||||
|
"Download as: ": "Завантажити як: ",
|
||||||
|
"%A %B %-d, %Y": "%-d %B %Y, %A",
|
||||||
|
"(edited)": "(змінено)",
|
||||||
|
"YouTube comment permalink": "Пряме посилання на коментар в YouTube",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "❤ цьому від каналу `x`",
|
||||||
|
"Audio mode": "Аудіорежим",
|
||||||
|
"Video mode": "Відеорежим",
|
||||||
|
"Videos": "Відео",
|
||||||
|
"Playlists": "Плейлисти",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": "Поточна версія: "
|
||||||
|
}
|
||||||
336
locales/zh-CN.json
Normal file
336
locales/zh-CN.json
Normal file
@@ -0,0 +1,336 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": "`x` 订阅者",
|
||||||
|
"`x` videos": "`x` 视频",
|
||||||
|
"`x` playlists": "",
|
||||||
|
"LIVE": "直播",
|
||||||
|
"Shared `x` ago": "`x` 前分享",
|
||||||
|
"Unsubscribe": "取消订阅",
|
||||||
|
"Subscribe": "订阅",
|
||||||
|
"View channel on YouTube": "在 YouTube 查看频道",
|
||||||
|
"View playlist on YouTube": "在 YouTube 查看播放列表",
|
||||||
|
"newest": "最新",
|
||||||
|
"oldest": "最老",
|
||||||
|
"popular": "时下流行",
|
||||||
|
"last": "last",
|
||||||
|
"Next page": "下一页",
|
||||||
|
"Previous page": "上一页",
|
||||||
|
"Clear watch history?": "清除观看历史?",
|
||||||
|
"New password": "新密码",
|
||||||
|
"New passwords must match": "新密码必须匹配",
|
||||||
|
"Cannot change password for Google accounts": "无法为 Google 账户更改密码",
|
||||||
|
"Authorize token?": "授权令牌?",
|
||||||
|
"Authorize token for `x`?": "`x` 的授权令牌?",
|
||||||
|
"Yes": "是",
|
||||||
|
"No": "否",
|
||||||
|
"Import and Export Data": "导入与导出数据",
|
||||||
|
"Import": "导入",
|
||||||
|
"Import Invidious data": "导入 Invidious 数据",
|
||||||
|
"Import YouTube subscriptions": "导入 YouTube 订阅",
|
||||||
|
"Import FreeTube subscriptions (.db)": "导入 FreeTube 订阅 (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "导入 NewPipe 订阅 (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "导入 NewPipe 数据 (.zip)",
|
||||||
|
"Export": "导出",
|
||||||
|
"Export subscriptions as OPML": "导出订阅到 OPML 格式",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "导出订阅到 OPML 格式(用于 NewPipe 及 FreeTube)",
|
||||||
|
"Export data as JSON": "导出数据为 JSON 格式",
|
||||||
|
"Delete account?": "删除账户?",
|
||||||
|
"History": "历史",
|
||||||
|
"An alternative front-end to YouTube": "另一个 YouTube 前端",
|
||||||
|
"JavaScript license information": "JavaScript 授权信息",
|
||||||
|
"source": "source",
|
||||||
|
"Log in": "登录",
|
||||||
|
"Log in/register": "登录/注册",
|
||||||
|
"Log in with Google": "使用 Google 账户登录",
|
||||||
|
"User ID": "用户 ID",
|
||||||
|
"Password": "密码",
|
||||||
|
"Time (h:mm:ss):": "时间 (h:mm:ss):",
|
||||||
|
"Text CAPTCHA": "文本验证码",
|
||||||
|
"Image CAPTCHA": "图片验证码",
|
||||||
|
"Sign In": "登录",
|
||||||
|
"Register": "注册",
|
||||||
|
"E-mail": "E-mail",
|
||||||
|
"Google verification code": "Google 验证代码",
|
||||||
|
"Preferences": "偏好设置",
|
||||||
|
"Player preferences": "播放器偏好设置",
|
||||||
|
"Always loop: ": "循环:",
|
||||||
|
"Autoplay: ": "自动播放:",
|
||||||
|
"Play next by default: ": "默认自动播放下一个视频:",
|
||||||
|
"Autoplay next video: ": "自动播放下一个视频:",
|
||||||
|
"Listen by default: ": "默认只聆听声音:",
|
||||||
|
"Proxy videos: ": "代理视频?",
|
||||||
|
"Default speed: ": "默认速度:",
|
||||||
|
"Preferred video quality: ": "视频质量偏好:",
|
||||||
|
"Player volume: ": "播放器音量:",
|
||||||
|
"Default comments: ": "默认评论源:",
|
||||||
|
"youtube": "YouTube",
|
||||||
|
"reddit": "Reddit",
|
||||||
|
"Default captions: ": "默认字幕语言:",
|
||||||
|
"Fallback captions: ": "后备字幕语言:",
|
||||||
|
"Show related videos: ": "显示相关视频?",
|
||||||
|
"Show annotations by default: ": "默认显示视频注释?",
|
||||||
|
"Visual preferences": "视觉选项",
|
||||||
|
"Player style: ": "",
|
||||||
|
"Dark mode: ": "暗色模式:",
|
||||||
|
"Theme: ": "",
|
||||||
|
"dark": "",
|
||||||
|
"light": "",
|
||||||
|
"Thin mode: ": "窄页模式:",
|
||||||
|
"Subscription preferences": "订阅设置",
|
||||||
|
"Show annotations by default for subscribed channels: ": "在订阅频道的视频默认显示注释?",
|
||||||
|
"Redirect homepage to feed: ": "跳转主页到 feed: ",
|
||||||
|
"Number of videos shown in feed: ": "Feed 中显示的视频数量:",
|
||||||
|
"Sort videos by: ": "视频排序方式:",
|
||||||
|
"published": "发布时间",
|
||||||
|
"published - reverse": "发布时间(反向)",
|
||||||
|
"alphabetically": "字母序",
|
||||||
|
"alphabetically - reverse": "字母序(反向)",
|
||||||
|
"channel name": "频道名称",
|
||||||
|
"channel name - reverse": "频道名称(反向)",
|
||||||
|
"Only show latest video from channel: ": "只显示订阅频道的最新一条视频:",
|
||||||
|
"Only show latest unwatched video from channel: ": "只显示订阅频道的最新未看过视频:",
|
||||||
|
"Only show unwatched: ": "只显示未看过的视频:",
|
||||||
|
"Only show notifications (if there are any): ": "只显示通知(如有):",
|
||||||
|
"Enable web notifications": "启用浏览器通知",
|
||||||
|
"`x` uploaded a video": "`x` 上传了视频",
|
||||||
|
"`x` is live": "`x` 正在直播",
|
||||||
|
"Data preferences": "数据选项",
|
||||||
|
"Clear watch history": "清除观看历史",
|
||||||
|
"Import/export data": "导入/导出数据",
|
||||||
|
"Change password": "更改密码",
|
||||||
|
"Manage subscriptions": "管理订阅",
|
||||||
|
"Manage tokens": "管理令牌",
|
||||||
|
"Watch history": "观看历史",
|
||||||
|
"Delete account": "删除账户",
|
||||||
|
"Administrator preferences": "管理员选项",
|
||||||
|
"Default homepage: ": "默认主页:",
|
||||||
|
"Feed menu: ": "Feed 菜单:",
|
||||||
|
"Top enabled: ": "启用“热门视频”页?",
|
||||||
|
"CAPTCHA enabled: ": "启用验证码?",
|
||||||
|
"Login enabled: ": "启用登录?",
|
||||||
|
"Registration enabled: ": "启用注册?",
|
||||||
|
"Report statistics: ": "报告统计信息?",
|
||||||
|
"Save preferences": "保存选项",
|
||||||
|
"Subscription manager": "订阅管理器",
|
||||||
|
"Token manager": "令牌管理器",
|
||||||
|
"Token": "令牌",
|
||||||
|
"`x` subscriptions": "`x` 个订阅",
|
||||||
|
"`x` tokens": "`x` 个令牌",
|
||||||
|
"Import/export": "导入/导出",
|
||||||
|
"unsubscribe": "取消订阅",
|
||||||
|
"revoke": "吊销",
|
||||||
|
"Subscriptions": "订阅",
|
||||||
|
"`x` unseen notifications": "`x` 条未读通知",
|
||||||
|
"search": "搜索",
|
||||||
|
"Log out": "登出",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "由 Omar Roth 开发,以 AGPLv3 授权。",
|
||||||
|
"Source available here.": "源码可在此查看。",
|
||||||
|
"View JavaScript license information.": "查看 JavaScript 协议信息。",
|
||||||
|
"View privacy policy.": "查看隐私政策。",
|
||||||
|
"Trending": "时下流行",
|
||||||
|
"Public": "",
|
||||||
|
"Unlisted": "不公开",
|
||||||
|
"Private": "",
|
||||||
|
"View all playlists": "",
|
||||||
|
"Updated `x` ago": "",
|
||||||
|
"Delete playlist `x`?": "",
|
||||||
|
"Delete playlist": "",
|
||||||
|
"Create playlist": "",
|
||||||
|
"Title": "",
|
||||||
|
"Playlist privacy": "",
|
||||||
|
"Editing playlist `x`": "",
|
||||||
|
"Watch on YouTube": "在 YouTube 观看",
|
||||||
|
"Hide annotations": "隐藏注释",
|
||||||
|
"Show annotations": "显示注释",
|
||||||
|
"Genre: ": "风格:",
|
||||||
|
"License: ": "协议:",
|
||||||
|
"Family friendly? ": "家庭友好?",
|
||||||
|
"Wilson score: ": "威尔逊得分:",
|
||||||
|
"Engagement: ": "参与度:",
|
||||||
|
"Whitelisted regions: ": "白名单区域:",
|
||||||
|
"Blacklisted regions: ": "黑名单区域:",
|
||||||
|
"Shared `x`": "`x`发布",
|
||||||
|
"`x` views": "`x` 播放",
|
||||||
|
"Premieres in `x`": "首映于 `x` 后",
|
||||||
|
"Premieres `x`": "首映于 `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "你好!看起来你关闭了 JavaScript。点击这里阅读评论。注意它们加载的时间可能会稍长。",
|
||||||
|
"View YouTube comments": "查看 YouTube 评论",
|
||||||
|
"View more comments on Reddit": "在 Reddit 查看更多评论",
|
||||||
|
"View `x` comments": "查看 `x` 条评论",
|
||||||
|
"View Reddit comments": "查看 Reddit 评论",
|
||||||
|
"Hide replies": "隐藏回复",
|
||||||
|
"Show replies": "显示回复",
|
||||||
|
"Incorrect password": "密码错误",
|
||||||
|
"Quota exceeded, try again in a few hours": "已超出限额,请于几小时后重试",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "无法登录。请确认你的短信或验证器的二步验证已打开。",
|
||||||
|
"Invalid TFA code": "无效的二步验证码",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "登录失败。可能是因为二步验证未打开。",
|
||||||
|
"Wrong answer": "错误的回复",
|
||||||
|
"Erroneous CAPTCHA": "验证码错误",
|
||||||
|
"CAPTCHA is a required field": "验证码必填",
|
||||||
|
"User ID is a required field": "用户名必填",
|
||||||
|
"Password is a required field": "密码必填",
|
||||||
|
"Wrong username or password": "用户名或密码错误",
|
||||||
|
"Please sign in using 'Log in with Google'": "请通过谷歌账户登录",
|
||||||
|
"Password cannot be empty": "密码不能为空",
|
||||||
|
"Password cannot be longer than 55 characters": "密码长度不能大于 55",
|
||||||
|
"Please log in": "请登录",
|
||||||
|
"Invidious Private Feed for `x`": "`x` 的 Invidious 私人 feed",
|
||||||
|
"channel:`x`": "频道:`x`",
|
||||||
|
"Deleted or invalid channel": "已删除或无效频道",
|
||||||
|
"This channel does not exist.": "频道不存在。",
|
||||||
|
"Could not get channel info.": "无法获取频道信息。",
|
||||||
|
"Could not fetch comments": "无法获取评论",
|
||||||
|
"View `x` replies": "查看 `x` 条回复",
|
||||||
|
"`x` ago": "`x` 前",
|
||||||
|
"Load more": "加载更多",
|
||||||
|
"`x` points": "`x` 分",
|
||||||
|
"Could not create mix.": "无法创建合集。",
|
||||||
|
"Empty playlist": "空播放列表",
|
||||||
|
"Not a playlist.": "非播放列表。",
|
||||||
|
"Playlist does not exist.": "播放列表不存在。",
|
||||||
|
"Could not pull trending pages.": "无法获取“时下流行”页面。",
|
||||||
|
"Hidden field \"challenge\" is a required field": "隐藏表单项 \"challenge\" 为必填",
|
||||||
|
"Hidden field \"token\" is a required field": "隐藏表单项 \"token\" 为必填",
|
||||||
|
"Erroneous challenge": "错误的验证回复(challenge)",
|
||||||
|
"Erroneous token": "错误的令牌",
|
||||||
|
"No such user": "用户不存在",
|
||||||
|
"Token is expired, please try again": "令牌过期,请重试",
|
||||||
|
"English": "英语",
|
||||||
|
"English (auto-generated)": "英语(自动生成)",
|
||||||
|
"Afrikaans": "南非荷兰语",
|
||||||
|
"Albanian": "阿尔巴尼亚语",
|
||||||
|
"Amharic": "阿姆哈拉语",
|
||||||
|
"Arabic": "阿拉伯语",
|
||||||
|
"Armenian": "亚美尼亚语",
|
||||||
|
"Azerbaijani": "阿塞拜疆语",
|
||||||
|
"Bangla": "孟加拉语",
|
||||||
|
"Basque": "巴斯克语",
|
||||||
|
"Belarusian": "白俄罗斯语",
|
||||||
|
"Bosnian": "波黑语",
|
||||||
|
"Bulgarian": "保加利亚语",
|
||||||
|
"Burmese": "缅甸语",
|
||||||
|
"Catalan": "加泰罗尼亚语",
|
||||||
|
"Cebuano": "宿雾语",
|
||||||
|
"Chinese (Simplified)": "中文(简体)",
|
||||||
|
"Chinese (Traditional)": "中文(繁体)",
|
||||||
|
"Corsican": "科西嘉语",
|
||||||
|
"Croatian": "克罗地亚语",
|
||||||
|
"Czech": "捷克语",
|
||||||
|
"Danish": "丹麦语",
|
||||||
|
"Dutch": "荷兰语",
|
||||||
|
"Esperanto": "世界语",
|
||||||
|
"Estonian": "爱沙尼亚语",
|
||||||
|
"Filipino": "菲律宾语",
|
||||||
|
"Finnish": "芬兰语",
|
||||||
|
"French": "法语",
|
||||||
|
"Galician": "加利西亚语",
|
||||||
|
"Georgian": "格鲁吉亚语",
|
||||||
|
"German": "德语",
|
||||||
|
"Greek": "希腊语",
|
||||||
|
"Gujarati": "古吉拉特语",
|
||||||
|
"Haitian Creole": "海地克里奥尔语",
|
||||||
|
"Hausa": "豪萨语",
|
||||||
|
"Hawaiian": "夏威夷语",
|
||||||
|
"Hebrew": "希伯来语",
|
||||||
|
"Hindi": "印地语",
|
||||||
|
"Hmong": "苗语",
|
||||||
|
"Hungarian": "匈牙利语",
|
||||||
|
"Icelandic": "冰岛语",
|
||||||
|
"Igbo": "伊博语",
|
||||||
|
"Indonesian": "印度尼西亚语",
|
||||||
|
"Irish": "爱尔兰语",
|
||||||
|
"Italian": "意大利语",
|
||||||
|
"Japanese": "日语",
|
||||||
|
"Javanese": "爪哇语",
|
||||||
|
"Kannada": "卡纳达语",
|
||||||
|
"Kazakh": "哈萨克语",
|
||||||
|
"Khmer": "高棉语",
|
||||||
|
"Korean": "韩语",
|
||||||
|
"Kurdish": "库尔德语",
|
||||||
|
"Kyrgyz": "柯尔克孜语",
|
||||||
|
"Lao": "老挝语",
|
||||||
|
"Latin": "拉丁语",
|
||||||
|
"Latvian": "拉脱维亚语",
|
||||||
|
"Lithuanian": "立陶宛语",
|
||||||
|
"Luxembourgish": "卢森堡语",
|
||||||
|
"Macedonian": "马其顿语",
|
||||||
|
"Malagasy": "马尔加什语",
|
||||||
|
"Malay": "马来语",
|
||||||
|
"Malayalam": "马拉雅拉姆语",
|
||||||
|
"Maltese": "马耳他语",
|
||||||
|
"Maori": "毛利语",
|
||||||
|
"Marathi": "马拉语",
|
||||||
|
"Mongolian": "蒙古语",
|
||||||
|
"Nepali": "尼泊尔语",
|
||||||
|
"Norwegian Bokmål": "书面挪威语",
|
||||||
|
"Nyanja": "尼昂加语",
|
||||||
|
"Pashto": "普什图语",
|
||||||
|
"Persian": "波斯语",
|
||||||
|
"Polish": "抛光",
|
||||||
|
"Portuguese": "葡萄牙语",
|
||||||
|
"Punjabi": "旁遮普语",
|
||||||
|
"Romanian": "罗马尼亚语",
|
||||||
|
"Russian": "俄语",
|
||||||
|
"Samoan": "萨摩亚语",
|
||||||
|
"Scottish Gaelic": "苏格兰盖尔语",
|
||||||
|
"Serbian": "塞尔维亚语",
|
||||||
|
"Shona": "绍纳语",
|
||||||
|
"Sindhi": "信德语",
|
||||||
|
"Sinhala": "僧伽罗语",
|
||||||
|
"Slovak": "斯洛伐克语",
|
||||||
|
"Slovenian": "斯洛文尼亚语",
|
||||||
|
"Somali": "索马里语",
|
||||||
|
"Southern Sotho": "南索托语",
|
||||||
|
"Spanish": "西班牙语",
|
||||||
|
"Spanish (Latin America)": "西班牙语(拉丁美洲)",
|
||||||
|
"Sundanese": "巽丹语",
|
||||||
|
"Swahili": "斯瓦希里语",
|
||||||
|
"Swedish": "瑞典语",
|
||||||
|
"Tajik": "塔吉克语",
|
||||||
|
"Tamil": "泰米尔语",
|
||||||
|
"Telugu": "泰卢固语",
|
||||||
|
"Thai": "泰语",
|
||||||
|
"Turkish": "土耳其语",
|
||||||
|
"Ukrainian": "乌克兰语",
|
||||||
|
"Urdu": "乌尔都语",
|
||||||
|
"Uzbek": "乌兹别克",
|
||||||
|
"Vietnamese": "越南语",
|
||||||
|
"Welsh": "威尔士语",
|
||||||
|
"Western Frisian": "西弗里西亚语",
|
||||||
|
"Xhosa": "科萨语",
|
||||||
|
"Yiddish": "意第绪语",
|
||||||
|
"Yoruba": "约鲁巴语",
|
||||||
|
"Zulu": "祖鲁语",
|
||||||
|
"`x` years": "`x` 年",
|
||||||
|
"`x` months": "`x` 月",
|
||||||
|
"`x` weeks": "`x` 周",
|
||||||
|
"`x` days": "`x` 天",
|
||||||
|
"`x` hours": "`x` 小时",
|
||||||
|
"`x` minutes": "`x` 分钟",
|
||||||
|
"`x` seconds": "`x` 秒",
|
||||||
|
"Fallback comments: ": "后备评论:",
|
||||||
|
"Popular": "热门频道",
|
||||||
|
"Top": "热门视频",
|
||||||
|
"About": "关于",
|
||||||
|
"Rating: ": "评分:",
|
||||||
|
"Language: ": "语言:",
|
||||||
|
"View as playlist": "作为播放列表查看",
|
||||||
|
"Default": "默认",
|
||||||
|
"Music": "音乐",
|
||||||
|
"Gaming": "游戏",
|
||||||
|
"News": "新闻",
|
||||||
|
"Movies": "电影",
|
||||||
|
"Download": "下载",
|
||||||
|
"Download as: ": "下载为:",
|
||||||
|
"%A %B %-d, %Y": "%Y年%-m月%-d日 %a",
|
||||||
|
"(edited)": "(已编辑)",
|
||||||
|
"YouTube comment permalink": "YouTube 评论永久链接",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "`x` 为此加 ❤",
|
||||||
|
"Audio mode": "音频模式",
|
||||||
|
"Video mode": "视频模式",
|
||||||
|
"Videos": "视频",
|
||||||
|
"Playlists": "播放列表",
|
||||||
|
"Community": "",
|
||||||
|
"Current version: ": "当前版本:"
|
||||||
|
}
|
||||||
381
locales/zh-TW.json
Normal file
381
locales/zh-TW.json
Normal file
@@ -0,0 +1,381 @@
|
|||||||
|
{
|
||||||
|
"`x` subscribers": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個訂閱者",
|
||||||
|
"": "`x` 個訂閱者"
|
||||||
|
},
|
||||||
|
"`x` videos": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 部影片",
|
||||||
|
"": "`x` 部影片"
|
||||||
|
},
|
||||||
|
"`x` playlists": "",
|
||||||
|
"LIVE": "直播",
|
||||||
|
"Shared `x` ago": "`x` 前分享",
|
||||||
|
"Unsubscribe": "取消訂閱",
|
||||||
|
"Subscribe": "訂閱",
|
||||||
|
"View channel on YouTube": "在 YouTube 上檢視頻道",
|
||||||
|
"View playlist on YouTube": "在 YouTube 上檢視播放清單",
|
||||||
|
"newest": "最新",
|
||||||
|
"oldest": "最舊",
|
||||||
|
"popular": "流行",
|
||||||
|
"last": "上一個",
|
||||||
|
"Next page": "下一頁",
|
||||||
|
"Previous page": "上一頁",
|
||||||
|
"Clear watch history?": "清除觀看歷史?",
|
||||||
|
"New password": "新密碼",
|
||||||
|
"New passwords must match": "新密碼必須符合",
|
||||||
|
"Cannot change password for Google accounts": "無法變更 Google 帳號的密碼",
|
||||||
|
"Authorize token?": "授權 token?",
|
||||||
|
"Authorize token for `x`?": "`x` 的授權 token?",
|
||||||
|
"Yes": "是",
|
||||||
|
"No": "否",
|
||||||
|
"Import and Export Data": "匯入與匯出資料",
|
||||||
|
"Import": "匯入",
|
||||||
|
"Import Invidious data": "匯入 Invidious 資料",
|
||||||
|
"Import YouTube subscriptions": "匯入 YouTube 訂閱",
|
||||||
|
"Import FreeTube subscriptions (.db)": "匯入 FreeTube 訂閱 (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "匯入 NewPipe 訂閱 (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "匯入 NewPipe 資料 (.zip)",
|
||||||
|
"Export": "匯出",
|
||||||
|
"Export subscriptions as OPML": "將訂閱匯出為 OPML",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "將訂閱匯出為 OPML(供 NewPipe 與 FreeTube 使用)",
|
||||||
|
"Export data as JSON": "將 JSON 匯出為 JSON",
|
||||||
|
"Delete account?": "刪除帳號?",
|
||||||
|
"History": "歷史",
|
||||||
|
"An alternative front-end to YouTube": "一個 YouTube 的替代前端",
|
||||||
|
"JavaScript license information": "JavaScript 授權條款資訊",
|
||||||
|
"source": "來源",
|
||||||
|
"Log in": "登入",
|
||||||
|
"Log in/register": "登入/註冊",
|
||||||
|
"Log in with Google": "使用 Google 登入",
|
||||||
|
"User ID": "使用者 ID",
|
||||||
|
"Password": "密碼",
|
||||||
|
"Time (h:mm:ss):": "時間 (h:mm:ss):",
|
||||||
|
"Text CAPTCHA": "文字 CAPTCHA",
|
||||||
|
"Image CAPTCHA": "圖片 CAPTCHA",
|
||||||
|
"Sign In": "登入",
|
||||||
|
"Register": "註冊",
|
||||||
|
"E-mail": "電子郵件",
|
||||||
|
"Google verification code": "Google 驗證碼",
|
||||||
|
"Preferences": "偏好設定",
|
||||||
|
"Player preferences": "播放器偏好設定",
|
||||||
|
"Always loop: ": "總是循環播放:",
|
||||||
|
"Autoplay: ": "自動播放:",
|
||||||
|
"Play next by default: ": "預設播放下一部:",
|
||||||
|
"Autoplay next video: ": "自動播放下一部影片:",
|
||||||
|
"Listen by default: ": "預設聆聽:",
|
||||||
|
"Proxy videos: ": "代理影片:",
|
||||||
|
"Default speed: ": "預設速度:",
|
||||||
|
"Preferred video quality: ": "偏好的影片畫質:",
|
||||||
|
"Player volume: ": "播放器音量:",
|
||||||
|
"Default comments: ": "預設留言:",
|
||||||
|
"youtube": "youtube",
|
||||||
|
"reddit": "reddit",
|
||||||
|
"Default captions: ": "預設字幕:",
|
||||||
|
"Fallback captions: ": "汰退字幕:",
|
||||||
|
"Show related videos: ": "顯示相關的影片:",
|
||||||
|
"Show annotations by default: ": "預設顯示註釋:",
|
||||||
|
"Visual preferences": "視覺偏好設定",
|
||||||
|
"Player style: ": "播放器樣式",
|
||||||
|
"Dark mode: ": "深色模式:",
|
||||||
|
"Theme: ": "佈景主題",
|
||||||
|
"dark": "深色",
|
||||||
|
"light": "淺色",
|
||||||
|
"Thin mode: ": "精簡模式:",
|
||||||
|
"Subscription preferences": "訂閱偏好設定",
|
||||||
|
"Show annotations by default for subscribed channels: ": "預設為已訂閱的頻道顯示註釋?",
|
||||||
|
"Redirect homepage to feed: ": "重新導向首頁至 feed:",
|
||||||
|
"Number of videos shown in feed: ": "顯示在 feed 中的影片數量:",
|
||||||
|
"Sort videos by: ": "以此種方式排序影片:",
|
||||||
|
"published": "已發佈",
|
||||||
|
"published - reverse": "已發佈 - 反向",
|
||||||
|
"alphabetically": "字母",
|
||||||
|
"alphabetically - reverse": "字母 - 反向",
|
||||||
|
"channel name": "頻道名稱",
|
||||||
|
"channel name - reverse": "頻道名稱 - 反向",
|
||||||
|
"Only show latest video from channel: ": "僅顯示從頻道而來的最新影片:",
|
||||||
|
"Only show latest unwatched video from channel: ": "僅顯示從頻道而來的未觀看影片:",
|
||||||
|
"Only show unwatched: ": "僅顯示未觀看的:",
|
||||||
|
"Only show notifications (if there are any): ": "僅顯示通知(如果有的話):",
|
||||||
|
"Enable web notifications": "啟用網路通知",
|
||||||
|
"`x` uploaded a video": "`x` 上傳了一部影片",
|
||||||
|
"`x` is live": "`x` 正在直播",
|
||||||
|
"Data preferences": "資料偏好設定",
|
||||||
|
"Clear watch history": "清除觀看歷史",
|
||||||
|
"Import/export data": "匯入/匯出資料",
|
||||||
|
"Change password": "變更密碼",
|
||||||
|
"Manage subscriptions": "管理訂閱",
|
||||||
|
"Manage tokens": "管理 tokens",
|
||||||
|
"Watch history": "觀看歷史",
|
||||||
|
"Delete account": "刪除帳號",
|
||||||
|
"Administrator preferences": "管理員偏好設定",
|
||||||
|
"Default homepage: ": "預設首頁:",
|
||||||
|
"Feed menu: ": "Feed 選單:",
|
||||||
|
"Top enabled: ": "頂部啟用:",
|
||||||
|
"CAPTCHA enabled: ": "CAPTCHA 啟用:",
|
||||||
|
"Login enabled: ": "啟用登入?",
|
||||||
|
"Registration enabled: ": "啟用註冊?",
|
||||||
|
"Report statistics: ": "回報統計?",
|
||||||
|
"Save preferences": "儲存偏好設定",
|
||||||
|
"Subscription manager": "訂閱管理員",
|
||||||
|
"Token manager": "Token 管理員",
|
||||||
|
"Token": "Token",
|
||||||
|
"`x` subscriptions": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個訂閱",
|
||||||
|
"": "`x` 個訂閱"
|
||||||
|
},
|
||||||
|
"`x` tokens": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` token",
|
||||||
|
"": "`x` tokens"
|
||||||
|
},
|
||||||
|
"Import/export": "匯入/匯出",
|
||||||
|
"unsubscribe": "取消訂閱",
|
||||||
|
"revoke": "撤銷",
|
||||||
|
"Subscriptions": "訂閱",
|
||||||
|
"`x` unseen notifications": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 個未讀的通知",
|
||||||
|
"": "`x` 個未讀的通知"
|
||||||
|
},
|
||||||
|
"search": "搜尋",
|
||||||
|
"Log out": "登出",
|
||||||
|
"Released under the AGPLv3 by Omar Roth.": "Omar Roth 以 AGPLv3 釋出。",
|
||||||
|
"Source available here.": "原始碼在此提供。",
|
||||||
|
"View JavaScript license information.": "檢視 JavaScript 授權條款資訊。",
|
||||||
|
"View privacy policy.": "檢視隱私權政策。",
|
||||||
|
"Trending": "趨勢",
|
||||||
|
"Public": "公開",
|
||||||
|
"Unlisted": "未列出",
|
||||||
|
"Private": "私人",
|
||||||
|
"View all playlists": "檢視所有播放清單",
|
||||||
|
"Updated `x` ago": "更新於 `x` 之前",
|
||||||
|
"Delete playlist `x`?": "刪除播放清單",
|
||||||
|
"Delete playlist": "刪除播放清單",
|
||||||
|
"Create playlist": "建立播放清單",
|
||||||
|
"Title": "標題",
|
||||||
|
"Playlist privacy": "播放清單隱私",
|
||||||
|
"Editing playlist `x`": "已編輯播放清單 `x`",
|
||||||
|
"Watch on YouTube": "在 YouTube 上觀看",
|
||||||
|
"Hide annotations": "隱藏註釋",
|
||||||
|
"Show annotations": "顯示註釋",
|
||||||
|
"Genre: ": "風格:",
|
||||||
|
"License: ": "授權條款:",
|
||||||
|
"Family friendly? ": "家庭友好?",
|
||||||
|
"Wilson score: ": "威爾遜分數:",
|
||||||
|
"Engagement: ": "參與度:",
|
||||||
|
"Whitelisted regions: ": "白名單區域:",
|
||||||
|
"Blacklisted regions: ": "黑名單區域:",
|
||||||
|
"Shared `x`": "`x` 發佈",
|
||||||
|
"`x` views": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 次檢視",
|
||||||
|
"": "`x` 次檢視"
|
||||||
|
},
|
||||||
|
"Premieres in `x`": "首映於 `x`",
|
||||||
|
"Premieres `x`": "首映於 `x`",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "嗨!看來您將 JavaScript 關閉了。點擊這裡以檢視留言,請注意,它們可能需要比較長的時間載入。",
|
||||||
|
"View YouTube comments": "檢視 YouTube 留言",
|
||||||
|
"View more comments on Reddit": "在 Reddit 上檢視更多留言",
|
||||||
|
"View `x` comments": "檢視 `x` 則留言",
|
||||||
|
"View Reddit comments": "檢視 Reddit 留言",
|
||||||
|
"Hide replies": "隱藏回覆",
|
||||||
|
"Show replies": "顯示回覆",
|
||||||
|
"Incorrect password": "不正確的密碼",
|
||||||
|
"Quota exceeded, try again in a few hours": "超過限額,請在幾個小時後再試一次",
|
||||||
|
"Unable to log in, make sure two-factor authentication (Authenticator or SMS) is turned on.": "無法登入,請確定雙因素驗證(驗證器或簡訊)已開啟。",
|
||||||
|
"Invalid TFA code": "無效的 TFA 代碼",
|
||||||
|
"Login failed. This may be because two-factor authentication is not turned on for your account.": "登入失敗。這可能是因為您的帳號未開啟雙因素驗證的關係。",
|
||||||
|
"Wrong answer": "錯誤的答案",
|
||||||
|
"Erroneous CAPTCHA": "錯誤的 CAPTCHA",
|
||||||
|
"CAPTCHA is a required field": "CAPTCHA 為必填欄位",
|
||||||
|
"User ID is a required field": "使用者 ID 為必填欄位",
|
||||||
|
"Password is a required field": "密碼為必填欄位",
|
||||||
|
"Wrong username or password": "錯誤的使用者名稱或密碼",
|
||||||
|
"Please sign in using 'Log in with Google'": "請使用「以 Google 登入」來登入",
|
||||||
|
"Password cannot be empty": "密碼不能為空",
|
||||||
|
"Password cannot be longer than 55 characters": "密碼不能長於55個字元",
|
||||||
|
"Please log in": "請登入",
|
||||||
|
"Invidious Private Feed for `x`": "`x` 的 Invidious 私密 feed",
|
||||||
|
"channel:`x`": "頻道:`x`",
|
||||||
|
"Deleted or invalid channel": "已刪除或無效的頻道",
|
||||||
|
"This channel does not exist.": "此頻道不存在。",
|
||||||
|
"Could not get channel info.": "無法取得頻道資訊。",
|
||||||
|
"Could not fetch comments": "無法擷取留言",
|
||||||
|
"View `x` replies": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "檢視 `x` 則回覆",
|
||||||
|
"": "檢視 `x` 則回覆"
|
||||||
|
},
|
||||||
|
"`x` ago": "`x` 以前",
|
||||||
|
"Load more": "載入更多",
|
||||||
|
"`x` points": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 點",
|
||||||
|
"": "`x` 點"
|
||||||
|
},
|
||||||
|
"Could not create mix.": "無法建立混合。",
|
||||||
|
"Empty playlist": "空的播放清單",
|
||||||
|
"Not a playlist.": "不是播放清單。",
|
||||||
|
"Playlist does not exist.": "播放清單不存在。",
|
||||||
|
"Could not pull trending pages.": "無法拉取趨勢頁面。",
|
||||||
|
"Hidden field \"challenge\" is a required field": "隱藏的欄位 \"challenge\" 是必填欄位",
|
||||||
|
"Hidden field \"token\" is a required field": "隱藏的欄位 \"token\" 是必填欄位",
|
||||||
|
"Erroneous challenge": "錯誤的 challenge",
|
||||||
|
"Erroneous token": "錯誤的 token",
|
||||||
|
"No such user": "無此使用者",
|
||||||
|
"Token is expired, please try again": "Token 已過期,請再試一次",
|
||||||
|
"English": "英文",
|
||||||
|
"English (auto-generated)": "英文(自動生成)",
|
||||||
|
"Afrikaans": "南非語",
|
||||||
|
"Albanian": "阿爾巴尼亞語",
|
||||||
|
"Amharic": "阿姆哈拉語",
|
||||||
|
"Arabic": "阿拉伯語",
|
||||||
|
"Armenian": "亞美尼亞語",
|
||||||
|
"Azerbaijani": "亞塞拜然語",
|
||||||
|
"Bangla": "孟加拉文",
|
||||||
|
"Basque": "巴斯克語",
|
||||||
|
"Belarusian": "白俄羅斯語",
|
||||||
|
"Bosnian": "波士尼亞語",
|
||||||
|
"Bulgarian": "保加利亞語",
|
||||||
|
"Burmese": "緬甸語",
|
||||||
|
"Catalan": "加泰隆尼亞語",
|
||||||
|
"Cebuano": "宿霧語",
|
||||||
|
"Chinese (Simplified)": "簡體中文",
|
||||||
|
"Chinese (Traditional)": "繁體中文",
|
||||||
|
"Corsican": "科西嘉語",
|
||||||
|
"Croatian": "克羅埃西亞語",
|
||||||
|
"Czech": "捷克語",
|
||||||
|
"Danish": "丹麥語",
|
||||||
|
"Dutch": "荷蘭語",
|
||||||
|
"Esperanto": "世界語",
|
||||||
|
"Estonian": "愛沙尼亞語",
|
||||||
|
"Filipino": "菲律賓語",
|
||||||
|
"Finnish": "芬蘭語",
|
||||||
|
"French": "法語",
|
||||||
|
"Galician": "加利西亞語",
|
||||||
|
"Georgian": "喬治亞語",
|
||||||
|
"German": "德語",
|
||||||
|
"Greek": "希臘語",
|
||||||
|
"Gujarati": "古吉拉特語",
|
||||||
|
"Haitian Creole": "海地克里奧爾語",
|
||||||
|
"Hausa": "豪薩語",
|
||||||
|
"Hawaiian": "夏威夷語",
|
||||||
|
"Hebrew": "希伯來語",
|
||||||
|
"Hindi": "印地語",
|
||||||
|
"Hmong": "苗文",
|
||||||
|
"Hungarian": "匈牙利語",
|
||||||
|
"Icelandic": "冰島語",
|
||||||
|
"Igbo": "伊博語",
|
||||||
|
"Indonesian": "印尼語",
|
||||||
|
"Irish": "愛爾蘭語",
|
||||||
|
"Italian": "義大利語",
|
||||||
|
"Japanese": "日語",
|
||||||
|
"Javanese": "爪哇語",
|
||||||
|
"Kannada": "康納達語",
|
||||||
|
"Kazakh": "哈薩克語",
|
||||||
|
"Khmer": "高棉文",
|
||||||
|
"Korean": "韓語",
|
||||||
|
"Kurdish": "庫德語",
|
||||||
|
"Kyrgyz": "吉爾吉斯語",
|
||||||
|
"Lao": "寮語",
|
||||||
|
"Latin": "拉丁語",
|
||||||
|
"Latvian": "拉脫維亞語",
|
||||||
|
"Lithuanian": "立陶宛語",
|
||||||
|
"Luxembourgish": "盧森堡語",
|
||||||
|
"Macedonian": "馬其頓語",
|
||||||
|
"Malagasy": "馬拉加斯語",
|
||||||
|
"Malay": "馬來語",
|
||||||
|
"Malayalam": "馬拉雅拉姆語",
|
||||||
|
"Maltese": "馬爾他語",
|
||||||
|
"Maori": "毛利語",
|
||||||
|
"Marathi": "馬拉提語",
|
||||||
|
"Mongolian": "蒙古語",
|
||||||
|
"Nepali": "尼泊爾語",
|
||||||
|
"Norwegian Bokmål": "書面挪威語",
|
||||||
|
"Nyanja": "尼揚賈語",
|
||||||
|
"Pashto": "普什圖語",
|
||||||
|
"Persian": "波斯語",
|
||||||
|
"Polish": "波蘭人",
|
||||||
|
"Portuguese": "葡萄牙語",
|
||||||
|
"Punjabi": "旁遮普語",
|
||||||
|
"Romanian": "羅馬尼亞語",
|
||||||
|
"Russian": "俄語",
|
||||||
|
"Samoan": "薩摩亞語",
|
||||||
|
"Scottish Gaelic": "蘇格蘭蓋爾語",
|
||||||
|
"Serbian": "塞爾維亞語",
|
||||||
|
"Shona": "修納語",
|
||||||
|
"Sindhi": "信德語",
|
||||||
|
"Sinhala": "僧伽羅語",
|
||||||
|
"Slovak": "斯洛伐克語",
|
||||||
|
"Slovenian": "斯洛維尼亞語",
|
||||||
|
"Somali": "索馬利亞語",
|
||||||
|
"Southern Sotho": "南塞索托語",
|
||||||
|
"Spanish": "西班牙語",
|
||||||
|
"Spanish (Latin America)": "西班牙語(拉丁美洲)",
|
||||||
|
"Sundanese": "巽他語",
|
||||||
|
"Swahili": "斯瓦希里語",
|
||||||
|
"Swedish": "瑞典語",
|
||||||
|
"Tajik": "塔吉克語",
|
||||||
|
"Tamil": "坦米爾語",
|
||||||
|
"Telugu": "泰盧固語",
|
||||||
|
"Thai": "泰語",
|
||||||
|
"Turkish": "土耳其語",
|
||||||
|
"Ukrainian": "烏克蘭語",
|
||||||
|
"Urdu": "烏爾都語",
|
||||||
|
"Uzbek": "烏茲別克語",
|
||||||
|
"Vietnamese": "越南語",
|
||||||
|
"Welsh": "威爾斯語",
|
||||||
|
"Western Frisian": "西菲士蘭語",
|
||||||
|
"Xhosa": "科薩語",
|
||||||
|
"Yiddish": "意第緒語",
|
||||||
|
"Yoruba": "約魯巴語",
|
||||||
|
"Zulu": "祖魯語",
|
||||||
|
"`x` years": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 年",
|
||||||
|
"": "`x` 年"
|
||||||
|
},
|
||||||
|
"`x` months": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 月",
|
||||||
|
"": "`x` 月"
|
||||||
|
},
|
||||||
|
"`x` weeks": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 週",
|
||||||
|
"": "`x` 週"
|
||||||
|
},
|
||||||
|
"`x` days": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 天",
|
||||||
|
"": "`x` 天"
|
||||||
|
},
|
||||||
|
"`x` hours": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 小時",
|
||||||
|
"": "`x` 小時"
|
||||||
|
},
|
||||||
|
"`x` minutes": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 天",
|
||||||
|
"": "`x` 天"
|
||||||
|
},
|
||||||
|
"`x` seconds": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` 秒",
|
||||||
|
"": "`x` 秒"
|
||||||
|
},
|
||||||
|
"Fallback comments: ": "汰退留言:",
|
||||||
|
"Popular": "熱門頻道",
|
||||||
|
"Top": "熱門影片",
|
||||||
|
"About": "關於",
|
||||||
|
"Rating: ": "評分:",
|
||||||
|
"Language: ": "語言:",
|
||||||
|
"View as playlist": "以播放清單檢視",
|
||||||
|
"Default": "預設值",
|
||||||
|
"Music": "音樂",
|
||||||
|
"Gaming": "遊戲",
|
||||||
|
"News": "新聞",
|
||||||
|
"Movies": "電影",
|
||||||
|
"Download": "下載",
|
||||||
|
"Download as: ": "下載為:",
|
||||||
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
|
"(edited)": "(已編輯)",
|
||||||
|
"YouTube comment permalink": "YouTube 留言永久連結",
|
||||||
|
"permalink": "",
|
||||||
|
"`x` marked it with a ❤": "`x` 為此標記 ❤",
|
||||||
|
"Audio mode": "音訊模式",
|
||||||
|
"Video mode": "視訊模式",
|
||||||
|
"Videos": "影片",
|
||||||
|
"Playlists": "播放清單",
|
||||||
|
"Community": "社群",
|
||||||
|
"Current version: ": "目前版本:"
|
||||||
|
}
|
||||||
BIN
screenshots/native_notification.png
Normal file
BIN
screenshots/native_notification.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 22 KiB |
20
shard.yml
20
shard.yml
@@ -1,5 +1,5 @@
|
|||||||
name: invidious
|
name: invidious
|
||||||
version: 0.16.0
|
version: 0.20.1
|
||||||
|
|
||||||
authors:
|
authors:
|
||||||
- Omar Roth <omarroth@protonmail.com>
|
- Omar Roth <omarroth@protonmail.com>
|
||||||
@@ -9,13 +9,25 @@ targets:
|
|||||||
main: src/invidious.cr
|
main: src/invidious.cr
|
||||||
|
|
||||||
dependencies:
|
dependencies:
|
||||||
kemal:
|
|
||||||
github: kemalcr/kemal
|
|
||||||
pg:
|
pg:
|
||||||
github: will/crystal-pg
|
github: will/crystal-pg
|
||||||
|
version: ~> 0.19.0
|
||||||
sqlite3:
|
sqlite3:
|
||||||
github: crystal-lang/crystal-sqlite3
|
github: crystal-lang/crystal-sqlite3
|
||||||
|
version: ~> 0.14.0
|
||||||
|
kemal:
|
||||||
|
github: kemalcr/kemal
|
||||||
|
version: ~> 0.26.0
|
||||||
|
pool:
|
||||||
|
github: ysbaddaden/pool
|
||||||
|
version: ~> 0.2.3
|
||||||
|
protodec:
|
||||||
|
github: omarroth/protodec
|
||||||
|
version: ~> 0.1.2
|
||||||
|
lsquic:
|
||||||
|
github: omarroth/lsquic.cr
|
||||||
|
version: ~> 0.1.3
|
||||||
|
|
||||||
crystal: 0.27.2
|
crystal: 0.31.1
|
||||||
|
|
||||||
license: AGPLv3
|
license: AGPLv3
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
5325
src/invidious.cr
5325
src/invidious.cr
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -22,6 +22,7 @@ class RedditComment
|
|||||||
replies: RedditThing | String,
|
replies: RedditThing | String,
|
||||||
score: Int32,
|
score: Int32,
|
||||||
depth: Int32,
|
depth: Int32,
|
||||||
|
permalink: String,
|
||||||
created_utc: {
|
created_utc: {
|
||||||
type: Time,
|
type: Time,
|
||||||
converter: RedditComment::TimeConverter,
|
converter: RedditComment::TimeConverter,
|
||||||
@@ -56,14 +57,22 @@ class RedditListing
|
|||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_mode, region, sort_by = "top")
|
def fetch_youtube_comments(id, db, cursor, format, locale, thin_mode, region, sort_by = "top")
|
||||||
video = get_video(id, db, proxies, region: region)
|
video = get_video(id, db, region: region)
|
||||||
session_token = video.info["session_token"]?
|
session_token = video.info["session_token"]?
|
||||||
|
|
||||||
ctoken = produce_comment_continuation(id, cursor: "", sort_by: sort_by)
|
case cursor
|
||||||
continuation ||= ctoken
|
when nil, ""
|
||||||
|
ctoken = produce_comment_continuation(id, cursor: "", sort_by: sort_by)
|
||||||
|
# when .starts_with? "Ug"
|
||||||
|
# ctoken = produce_comment_reply_continuation(id, video.ucid, cursor)
|
||||||
|
when .starts_with? "ADSJ"
|
||||||
|
ctoken = produce_comment_continuation(id, cursor: cursor, sort_by: sort_by)
|
||||||
|
else
|
||||||
|
ctoken = cursor
|
||||||
|
end
|
||||||
|
|
||||||
if !continuation || !session_token
|
if !session_token
|
||||||
if format == "json"
|
if format == "json"
|
||||||
return {"comments" => [] of String}.to_json
|
return {"comments" => [] of String}.to_json
|
||||||
else
|
else
|
||||||
@@ -72,11 +81,10 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
end
|
end
|
||||||
|
|
||||||
post_req = {
|
post_req = {
|
||||||
"session_token" => session_token,
|
page_token: ctoken,
|
||||||
|
session_token: session_token,
|
||||||
}
|
}
|
||||||
post_req = HTTP::Params.encode(post_req)
|
|
||||||
|
|
||||||
client = make_client(YT_URL, proxies, video.info["region"]?)
|
|
||||||
headers = HTTP::Headers.new
|
headers = HTTP::Headers.new
|
||||||
|
|
||||||
headers["content-type"] = "application/x-www-form-urlencoded"
|
headers["content-type"] = "application/x-www-form-urlencoded"
|
||||||
@@ -89,7 +97,7 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
headers["x-youtube-client-name"] = "1"
|
headers["x-youtube-client-name"] = "1"
|
||||||
headers["x-youtube-client-version"] = "2.20180719"
|
headers["x-youtube-client-version"] = "2.20180719"
|
||||||
|
|
||||||
response = client.post("/comment_service_ajax?action_get_comments=1&ctoken=#{continuation}&continuation=#{continuation}&hl=en&gl=US", headers, post_req)
|
response = YT_POOL.client(region, &.post("/comment_service_ajax?action_get_comments=1&hl=en&gl=US", headers, form: post_req))
|
||||||
response = JSON.parse(response.body)
|
response = JSON.parse(response.body)
|
||||||
|
|
||||||
if !response["response"]["continuationContents"]?
|
if !response["response"]["continuationContents"]?
|
||||||
@@ -112,10 +120,13 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
comments = JSON.build do |json|
|
response = JSON.build do |json|
|
||||||
json.object do
|
json.object do
|
||||||
if body["header"]?
|
if body["header"]?
|
||||||
comment_count = body["header"]["commentsHeaderRenderer"]["countText"]["simpleText"].as_s.delete("Comments,").to_i
|
count_text = body["header"]["commentsHeaderRenderer"]["countText"]
|
||||||
|
comment_count = (count_text["simpleText"]? || count_text["runs"]?.try &.[0]?.try &.["text"]?)
|
||||||
|
.try &.as_s.gsub(/\D/, "").to_i? || 0
|
||||||
|
|
||||||
json.field "commentCount", comment_count
|
json.field "commentCount", comment_count
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -139,16 +150,9 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
node_comment = node["commentRenderer"]
|
node_comment = node["commentRenderer"]
|
||||||
end
|
end
|
||||||
|
|
||||||
content_html = node_comment["contentText"]["simpleText"]?.try &.as_s.rchop('\ufeff')
|
content_html = node_comment["contentText"]["simpleText"]?.try &.as_s.rchop('\ufeff').try { |block| HTML.escape(block) }.to_s ||
|
||||||
if content_html
|
content_to_comment_html(node_comment["contentText"]["runs"].as_a).try &.to_s || ""
|
||||||
content_html = HTML.escape(content_html)
|
author = node_comment["authorText"]?.try &.["simpleText"]? || ""
|
||||||
end
|
|
||||||
|
|
||||||
content_html ||= content_to_comment_html(node_comment["contentText"]["runs"].as_a)
|
|
||||||
content_html, content = html_to_content(content_html)
|
|
||||||
|
|
||||||
author = node_comment["authorText"]?.try &.["simpleText"]
|
|
||||||
author ||= ""
|
|
||||||
|
|
||||||
json.field "author", author
|
json.field "author", author
|
||||||
json.field "authorThumbnails" do
|
json.field "authorThumbnails" do
|
||||||
@@ -180,10 +184,12 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
json.field "isEdited", false
|
json.field "isEdited", false
|
||||||
end
|
end
|
||||||
|
|
||||||
json.field "content", content
|
json.field "content", html_to_content(content_html)
|
||||||
json.field "contentHtml", content_html
|
json.field "contentHtml", content_html
|
||||||
|
|
||||||
json.field "published", published.to_unix
|
json.field "published", published.to_unix
|
||||||
json.field "publishedText", translate(locale, "`x` ago", recode_date(published, locale))
|
json.field "publishedText", translate(locale, "`x` ago", recode_date(published, locale))
|
||||||
|
|
||||||
json.field "likeCount", node_comment["likeCount"]
|
json.field "likeCount", node_comment["likeCount"]
|
||||||
json.field "commentId", node_comment["commentId"]
|
json.field "commentId", node_comment["commentId"]
|
||||||
json.field "authorIsChannelOwner", node_comment["authorIsChannelOwner"]
|
json.field "authorIsChannelOwner", node_comment["authorIsChannelOwner"]
|
||||||
@@ -199,13 +205,8 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
end
|
end
|
||||||
|
|
||||||
if node_replies && !response["commentRepliesContinuation"]?
|
if node_replies && !response["commentRepliesContinuation"]?
|
||||||
reply_count = node_replies["moreText"]["simpleText"].as_s.delete("View all reply replies,")
|
reply_count = (node_replies["moreText"]["simpleText"]? || node_replies["moreText"]["runs"]?.try &.[0]?.try &.["text"]?)
|
||||||
if reply_count.empty?
|
.try &.as_s.gsub(/\D/, "").to_i? || 1
|
||||||
reply_count = 1
|
|
||||||
else
|
|
||||||
reply_count = reply_count.try &.to_i?
|
|
||||||
reply_count ||= 1
|
|
||||||
end
|
|
||||||
|
|
||||||
continuation = node_replies["continuations"]?.try &.as_a[0]["nextContinuationData"]["continuation"].as_s
|
continuation = node_replies["continuations"]?.try &.as_a[0]["nextContinuationData"]["continuation"].as_s
|
||||||
continuation ||= ""
|
continuation ||= ""
|
||||||
@@ -223,22 +224,22 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
end
|
end
|
||||||
|
|
||||||
if body["continuations"]?
|
if body["continuations"]?
|
||||||
continuation = body["continuations"][0]["nextContinuationData"]["continuation"]
|
continuation = body["continuations"][0]["nextContinuationData"]["continuation"].as_s
|
||||||
json.field "continuation", continuation
|
json.field "continuation", cursor.try &.starts_with?("E") ? continuation : extract_comment_cursor(continuation)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if format == "html"
|
if format == "html"
|
||||||
comments = JSON.parse(comments)
|
response = JSON.parse(response)
|
||||||
content_html = template_youtube_comments(comments, locale, thin_mode)
|
content_html = template_youtube_comments(response, locale, thin_mode)
|
||||||
|
|
||||||
comments = JSON.build do |json|
|
response = JSON.build do |json|
|
||||||
json.object do
|
json.object do
|
||||||
json.field "contentHtml", content_html
|
json.field "contentHtml", content_html
|
||||||
|
|
||||||
if comments["commentCount"]?
|
if response["commentCount"]?
|
||||||
json.field "commentCount", comments["commentCount"]
|
json.field "commentCount", response["commentCount"]
|
||||||
else
|
else
|
||||||
json.field "commentCount", 0
|
json.field "commentCount", 0
|
||||||
end
|
end
|
||||||
@@ -246,25 +247,30 @@ def fetch_youtube_comments(id, db, continuation, proxies, format, locale, thin_m
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
return comments
|
return response
|
||||||
end
|
end
|
||||||
|
|
||||||
def fetch_reddit_comments(id)
|
def fetch_reddit_comments(id, sort_by = "confidence")
|
||||||
client = make_client(REDDIT_URL)
|
client = make_client(REDDIT_URL)
|
||||||
headers = HTTP::Headers{"User-Agent" => "web:invidious:v#{CURRENT_VERSION} (by /u/omarroth)"}
|
headers = HTTP::Headers{"User-Agent" => "web:invidious:v#{CURRENT_VERSION} (by /u/omarroth)"}
|
||||||
|
|
||||||
query = "(url:3D#{id}%20OR%20url:#{id})%20(site:youtube.com%20OR%20site:youtu.be)"
|
# TODO: Use something like #479 for a static list of instances to use here
|
||||||
|
query = "(url:3D#{id}%20OR%20url:#{id})%20(site:invidio.us%20OR%20site:youtube.com%20OR%20site:youtu.be)"
|
||||||
search_results = client.get("/search.json?q=#{query}", headers)
|
search_results = client.get("/search.json?q=#{query}", headers)
|
||||||
|
|
||||||
if search_results.status_code == 200
|
if search_results.status_code == 200
|
||||||
search_results = RedditThing.from_json(search_results.body)
|
search_results = RedditThing.from_json(search_results.body)
|
||||||
|
|
||||||
|
# For videos that have more than one thread, choose the one with the highest score
|
||||||
thread = search_results.data.as(RedditListing).children.sort_by { |child| child.data.as(RedditLink).score }[-1]
|
thread = search_results.data.as(RedditListing).children.sort_by { |child| child.data.as(RedditLink).score }[-1]
|
||||||
thread = thread.data.as(RedditLink)
|
thread = thread.data.as(RedditLink)
|
||||||
|
|
||||||
result = client.get("/r/#{thread.subreddit}/comments/#{thread.id}.json?limit=100&sort=top", headers).body
|
result = client.get("/r/#{thread.subreddit}/comments/#{thread.id}.json?limit=100&sort=#{sort_by}", headers).body
|
||||||
result = Array(RedditThing).from_json(result)
|
result = Array(RedditThing).from_json(result)
|
||||||
elsif search_results.status_code == 302
|
elsif search_results.status_code == 302
|
||||||
|
# Previously, if there was only one result then the API would redirect to that result.
|
||||||
|
# Now, it appears it will still return a listing so this section is likely unnecessary.
|
||||||
|
|
||||||
result = client.get(search_results.headers["Location"], headers).body
|
result = client.get(search_results.headers["Location"], headers).body
|
||||||
result = Array(RedditThing).from_json(result)
|
result = Array(RedditThing).from_json(result)
|
||||||
|
|
||||||
@@ -278,56 +284,110 @@ def fetch_reddit_comments(id)
|
|||||||
end
|
end
|
||||||
|
|
||||||
def template_youtube_comments(comments, locale, thin_mode)
|
def template_youtube_comments(comments, locale, thin_mode)
|
||||||
html = ""
|
String.build do |html|
|
||||||
|
root = comments["comments"].as_a
|
||||||
root = comments["comments"].as_a
|
root.each do |child|
|
||||||
root.each do |child|
|
if child["replies"]?
|
||||||
if child["replies"]?
|
replies_html = <<-END_HTML
|
||||||
replies_html = <<-END_HTML
|
<div id="replies" class="pure-g">
|
||||||
<div id="replies" class="pure-g">
|
<div class="pure-u-1-24"></div>
|
||||||
<div class="pure-u-1-24"></div>
|
<div class="pure-u-23-24">
|
||||||
<div class="pure-u-23-24">
|
<p>
|
||||||
<p>
|
<a href="javascript:void(0)" data-continuation="#{child["replies"]["continuation"]}"
|
||||||
<a href="javascript:void(0)" data-continuation="#{child["replies"]["continuation"]}"
|
onclick="get_youtube_replies(this)">#{translate(locale, "View `x` replies", number_with_separator(child["replies"]["replyCount"]))}</a>
|
||||||
onclick="get_youtube_replies(this)">#{translate(locale, "View `x` replies", child["replies"]["replyCount"].to_s)}</a>
|
</p>
|
||||||
</p>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
END_HTML
|
||||||
END_HTML
|
|
||||||
end
|
|
||||||
|
|
||||||
if !thin_mode
|
|
||||||
author_thumbnail = "/ggpht#{URI.parse(child["authorThumbnails"][-1]["url"].as_s).full_path}"
|
|
||||||
else
|
|
||||||
author_thumbnail = ""
|
|
||||||
end
|
|
||||||
|
|
||||||
html += <<-END_HTML
|
|
||||||
<div class="pure-g">
|
|
||||||
<div class="pure-u-4-24 pure-u-md-2-24">
|
|
||||||
<img style="width:90%; padding-right:1em; padding-top:1em;" src="#{author_thumbnail}">
|
|
||||||
</div>
|
|
||||||
<div class="pure-u-20-24 pure-u-md-22-24">
|
|
||||||
<p>
|
|
||||||
<b>
|
|
||||||
<a class="#{child["authorIsChannelOwner"] == true ? "channel-owner" : ""}" href="#{child["authorUrl"]}">#{child["author"]}</a>
|
|
||||||
</b>
|
|
||||||
<p style="white-space:pre-wrap">#{child["contentHtml"]}</p>
|
|
||||||
<span title="#{Time.unix(child["published"].as_i64).to_s(translate(locale, "%A %B %-d, %Y"))}">#{translate(locale, "`x` ago", recode_date(Time.unix(child["published"].as_i64), locale))} #{child["isEdited"] == true ? translate(locale, "(edited)") : ""}</span>
|
|
||||||
|
|
|
||||||
<a href="https://www.youtube.com/watch?v=#{comments["videoId"]}&lc=#{child["commentId"]}" title="#{translate(locale, "Youtube permalink of the comment")}">[YT]</a>
|
|
||||||
|
|
|
||||||
<i class="icon ion-ios-thumbs-up"></i> #{number_with_separator(child["likeCount"])}
|
|
||||||
END_HTML
|
|
||||||
|
|
||||||
if child["creatorHeart"]?
|
|
||||||
if !thin_mode
|
|
||||||
creator_thumbnail = "/ggpht#{URI.parse(child["creatorHeart"]["creatorThumbnail"].as_s).full_path}"
|
|
||||||
else
|
|
||||||
creator_thumbnail = ""
|
|
||||||
end
|
end
|
||||||
|
|
||||||
html += <<-END_HTML
|
if !thin_mode
|
||||||
|
author_thumbnail = "/ggpht#{URI.parse(child["authorThumbnails"][-1]["url"].as_s).full_path}"
|
||||||
|
else
|
||||||
|
author_thumbnail = ""
|
||||||
|
end
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
<div class="pure-g" style="width:100%">
|
||||||
|
<div class="channel-profile pure-u-4-24 pure-u-md-2-24">
|
||||||
|
<img style="padding-right:1em;padding-top:1em;width:90%" src="#{author_thumbnail}">
|
||||||
|
</div>
|
||||||
|
<div class="pure-u-20-24 pure-u-md-22-24">
|
||||||
|
<p>
|
||||||
|
<b>
|
||||||
|
<a class="#{child["authorIsChannelOwner"] == true ? "channel-owner" : ""}" href="#{child["authorUrl"]}">#{child["author"]}</a>
|
||||||
|
</b>
|
||||||
|
<p style="white-space:pre-wrap">#{child["contentHtml"]}</p>
|
||||||
|
END_HTML
|
||||||
|
|
||||||
|
if child["attachment"]?
|
||||||
|
attachment = child["attachment"]
|
||||||
|
|
||||||
|
case attachment["type"]
|
||||||
|
when "image"
|
||||||
|
attachment = attachment["imageThumbnails"][1]
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
<div class="pure-g">
|
||||||
|
<div class="pure-u-1 pure-u-md-1-2">
|
||||||
|
<img style="width:100%" src="/ggpht#{URI.parse(attachment["url"].as_s).full_path}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
END_HTML
|
||||||
|
when "video"
|
||||||
|
html << <<-END_HTML
|
||||||
|
<div class="pure-g">
|
||||||
|
<div class="pure-u-1 pure-u-md-1-2">
|
||||||
|
<div style="position:relative;width:100%;height:0;padding-bottom:56.25%;margin-bottom:5px">
|
||||||
|
END_HTML
|
||||||
|
|
||||||
|
if attachment["error"]?
|
||||||
|
html << <<-END_HTML
|
||||||
|
<p>#{attachment["error"]}</p>
|
||||||
|
END_HTML
|
||||||
|
else
|
||||||
|
html << <<-END_HTML
|
||||||
|
<iframe id='ivplayer' type='text/html' style='position:absolute;width:100%;height:100%;left:0;top:0' src='/embed/#{attachment["videoId"]?}?autoplay=0' frameborder='0'></iframe>
|
||||||
|
END_HTML
|
||||||
|
end
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
END_HTML
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
<span title="#{Time.unix(child["published"].as_i64).to_s(translate(locale, "%A %B %-d, %Y"))}">#{translate(locale, "`x` ago", recode_date(Time.unix(child["published"].as_i64), locale))} #{child["isEdited"] == true ? translate(locale, "(edited)") : ""}</span>
|
||||||
|
|
|
||||||
|
END_HTML
|
||||||
|
|
||||||
|
if comments["videoId"]?
|
||||||
|
html << <<-END_HTML
|
||||||
|
<a href="https://www.youtube.com/watch?v=#{comments["videoId"]}&lc=#{child["commentId"]}" title="#{translate(locale, "YouTube comment permalink")}">[YT]</a>
|
||||||
|
|
|
||||||
|
END_HTML
|
||||||
|
elsif comments["authorId"]?
|
||||||
|
html << <<-END_HTML
|
||||||
|
<a href="https://www.youtube.com/channel/#{comments["authorId"]}/community?lb=#{child["commentId"]}" title="#{translate(locale, "YouTube comment permalink")}">[YT]</a>
|
||||||
|
|
|
||||||
|
END_HTML
|
||||||
|
end
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
<i class="icon ion-ios-thumbs-up"></i> #{number_with_separator(child["likeCount"])}
|
||||||
|
END_HTML
|
||||||
|
|
||||||
|
if child["creatorHeart"]?
|
||||||
|
if !thin_mode
|
||||||
|
creator_thumbnail = "/ggpht#{URI.parse(child["creatorHeart"]["creatorThumbnail"].as_s).full_path}"
|
||||||
|
else
|
||||||
|
creator_thumbnail = ""
|
||||||
|
end
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
<span class="creator-heart-container" title="#{translate(locale, "`x` marked it with a ❤", child["creatorHeart"]["creatorName"].as_s)}">
|
<span class="creator-heart-container" title="#{translate(locale, "`x` marked it with a ❤", child["creatorHeart"]["creatorName"].as_s)}">
|
||||||
<div class="creator-heart">
|
<div class="creator-heart">
|
||||||
<img class="creator-heart-background-hearted" src="#{creator_thumbnail}"></img>
|
<img class="creator-heart-background-hearted" src="#{creator_thumbnail}"></img>
|
||||||
@@ -336,84 +396,77 @@ def template_youtube_comments(comments, locale, thin_mode)
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</span>
|
</span>
|
||||||
|
END_HTML
|
||||||
|
end
|
||||||
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
</p>
|
||||||
|
#{replies_html}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
END_HTML
|
END_HTML
|
||||||
end
|
end
|
||||||
|
|
||||||
html += <<-END_HTML
|
if comments["continuation"]?
|
||||||
</p>
|
html << <<-END_HTML
|
||||||
#{replies_html}
|
<div class="pure-g">
|
||||||
|
<div class="pure-u-1">
|
||||||
|
<p>
|
||||||
|
<a href="javascript:void(0)" data-continuation="#{comments["continuation"]}"
|
||||||
|
onclick="get_youtube_replies(this, true)">#{translate(locale, "Load more")}</a>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
END_HTML
|
||||||
END_HTML
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if comments["continuation"]?
|
|
||||||
html += <<-END_HTML
|
|
||||||
<div class="pure-g">
|
|
||||||
<div class="pure-u-1">
|
|
||||||
<p>
|
|
||||||
<a href="javascript:void(0)" data-continuation="#{comments["continuation"]}"
|
|
||||||
onclick="get_youtube_replies(this, true)">#{translate(locale, "Load more")}</a>
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
END_HTML
|
|
||||||
end
|
|
||||||
|
|
||||||
return html
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def template_reddit_comments(root, locale)
|
def template_reddit_comments(root, locale)
|
||||||
html = ""
|
String.build do |html|
|
||||||
root.each do |child|
|
root.each do |child|
|
||||||
if child.data.is_a?(RedditComment)
|
if child.data.is_a?(RedditComment)
|
||||||
child = child.data.as(RedditComment)
|
child = child.data.as(RedditComment)
|
||||||
author = child.author
|
body_html = HTML.unescape(child.body_html)
|
||||||
score = child.score
|
|
||||||
body_html = HTML.unescape(child.body_html)
|
|
||||||
|
|
||||||
replies_html = ""
|
replies_html = ""
|
||||||
if child.replies.is_a?(RedditThing)
|
if child.replies.is_a?(RedditThing)
|
||||||
replies = child.replies.as(RedditThing)
|
replies = child.replies.as(RedditThing)
|
||||||
replies_html = template_reddit_comments(replies.data.as(RedditListing).children, locale)
|
replies_html = template_reddit_comments(replies.data.as(RedditListing).children, locale)
|
||||||
end
|
end
|
||||||
|
|
||||||
content = <<-END_HTML
|
if child.depth > 0
|
||||||
<p>
|
html << <<-END_HTML
|
||||||
<a href="javascript:void(0)" onclick="toggle_parent(this)">[ - ]</a>
|
|
||||||
<b><a href="https://www.reddit.com/user/#{author}">#{author}</a></b>
|
|
||||||
#{translate(locale, "`x` points", number_with_separator(score))}
|
|
||||||
#{translate(locale, "`x` ago", recode_date(child.created_utc, locale))}
|
|
||||||
</p>
|
|
||||||
<div>
|
|
||||||
#{body_html}
|
|
||||||
#{replies_html}
|
|
||||||
</div>
|
|
||||||
END_HTML
|
|
||||||
|
|
||||||
if child.depth > 0
|
|
||||||
html += <<-END_HTML
|
|
||||||
<div class="pure-g">
|
<div class="pure-g">
|
||||||
<div class="pure-u-1-24">
|
<div class="pure-u-1-24">
|
||||||
</div>
|
</div>
|
||||||
<div class="pure-u-23-24">
|
<div class="pure-u-23-24">
|
||||||
#{content}
|
END_HTML
|
||||||
</div>
|
else
|
||||||
</div>
|
html << <<-END_HTML
|
||||||
END_HTML
|
|
||||||
else
|
|
||||||
html += <<-END_HTML
|
|
||||||
<div class="pure-g">
|
<div class="pure-g">
|
||||||
<div class="pure-u-1">
|
<div class="pure-u-1">
|
||||||
#{content}
|
END_HTML
|
||||||
</div>
|
end
|
||||||
</div>
|
|
||||||
|
html << <<-END_HTML
|
||||||
|
<p>
|
||||||
|
<a href="javascript:void(0)" onclick="toggle_parent(this)">[ - ]</a>
|
||||||
|
<b><a href="https://www.reddit.com/user/#{child.author}">#{child.author}</a></b>
|
||||||
|
#{translate(locale, "`x` points", number_with_separator(child.score))}
|
||||||
|
<span title="#{child.created_utc.to_s(translate(locale, "%a %B %-d %T %Y UTC"))}">#{translate(locale, "`x` ago", recode_date(child.created_utc, locale))}</span>
|
||||||
|
<a href="https://www.reddit.com#{child.permalink}" title="#{translate(locale, "permalink")}">#{translate(locale, "permalink")}</a>
|
||||||
|
</p>
|
||||||
|
<div>
|
||||||
|
#{body_html}
|
||||||
|
#{replies_html}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
END_HTML
|
END_HTML
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
return html
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def replace_links(html)
|
def replace_links(html)
|
||||||
@@ -441,8 +494,12 @@ def replace_links(html)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
html = html.to_xml(options: XML::SaveOptions::NO_DECL)
|
html = html.xpath_node(%q(//body)).not_nil!
|
||||||
return html
|
if node = html.xpath_node(%q(./p))
|
||||||
|
html = node
|
||||||
|
end
|
||||||
|
|
||||||
|
return html.to_xml(options: XML::SaveOptions::NO_DECL)
|
||||||
end
|
end
|
||||||
|
|
||||||
def fill_links(html, scheme, host)
|
def fill_links(html, scheme, host)
|
||||||
@@ -459,12 +516,10 @@ def fill_links(html, scheme, host)
|
|||||||
end
|
end
|
||||||
|
|
||||||
if host == "www.youtube.com"
|
if host == "www.youtube.com"
|
||||||
html = html.xpath_node(%q(//body)).not_nil!.to_xml
|
html = html.xpath_node(%q(//body/p)).not_nil!
|
||||||
else
|
|
||||||
html = html.to_xml(options: XML::SaveOptions::NO_DECL)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
return html
|
return html.to_xml(options: XML::SaveOptions::NO_DECL)
|
||||||
end
|
end
|
||||||
|
|
||||||
def content_to_comment_html(content)
|
def content_to_comment_html(content)
|
||||||
@@ -511,115 +566,90 @@ def content_to_comment_html(content)
|
|||||||
end
|
end
|
||||||
|
|
||||||
text
|
text
|
||||||
end.join.rchop('\ufeff')
|
end.join("").delete('\ufeff')
|
||||||
|
|
||||||
return comment_html
|
return comment_html
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def extract_comment_cursor(continuation)
|
||||||
|
cursor = URI.decode_www_form(continuation)
|
||||||
|
.try { |i| Base64.decode(i) }
|
||||||
|
.try { |i| IO::Memory.new(i) }
|
||||||
|
.try { |i| Protodec::Any.parse(i) }
|
||||||
|
.try { |i| i["6:2:embedded"]["1:0:string"].as_s }
|
||||||
|
|
||||||
|
return cursor
|
||||||
|
end
|
||||||
|
|
||||||
def produce_comment_continuation(video_id, cursor = "", sort_by = "top")
|
def produce_comment_continuation(video_id, cursor = "", sort_by = "top")
|
||||||
continuation = IO::Memory.new
|
object = {
|
||||||
|
"2:embedded" => {
|
||||||
|
"2:string" => video_id,
|
||||||
|
"24:varint" => 1_i64,
|
||||||
|
"25:varint" => 1_i64,
|
||||||
|
"28:varint" => 1_i64,
|
||||||
|
"36:embedded" => {
|
||||||
|
"5:varint" => -1_i64,
|
||||||
|
"8:varint" => 0_i64,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"3:varint" => 6_i64,
|
||||||
|
"6:embedded" => {
|
||||||
|
"1:string" => cursor,
|
||||||
|
"4:embedded" => {
|
||||||
|
"4:string" => video_id,
|
||||||
|
"6:varint" => 0_i64,
|
||||||
|
},
|
||||||
|
"5:varint" => 20_i64,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
continuation.write(Bytes[0x12, 0x26])
|
case sort_by
|
||||||
|
when "top"
|
||||||
continuation.write(Bytes[0x12, video_id.size])
|
object["6:embedded"].as(Hash)["4:embedded"].as(Hash)["6:varint"] = 0_i64
|
||||||
continuation.print(video_id)
|
when "new", "newest"
|
||||||
|
object["6:embedded"].as(Hash)["4:embedded"].as(Hash)["6:varint"] = 1_i64
|
||||||
continuation.write(Bytes[0xc0, 0x01, 0x01])
|
|
||||||
continuation.write(Bytes[0xc8, 0x01, 0x01])
|
|
||||||
continuation.write(Bytes[0xe0, 0x01, 0x01])
|
|
||||||
|
|
||||||
continuation.write(Bytes[0xa2, 0x02, 0x0d])
|
|
||||||
continuation.write(Bytes[0x28, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01])
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x40, 0x00])
|
|
||||||
continuation.write(Bytes[0x18, 0x06])
|
|
||||||
|
|
||||||
if cursor.empty?
|
|
||||||
continuation.write(Bytes[0x32])
|
|
||||||
continuation.write(write_var_int(video_id.size + 8))
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x22, video_id.size + 4])
|
|
||||||
continuation.write(Bytes[0x22, video_id.size])
|
|
||||||
continuation.print(video_id)
|
|
||||||
|
|
||||||
case sort_by
|
|
||||||
when "top"
|
|
||||||
continuation.write(Bytes[0x30, 0x00])
|
|
||||||
when "new", "newest"
|
|
||||||
continuation.write(Bytes[0x30, 0x01])
|
|
||||||
end
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x78, 0x02])
|
|
||||||
else
|
|
||||||
continuation.write(Bytes[0x32])
|
|
||||||
continuation.write(write_var_int(cursor.size + video_id.size + 11))
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x0a])
|
|
||||||
continuation.write(write_var_int(cursor.size))
|
|
||||||
continuation.print(cursor)
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x22, video_id.size + 4])
|
|
||||||
continuation.write(Bytes[0x22, video_id.size])
|
|
||||||
continuation.print(video_id)
|
|
||||||
|
|
||||||
case sort_by
|
|
||||||
when "top"
|
|
||||||
continuation.write(Bytes[0x30, 0x00])
|
|
||||||
when "new", "newest"
|
|
||||||
continuation.write(Bytes[0x30, 0x01])
|
|
||||||
end
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x28, 0x14])
|
|
||||||
end
|
end
|
||||||
|
|
||||||
continuation.rewind
|
continuation = object.try { |i| Protodec::Any.cast_json(object) }
|
||||||
continuation = continuation.gets_to_end
|
.try { |i| Protodec::Any.from_json(i) }
|
||||||
|
.try { |i| Base64.urlsafe_encode(i) }
|
||||||
continuation = Base64.urlsafe_encode(continuation.to_slice)
|
.try { |i| URI.encode_www_form(i) }
|
||||||
continuation = URI.escape(continuation)
|
|
||||||
|
|
||||||
return continuation
|
return continuation
|
||||||
end
|
end
|
||||||
|
|
||||||
def produce_comment_reply_continuation(video_id, ucid, comment_id)
|
def produce_comment_reply_continuation(video_id, ucid, comment_id)
|
||||||
continuation = IO::Memory.new
|
object = {
|
||||||
|
"2:embedded" => {
|
||||||
|
"2:string" => video_id,
|
||||||
|
"24:varint" => 1_i64,
|
||||||
|
"25:varint" => 1_i64,
|
||||||
|
"28:varint" => 1_i64,
|
||||||
|
"36:embedded" => {
|
||||||
|
"5:varint" => -1_i64,
|
||||||
|
"8:varint" => 0_i64,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"3:varint" => 6_i64,
|
||||||
|
"6:embedded" => {
|
||||||
|
"3:embedded" => {
|
||||||
|
"2:string" => comment_id,
|
||||||
|
"4:embedded" => {
|
||||||
|
"1:varint" => 0_i64,
|
||||||
|
},
|
||||||
|
"5:string" => ucid,
|
||||||
|
"6:string" => video_id,
|
||||||
|
"8:varint" => 1_i64,
|
||||||
|
"9:varint" => 10_i64,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
continuation.write(Bytes[0x12, 0x26])
|
continuation = object.try { |i| Protodec::Any.cast_json(object) }
|
||||||
|
.try { |i| Protodec::Any.from_json(i) }
|
||||||
continuation.write(Bytes[0x12, video_id.size])
|
.try { |i| Base64.urlsafe_encode(i) }
|
||||||
continuation.print(video_id)
|
.try { |i| URI.encode_www_form(i) }
|
||||||
|
|
||||||
continuation.write(Bytes[0xc0, 0x01, 0x01])
|
|
||||||
continuation.write(Bytes[0xc8, 0x01, 0x01])
|
|
||||||
continuation.write(Bytes[0xe0, 0x01, 0x01])
|
|
||||||
|
|
||||||
continuation.write(Bytes[0xa2, 0x02, 0x0d])
|
|
||||||
continuation.write(Bytes[0x28, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01])
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x40, 0x00])
|
|
||||||
continuation.write(Bytes[0x18, 0x06])
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x32, ucid.size + video_id.size + comment_id.size + 16])
|
|
||||||
continuation.write(Bytes[0x1a, ucid.size + video_id.size + comment_id.size + 14])
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x12, comment_id.size])
|
|
||||||
continuation.print(comment_id)
|
|
||||||
|
|
||||||
continuation.write(Bytes[0x22, 0x02, 0x08, 0x00]) # ??
|
|
||||||
|
|
||||||
continuation.write(Bytes[ucid.size + video_id.size + 7])
|
|
||||||
continuation.write(Bytes[ucid.size])
|
|
||||||
continuation.print(ucid)
|
|
||||||
continuation.write(Bytes[0x32, video_id.size])
|
|
||||||
continuation.print(video_id)
|
|
||||||
continuation.write(Bytes[0x40, 0x01])
|
|
||||||
continuation.write(Bytes[0x48, 0x0a])
|
|
||||||
|
|
||||||
continuation.rewind
|
|
||||||
continuation = continuation.gets_to_end
|
|
||||||
|
|
||||||
continuation = Base64.urlsafe_encode(continuation.to_slice)
|
|
||||||
continuation = URI.escape(continuation)
|
|
||||||
|
|
||||||
return continuation
|
return continuation
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -20,7 +20,9 @@ module HTTP::Handler
|
|||||||
end
|
end
|
||||||
|
|
||||||
class Kemal::RouteHandler
|
class Kemal::RouteHandler
|
||||||
exclude ["/api/v1/*"]
|
{% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
|
||||||
|
exclude ["/api/v1/*"], {{method}}
|
||||||
|
{% end %}
|
||||||
|
|
||||||
# Processes the route if it's a match. Otherwise renders 404.
|
# Processes the route if it's a match. Otherwise renders 404.
|
||||||
private def process_request(context)
|
private def process_request(context)
|
||||||
@@ -31,13 +33,19 @@ class Kemal::RouteHandler
|
|||||||
raise Kemal::Exceptions::CustomException.new(context)
|
raise Kemal::Exceptions::CustomException.new(context)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if context.request.method == "HEAD" && context.request.path.ends_with? ".jpg"
|
||||||
|
context.response.headers["Content-Type"] = "image/jpeg"
|
||||||
|
end
|
||||||
|
|
||||||
context.response.print(content)
|
context.response.print(content)
|
||||||
context
|
context
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class Kemal::ExceptionHandler
|
class Kemal::ExceptionHandler
|
||||||
exclude ["/api/v1/*"]
|
{% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
|
||||||
|
exclude ["/api/v1/*"], {{method}}
|
||||||
|
{% end %}
|
||||||
|
|
||||||
private def call_exception_with_status_code(context : HTTP::Server::Context, exception : Exception, status_code : Int32)
|
private def call_exception_with_status_code(context : HTTP::Server::Context, exception : Exception, status_code : Int32)
|
||||||
return if context.response.closed?
|
return if context.response.closed?
|
||||||
@@ -53,37 +61,95 @@ class Kemal::ExceptionHandler
|
|||||||
end
|
end
|
||||||
|
|
||||||
class FilteredCompressHandler < Kemal::Handler
|
class FilteredCompressHandler < Kemal::Handler
|
||||||
exclude ["/videoplayback", "/videoplayback/*", "/vi/*", "/ggpht/*"]
|
exclude ["/videoplayback", "/videoplayback/*", "/vi/*", "/ggpht/*", "/api/v1/auth/notifications"]
|
||||||
|
exclude ["/api/v1/auth/notifications", "/data_control"], "POST"
|
||||||
|
|
||||||
def call(env)
|
def call(env)
|
||||||
return call_next env if exclude_match? env
|
return call_next env if exclude_match? env
|
||||||
|
|
||||||
{% if flag?(:without_zlib) %}
|
{% if flag?(:without_zlib) %}
|
||||||
call_next env
|
call_next env
|
||||||
{% else %}
|
{% else %}
|
||||||
request_headers = env.request.headers
|
request_headers = env.request.headers
|
||||||
|
|
||||||
if request_headers.includes_word?("Accept-Encoding", "gzip")
|
if request_headers.includes_word?("Accept-Encoding", "gzip")
|
||||||
env.response.headers["Content-Encoding"] = "gzip"
|
env.response.headers["Content-Encoding"] = "gzip"
|
||||||
env.response.output = Gzip::Writer.new(env.response.output, sync_close: true)
|
env.response.output = Gzip::Writer.new(env.response.output, sync_close: true)
|
||||||
elsif request_headers.includes_word?("Accept-Encoding", "deflate")
|
elsif request_headers.includes_word?("Accept-Encoding", "deflate")
|
||||||
env.response.headers["Content-Encoding"] = "deflate"
|
env.response.headers["Content-Encoding"] = "deflate"
|
||||||
env.response.output = Flate::Writer.new(env.response.output, sync_close: true)
|
env.response.output = Flate::Writer.new(env.response.output, sync_close: true)
|
||||||
|
end
|
||||||
|
|
||||||
|
call_next env
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class AuthHandler < Kemal::Handler
|
||||||
|
{% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
|
||||||
|
only ["/api/v1/auth/*"], {{method}}
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
def call(env)
|
||||||
|
return call_next env unless only_match? env
|
||||||
|
|
||||||
|
begin
|
||||||
|
if token = env.request.headers["Authorization"]?
|
||||||
|
token = JSON.parse(URI.decode_www_form(token.lchop("Bearer ")))
|
||||||
|
session = URI.decode_www_form(token["session"].as_s)
|
||||||
|
scopes, expire, signature = validate_request(token, session, env.request, HMAC_KEY, PG_DB, nil)
|
||||||
|
|
||||||
|
if email = PG_DB.query_one?("SELECT email FROM session_ids WHERE id = $1", session, as: String)
|
||||||
|
user = PG_DB.query_one("SELECT * FROM users WHERE email = $1", email, as: User)
|
||||||
|
end
|
||||||
|
elsif sid = env.request.cookies["SID"]?.try &.value
|
||||||
|
if sid.starts_with? "v1:"
|
||||||
|
raise "Cannot use token as SID"
|
||||||
end
|
end
|
||||||
|
|
||||||
call_next env
|
if email = PG_DB.query_one?("SELECT email FROM session_ids WHERE id = $1", sid, as: String)
|
||||||
{% end %}
|
user = PG_DB.query_one("SELECT * FROM users WHERE email = $1", email, as: User)
|
||||||
|
end
|
||||||
|
|
||||||
|
scopes = [":*"]
|
||||||
|
session = sid
|
||||||
|
end
|
||||||
|
|
||||||
|
if !user
|
||||||
|
raise "Request must be authenticated"
|
||||||
|
end
|
||||||
|
|
||||||
|
env.set "scopes", scopes
|
||||||
|
env.set "user", user
|
||||||
|
env.set "session", session
|
||||||
|
|
||||||
|
call_next env
|
||||||
|
rescue ex
|
||||||
|
env.response.content_type = "application/json"
|
||||||
|
|
||||||
|
error_message = {"error" => ex.message}.to_json
|
||||||
|
env.response.status_code = 403
|
||||||
|
env.response.print error_message
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
class APIHandler < Kemal::Handler
|
class APIHandler < Kemal::Handler
|
||||||
only ["/api/v1/*"]
|
{% for method in %w(GET POST PUT HEAD DELETE PATCH OPTIONS) %}
|
||||||
|
only ["/api/v1/*"], {{method}}
|
||||||
|
{% end %}
|
||||||
|
exclude ["/api/v1/auth/notifications"], "GET"
|
||||||
|
exclude ["/api/v1/auth/notifications"], "POST"
|
||||||
|
|
||||||
def call(env)
|
def call(env)
|
||||||
return call_next env unless only_match? env
|
return call_next env unless only_match? env
|
||||||
|
|
||||||
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
env.response.headers["Access-Control-Allow-Origin"] = "*"
|
||||||
|
|
||||||
|
# Since /api/v1/notifications is an event-stream, we don't want
|
||||||
|
# to wrap the response
|
||||||
|
return call_next env if exclude_match? env
|
||||||
|
|
||||||
# Here we swap out the socket IO so we can modify the response as needed
|
# Here we swap out the socket IO so we can modify the response as needed
|
||||||
output = env.response.output
|
output = env.response.output
|
||||||
env.response.output = IO::Memory.new
|
env.response.output = IO::Memory.new
|
||||||
@@ -92,13 +158,12 @@ class APIHandler < Kemal::Handler
|
|||||||
call_next env
|
call_next env
|
||||||
|
|
||||||
env.response.output.rewind
|
env.response.output.rewind
|
||||||
response = env.response.output.gets_to_end
|
|
||||||
|
|
||||||
if env.response.headers["Content-Type"]?.try &.== "application/json"
|
if env.response.output.as(IO::Memory).size != 0 &&
|
||||||
response = JSON.parse(response)
|
env.response.headers.includes_word?("Content-Type", "application/json")
|
||||||
|
response = JSON.parse(env.response.output)
|
||||||
|
|
||||||
if env.params.query["fields"]?
|
if fields_text = env.params.query["fields"]?
|
||||||
fields_text = env.params.query["fields"]
|
|
||||||
begin
|
begin
|
||||||
JSONFilter.filter(response, fields_text)
|
JSONFilter.filter(response, fields_text)
|
||||||
rescue ex
|
rescue ex
|
||||||
@@ -107,16 +172,30 @@ class APIHandler < Kemal::Handler
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if env.params.query["pretty"]? && env.params.query["pretty"] == "1"
|
if env.params.query["pretty"]?.try &.== "1"
|
||||||
|
response = response.to_pretty_json
|
||||||
|
else
|
||||||
|
response = response.to_json
|
||||||
|
end
|
||||||
|
else
|
||||||
|
response = env.response.output.gets_to_end
|
||||||
|
end
|
||||||
|
rescue ex
|
||||||
|
env.response.content_type = "application/json" if env.response.headers.includes_word?("Content-Type", "text/html")
|
||||||
|
env.response.status_code = 500
|
||||||
|
|
||||||
|
if env.response.headers.includes_word?("Content-Type", "application/json")
|
||||||
|
response = {"error" => ex.message || "Unspecified error"}
|
||||||
|
|
||||||
|
if env.params.query["pretty"]?.try &.== "1"
|
||||||
response = response.to_pretty_json
|
response = response.to_pretty_json
|
||||||
else
|
else
|
||||||
response = response.to_json
|
response = response.to_json
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
rescue
|
|
||||||
ensure
|
ensure
|
||||||
env.response.output = output
|
env.response.output = output
|
||||||
env.response.puts response
|
env.response.print response
|
||||||
|
|
||||||
env.response.flush
|
env.response.flush
|
||||||
end
|
end
|
||||||
@@ -134,10 +213,28 @@ class DenyFrame < Kemal::Handler
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Temp fix for https://github.com/crystal-lang/crystal/issues/7383
|
# Temp fixes for https://github.com/crystal-lang/crystal/issues/7383
|
||||||
|
class HTTP::UnknownLengthContent
|
||||||
|
def read_byte
|
||||||
|
ensure_send_continue
|
||||||
|
if @io.is_a?(OpenSSL::SSL::Socket::Client)
|
||||||
|
return if @io.as(OpenSSL::SSL::Socket::Client).@in_buffer_rem.empty?
|
||||||
|
end
|
||||||
|
@io.read_byte
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
class HTTP::Client
|
class HTTP::Client
|
||||||
private def handle_response(response)
|
private def handle_response(response)
|
||||||
# close unless response.keep_alive?
|
if @socket.is_a?(OpenSSL::SSL::Socket::Client) && @host.ends_with?("googlevideo.com")
|
||||||
|
close unless response.keep_alive? || @socket.as(OpenSSL::SSL::Socket::Client).@in_buffer_rem.empty?
|
||||||
|
|
||||||
|
if @socket.as(OpenSSL::SSL::Socket::Client).@in_buffer_rem.empty?
|
||||||
|
@socket = nil
|
||||||
|
end
|
||||||
|
else
|
||||||
|
close unless response.keep_alive?
|
||||||
|
end
|
||||||
response
|
response
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -7,8 +7,24 @@ def translate(locale : Hash(String, JSON::Any) | Nil, translation : String, text
|
|||||||
# puts "Could not find translation for #{translation.dump}"
|
# puts "Could not find translation for #{translation.dump}"
|
||||||
# end
|
# end
|
||||||
|
|
||||||
if locale && locale[translation]? && !locale[translation].as_s.empty?
|
if locale && locale[translation]?
|
||||||
translation = locale[translation].as_s
|
case locale[translation]
|
||||||
|
when .as_h?
|
||||||
|
match_length = 0
|
||||||
|
|
||||||
|
locale[translation].as_h.each do |key, value|
|
||||||
|
if md = text.try &.match(/#{key}/)
|
||||||
|
if md[0].size >= match_length
|
||||||
|
translation = value.as_s
|
||||||
|
match_length = md[0].size
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
when .as_s?
|
||||||
|
if !locale[translation].as_s.empty?
|
||||||
|
translation = locale[translation].as_s
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if text
|
if text
|
||||||
@@ -17,3 +33,12 @@ def translate(locale : Hash(String, JSON::Any) | Nil, translation : String, text
|
|||||||
|
|
||||||
return translation
|
return translation
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def translate_bool(locale : Hash(String, JSON::Any) | Nil, translation : Bool)
|
||||||
|
case translation
|
||||||
|
when true
|
||||||
|
return translate(locale, "Yes")
|
||||||
|
when false
|
||||||
|
return translate(locale, "No")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|||||||
318
src/invidious/helpers/jobs.cr
Normal file
318
src/invidious/helpers/jobs.cr
Normal file
@@ -0,0 +1,318 @@
|
|||||||
|
def refresh_channels(db, logger, config)
|
||||||
|
max_channel = Channel(Int32).new
|
||||||
|
|
||||||
|
spawn do
|
||||||
|
max_threads = max_channel.receive
|
||||||
|
active_threads = 0
|
||||||
|
active_channel = Channel(Bool).new
|
||||||
|
|
||||||
|
loop do
|
||||||
|
db.query("SELECT id FROM channels ORDER BY updated") do |rs|
|
||||||
|
rs.each do
|
||||||
|
id = rs.read(String)
|
||||||
|
|
||||||
|
if active_threads >= max_threads
|
||||||
|
if active_channel.receive
|
||||||
|
active_threads -= 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
active_threads += 1
|
||||||
|
spawn do
|
||||||
|
begin
|
||||||
|
channel = fetch_channel(id, db, config.full_refresh)
|
||||||
|
|
||||||
|
db.exec("UPDATE channels SET updated = $1, author = $2, deleted = false WHERE id = $3", Time.utc, channel.author, id)
|
||||||
|
rescue ex
|
||||||
|
if ex.message == "Deleted or invalid channel"
|
||||||
|
db.exec("UPDATE channels SET updated = $1, deleted = true WHERE id = $2", Time.utc, id)
|
||||||
|
end
|
||||||
|
logger.puts("#{id} : #{ex.message}")
|
||||||
|
end
|
||||||
|
|
||||||
|
active_channel.send(true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
max_channel.send(config.channel_threads)
|
||||||
|
end
|
||||||
|
|
||||||
|
def refresh_feeds(db, logger, config)
|
||||||
|
max_channel = Channel(Int32).new
|
||||||
|
spawn do
|
||||||
|
max_threads = max_channel.receive
|
||||||
|
active_threads = 0
|
||||||
|
active_channel = Channel(Bool).new
|
||||||
|
|
||||||
|
loop do
|
||||||
|
db.query("SELECT email FROM users WHERE feed_needs_update = true OR feed_needs_update IS NULL") do |rs|
|
||||||
|
rs.each do
|
||||||
|
email = rs.read(String)
|
||||||
|
view_name = "subscriptions_#{sha256(email)}"
|
||||||
|
|
||||||
|
if active_threads >= max_threads
|
||||||
|
if active_channel.receive
|
||||||
|
active_threads -= 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
active_threads += 1
|
||||||
|
spawn do
|
||||||
|
begin
|
||||||
|
# Drop outdated views
|
||||||
|
column_array = get_column_array(db, view_name)
|
||||||
|
ChannelVideo.to_type_tuple.each_with_index do |name, i|
|
||||||
|
if name != column_array[i]?
|
||||||
|
logger.puts("DROP MATERIALIZED VIEW #{view_name}")
|
||||||
|
db.exec("DROP MATERIALIZED VIEW #{view_name}")
|
||||||
|
raise "view does not exist"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if !db.query_one("SELECT pg_get_viewdef('#{view_name}')", as: String).includes? "WHERE ((cv.ucid = ANY (u.subscriptions))"
|
||||||
|
logger.puts("Materialized view #{view_name} is out-of-date, recreating...")
|
||||||
|
db.exec("DROP MATERIALIZED VIEW #{view_name}")
|
||||||
|
end
|
||||||
|
|
||||||
|
db.exec("REFRESH MATERIALIZED VIEW #{view_name}")
|
||||||
|
db.exec("UPDATE users SET feed_needs_update = false WHERE email = $1", email)
|
||||||
|
rescue ex
|
||||||
|
# Rename old views
|
||||||
|
begin
|
||||||
|
legacy_view_name = "subscriptions_#{sha256(email)[0..7]}"
|
||||||
|
|
||||||
|
db.exec("SELECT * FROM #{legacy_view_name} LIMIT 0")
|
||||||
|
logger.puts("RENAME MATERIALIZED VIEW #{legacy_view_name}")
|
||||||
|
db.exec("ALTER MATERIALIZED VIEW #{legacy_view_name} RENAME TO #{view_name}")
|
||||||
|
rescue ex
|
||||||
|
begin
|
||||||
|
# While iterating through, we may have an email stored from a deleted account
|
||||||
|
if db.query_one?("SELECT true FROM users WHERE email = $1", email, as: Bool)
|
||||||
|
logger.puts("CREATE #{view_name}")
|
||||||
|
db.exec("CREATE MATERIALIZED VIEW #{view_name} AS #{MATERIALIZED_VIEW_SQL.call(email)}")
|
||||||
|
db.exec("UPDATE users SET feed_needs_update = false WHERE email = $1", email)
|
||||||
|
end
|
||||||
|
rescue ex
|
||||||
|
logger.puts("REFRESH #{email} : #{ex.message}")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
active_channel.send(true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
sleep 5.seconds
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
max_channel.send(config.feed_threads)
|
||||||
|
end
|
||||||
|
|
||||||
|
def subscribe_to_feeds(db, logger, key, config)
|
||||||
|
if config.use_pubsub_feeds
|
||||||
|
case config.use_pubsub_feeds
|
||||||
|
when Bool
|
||||||
|
max_threads = config.use_pubsub_feeds.as(Bool).to_unsafe
|
||||||
|
when Int32
|
||||||
|
max_threads = config.use_pubsub_feeds.as(Int32)
|
||||||
|
end
|
||||||
|
max_channel = Channel(Int32).new
|
||||||
|
|
||||||
|
client_pool = HTTPPool.new(PUBSUB_URL, capacity: max_threads, timeout: 0.05)
|
||||||
|
|
||||||
|
spawn do
|
||||||
|
max_threads = max_channel.receive
|
||||||
|
active_threads = 0
|
||||||
|
active_channel = Channel(Bool).new
|
||||||
|
|
||||||
|
loop do
|
||||||
|
db.query_all("SELECT id FROM channels WHERE CURRENT_TIMESTAMP - subscribed > interval '4 days' OR subscribed IS NULL") do |rs|
|
||||||
|
rs.each do
|
||||||
|
ucid = rs.read(String)
|
||||||
|
|
||||||
|
if active_threads >= max_threads.as(Int32)
|
||||||
|
if active_channel.receive
|
||||||
|
active_threads -= 1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
active_threads += 1
|
||||||
|
|
||||||
|
spawn do
|
||||||
|
begin
|
||||||
|
response = subscribe_pubsub(ucid, key, config, client_pool)
|
||||||
|
|
||||||
|
if response.status_code >= 400
|
||||||
|
logger.puts("#{ucid} : #{response.body}")
|
||||||
|
end
|
||||||
|
rescue ex
|
||||||
|
logger.puts("#{ucid} : #{ex.message}")
|
||||||
|
end
|
||||||
|
|
||||||
|
active_channel.send(true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
max_channel.send(max_threads.as(Int32))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def pull_top_videos(config, db)
|
||||||
|
loop do
|
||||||
|
begin
|
||||||
|
top = rank_videos(db, 40)
|
||||||
|
rescue ex
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
|
||||||
|
next
|
||||||
|
end
|
||||||
|
|
||||||
|
if top.size == 0
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
|
||||||
|
next
|
||||||
|
end
|
||||||
|
|
||||||
|
videos = [] of Video
|
||||||
|
|
||||||
|
top.each do |id|
|
||||||
|
begin
|
||||||
|
videos << get_video(id, db)
|
||||||
|
rescue ex
|
||||||
|
next
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
yield videos
|
||||||
|
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def pull_popular_videos(db)
|
||||||
|
loop do
|
||||||
|
videos = db.query_all("SELECT DISTINCT ON (ucid) * FROM channel_videos WHERE ucid IN \
|
||||||
|
(SELECT channel FROM (SELECT UNNEST(subscriptions) AS channel FROM users) AS d \
|
||||||
|
GROUP BY channel ORDER BY COUNT(channel) DESC LIMIT 40) \
|
||||||
|
ORDER BY ucid, published DESC", as: ChannelVideo).sort_by { |video| video.published }.reverse
|
||||||
|
|
||||||
|
yield videos
|
||||||
|
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def update_decrypt_function
|
||||||
|
loop do
|
||||||
|
begin
|
||||||
|
decrypt_function = fetch_decrypt_function
|
||||||
|
yield decrypt_function
|
||||||
|
rescue ex
|
||||||
|
next
|
||||||
|
ensure
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def bypass_captcha(captcha_key, logger)
|
||||||
|
loop do
|
||||||
|
begin
|
||||||
|
response = YT_POOL.client &.get("/watch?v=CvFH_6DNRCY&gl=US&hl=en&disable_polymer=1&has_verified=1&bpctr=9999999999")
|
||||||
|
if response.body.includes?("To continue with your YouTube experience, please fill out the form below.")
|
||||||
|
html = XML.parse_html(response.body)
|
||||||
|
form = html.xpath_node(%(//form[@action="/das_captcha"])).not_nil!
|
||||||
|
site_key = form.xpath_node(%(.//div[@class="g-recaptcha"])).try &.["data-sitekey"]
|
||||||
|
|
||||||
|
inputs = {} of String => String
|
||||||
|
form.xpath_nodes(%(.//input[@name])).map do |node|
|
||||||
|
inputs[node["name"]] = node["value"]
|
||||||
|
end
|
||||||
|
|
||||||
|
headers = response.cookies.add_request_headers(HTTP::Headers.new)
|
||||||
|
|
||||||
|
response = JSON.parse(HTTP::Client.post("https://api.anti-captcha.com/createTask", body: {
|
||||||
|
"clientKey" => CONFIG.captcha_key,
|
||||||
|
"task" => {
|
||||||
|
"type" => "NoCaptchaTaskProxyless",
|
||||||
|
# "type" => "NoCaptchaTask",
|
||||||
|
"websiteURL" => "https://www.youtube.com/watch?v=CvFH_6DNRCY&gl=US&hl=en&disable_polymer=1&has_verified=1&bpctr=9999999999",
|
||||||
|
"websiteKey" => site_key,
|
||||||
|
# "proxyType" => "http",
|
||||||
|
# "proxyAddress" => CONFIG.proxy_address,
|
||||||
|
# "proxyPort" => CONFIG.proxy_port,
|
||||||
|
# "proxyLogin" => CONFIG.proxy_user,
|
||||||
|
# "proxyPassword" => CONFIG.proxy_pass,
|
||||||
|
# "userAgent" => "User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.97 Safari/537.36",
|
||||||
|
},
|
||||||
|
}.to_json).body)
|
||||||
|
|
||||||
|
if response["error"]?
|
||||||
|
raise response["error"].as_s
|
||||||
|
end
|
||||||
|
|
||||||
|
task_id = response["taskId"].as_i
|
||||||
|
|
||||||
|
loop do
|
||||||
|
sleep 10.seconds
|
||||||
|
|
||||||
|
response = JSON.parse(HTTP::Client.post("https://api.anti-captcha.com/getTaskResult", body: {
|
||||||
|
"clientKey" => CONFIG.captcha_key,
|
||||||
|
"taskId" => task_id,
|
||||||
|
}.to_json).body)
|
||||||
|
|
||||||
|
if response["status"]?.try &.== "ready"
|
||||||
|
break
|
||||||
|
elsif response["errorId"]?.try &.as_i != 0
|
||||||
|
raise response["errorDescription"].as_s
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
inputs["g-recaptcha-response"] = response["solution"]["gRecaptchaResponse"].as_s
|
||||||
|
response = YT_POOL.client &.post("/das_captcha", headers, form: inputs)
|
||||||
|
|
||||||
|
yield response.cookies.select { |cookie| cookie.name != "PREF" }
|
||||||
|
end
|
||||||
|
rescue ex
|
||||||
|
logger.puts("Exception: #{ex.message}")
|
||||||
|
ensure
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def find_working_proxies(regions)
|
||||||
|
loop do
|
||||||
|
regions.each do |region|
|
||||||
|
proxies = get_proxies(region).first(20)
|
||||||
|
proxies = proxies.map { |proxy| {ip: proxy[:ip], port: proxy[:port]} }
|
||||||
|
# proxies = filter_proxies(proxies)
|
||||||
|
|
||||||
|
yield region, proxies
|
||||||
|
end
|
||||||
|
|
||||||
|
sleep 1.minute
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,13 +1,20 @@
|
|||||||
require "logger"
|
require "logger"
|
||||||
|
|
||||||
|
enum LogLevel
|
||||||
|
Debug
|
||||||
|
Info
|
||||||
|
Warn
|
||||||
|
Error
|
||||||
|
end
|
||||||
|
|
||||||
class Invidious::LogHandler < Kemal::BaseLogHandler
|
class Invidious::LogHandler < Kemal::BaseLogHandler
|
||||||
def initialize(@io : IO = STDOUT)
|
def initialize(@io : IO = STDOUT, @level = LogLevel::Warn)
|
||||||
end
|
end
|
||||||
|
|
||||||
def call(context : HTTP::Server::Context)
|
def call(context : HTTP::Server::Context)
|
||||||
time = Time.now
|
time = Time.utc
|
||||||
call_next(context)
|
call_next(context)
|
||||||
elapsed_text = elapsed_text(Time.now - time)
|
elapsed_text = elapsed_text(Time.utc - time)
|
||||||
|
|
||||||
@io << time << ' ' << context.response.status_code << ' ' << context.request.method << ' ' << context.request.resource << ' ' << elapsed_text << '\n'
|
@io << time << ' ' << context.response.status_code << ' ' << context.request.method << ' ' << context.request.resource << ' ' << elapsed_text << '\n'
|
||||||
|
|
||||||
@@ -18,7 +25,15 @@ class Invidious::LogHandler < Kemal::BaseLogHandler
|
|||||||
context
|
context
|
||||||
end
|
end
|
||||||
|
|
||||||
def write(message : String)
|
def puts(message : String)
|
||||||
|
@io << message << '\n'
|
||||||
|
|
||||||
|
if @io.is_a? File
|
||||||
|
@io.flush
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def write(message : String, level = @level)
|
||||||
@io << message
|
@io << message
|
||||||
|
|
||||||
if @io.is_a? File
|
if @io.is_a? File
|
||||||
@@ -26,6 +41,29 @@ class Invidious::LogHandler < Kemal::BaseLogHandler
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def set_log_level(level : String)
|
||||||
|
case level.downcase
|
||||||
|
when "debug"
|
||||||
|
set_log_level(LogLevel::Debug)
|
||||||
|
when "info"
|
||||||
|
set_log_level(LogLevel::Info)
|
||||||
|
when "warn"
|
||||||
|
set_log_level(LogLevel::Warn)
|
||||||
|
when "error"
|
||||||
|
set_log_level(LogLevel::Error)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def set_log_level(level : LogLevel)
|
||||||
|
@level = level
|
||||||
|
end
|
||||||
|
|
||||||
|
{% for level in %w(debug info warn error) %}
|
||||||
|
def {{level.id}}(message : String)
|
||||||
|
puts(message, LogLevel::{{level.id.capitalize}})
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
|
||||||
private def elapsed_text(elapsed)
|
private def elapsed_text(elapsed)
|
||||||
millis = elapsed.total_milliseconds
|
millis = elapsed.total_milliseconds
|
||||||
return "#{millis.round(2)}ms" if millis >= 1
|
return "#{millis.round(2)}ms" if millis >= 1
|
||||||
|
|||||||
@@ -1,45 +1,49 @@
|
|||||||
macro db_mapping(mapping)
|
macro db_mapping(mapping)
|
||||||
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
|
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
|
||||||
end
|
end
|
||||||
|
|
||||||
def to_a
|
def to_a
|
||||||
return [{{*mapping.keys.map { |id| "@#{id}".id }}}]
|
return [ {{*mapping.keys.map { |id| "@#{id}".id }}} ]
|
||||||
end
|
end
|
||||||
|
|
||||||
DB.mapping({{mapping}})
|
def self.to_type_tuple
|
||||||
|
return { {{*mapping.keys.map { |id| "#{id}" }}} }
|
||||||
|
end
|
||||||
|
|
||||||
|
DB.mapping( {{mapping}} )
|
||||||
end
|
end
|
||||||
|
|
||||||
macro json_mapping(mapping)
|
macro json_mapping(mapping)
|
||||||
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
|
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
|
||||||
end
|
end
|
||||||
|
|
||||||
def to_a
|
def to_a
|
||||||
return [{{*mapping.keys.map { |id| "@#{id}".id }}}]
|
return [ {{*mapping.keys.map { |id| "@#{id}".id }}} ]
|
||||||
end
|
end
|
||||||
|
|
||||||
JSON.mapping({{mapping}})
|
patched_json_mapping( {{mapping}} )
|
||||||
YAML.mapping({{mapping}})
|
YAML.mapping( {{mapping}} )
|
||||||
end
|
end
|
||||||
|
|
||||||
macro yaml_mapping(mapping)
|
macro yaml_mapping(mapping)
|
||||||
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
|
def initialize({{*mapping.keys.map { |id| "@#{id}".id }}})
|
||||||
end
|
end
|
||||||
|
|
||||||
def to_a
|
def to_a
|
||||||
return [{{*mapping.keys.map { |id| "@#{id}".id }}}]
|
return [ {{*mapping.keys.map { |id| "@#{id}".id }}} ]
|
||||||
end
|
end
|
||||||
|
|
||||||
def to_tuple
|
def to_tuple
|
||||||
return { {{*mapping.keys.map { |id| "@#{id}".id }}} }
|
return { {{*mapping.keys.map { |id| "@#{id}".id }}} }
|
||||||
end
|
end
|
||||||
|
|
||||||
YAML.mapping({{mapping}})
|
YAML.mapping({{mapping}})
|
||||||
end
|
end
|
||||||
|
|
||||||
macro templated(filename, template = "template")
|
macro templated(filename, template = "template")
|
||||||
render "src/invidious/views/#{{{filename}}}.ecr", "src/invidious/views/#{{{template}}}.ecr"
|
render "src/invidious/views/#{{{filename}}}.ecr", "src/invidious/views/#{{{template}}}.ecr"
|
||||||
end
|
end
|
||||||
|
|
||||||
macro rendered(filename)
|
macro rendered(filename)
|
||||||
render "src/invidious/views/#{{{filename}}}.ecr"
|
render "src/invidious/views/#{{{filename}}}.ecr"
|
||||||
end
|
end
|
||||||
|
|||||||
166
src/invidious/helpers/patch_mapping.cr
Normal file
166
src/invidious/helpers/patch_mapping.cr
Normal file
@@ -0,0 +1,166 @@
|
|||||||
|
# Overloads https://github.com/crystal-lang/crystal/blob/0.28.0/src/json/from_json.cr#L24
|
||||||
|
def Object.from_json(string_or_io, default) : self
|
||||||
|
parser = JSON::PullParser.new(string_or_io)
|
||||||
|
new parser, default
|
||||||
|
end
|
||||||
|
|
||||||
|
# Adds configurable 'default'
|
||||||
|
macro patched_json_mapping(_properties_, strict = false)
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
{% _properties_[key] = {type: value} unless value.is_a?(HashLiteral) || value.is_a?(NamedTupleLiteral) %}
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
{% _properties_[key][:key_id] = key.id.gsub(/\?$/, "") %}
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
@{{value[:key_id]}} : {{value[:type]}}{{ (value[:nilable] ? "?" : "").id }}
|
||||||
|
|
||||||
|
{% if value[:setter] == nil ? true : value[:setter] %}
|
||||||
|
def {{value[:key_id]}}=(_{{value[:key_id]}} : {{value[:type]}}{{ (value[:nilable] ? "?" : "").id }})
|
||||||
|
@{{value[:key_id]}} = _{{value[:key_id]}}
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:getter] == nil ? true : value[:getter] %}
|
||||||
|
def {{key.id}} : {{value[:type]}}{{ (value[:nilable] ? "?" : "").id }}
|
||||||
|
@{{value[:key_id]}}
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:presence] %}
|
||||||
|
@{{value[:key_id]}}_present : Bool = false
|
||||||
|
|
||||||
|
def {{value[:key_id]}}_present?
|
||||||
|
@{{value[:key_id]}}_present
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
def initialize(%pull : ::JSON::PullParser, default = nil)
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
%var{key.id} = nil
|
||||||
|
%found{key.id} = false
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
%location = %pull.location
|
||||||
|
begin
|
||||||
|
%pull.read_begin_object
|
||||||
|
rescue exc : ::JSON::ParseException
|
||||||
|
raise ::JSON::MappingError.new(exc.message, self.class.to_s, nil, *%location, exc)
|
||||||
|
end
|
||||||
|
until %pull.kind.end_object?
|
||||||
|
%key_location = %pull.location
|
||||||
|
key = %pull.read_object_key
|
||||||
|
case key
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
when {{value[:key] || value[:key_id].stringify}}
|
||||||
|
%found{key.id} = true
|
||||||
|
begin
|
||||||
|
%var{key.id} =
|
||||||
|
{% if value[:nilable] || value[:default] != nil %} %pull.read_null_or { {% end %}
|
||||||
|
|
||||||
|
{% if value[:root] %}
|
||||||
|
%pull.on_key!({{value[:root]}}) do
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:converter] %}
|
||||||
|
{{value[:converter]}}.from_json(%pull)
|
||||||
|
{% elsif value[:type].is_a?(Path) || value[:type].is_a?(Generic) %}
|
||||||
|
{{value[:type]}}.new(%pull)
|
||||||
|
{% else %}
|
||||||
|
::Union({{value[:type]}}).new(%pull)
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:root] %}
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:nilable] || value[:default] != nil %} } {% end %}
|
||||||
|
rescue exc : ::JSON::ParseException
|
||||||
|
raise ::JSON::MappingError.new(exc.message, self.class.to_s, {{value[:key] || value[:key_id].stringify}}, *%key_location, exc)
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
else
|
||||||
|
{% if strict %}
|
||||||
|
raise ::JSON::MappingError.new("Unknown JSON attribute: #{key}", self.class.to_s, nil, *%key_location, nil)
|
||||||
|
{% else %}
|
||||||
|
%pull.skip
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
%pull.read_next
|
||||||
|
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
{% unless value[:nilable] || value[:default] != nil %}
|
||||||
|
if %var{key.id}.nil? && !%found{key.id} && !::Union({{value[:type]}}).nilable?
|
||||||
|
raise ::JSON::MappingError.new("Missing JSON attribute: {{(value[:key] || value[:key_id]).id}}", self.class.to_s, nil, *%location, nil)
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:nilable] %}
|
||||||
|
{% if value[:default] != nil %}
|
||||||
|
@{{value[:key_id]}} = %found{key.id} ? %var{key.id} : (default.responds_to?(:{{value[:key_id]}}) ? default.{{value[:key_id]}} : {{value[:default]}})
|
||||||
|
{% else %}
|
||||||
|
@{{value[:key_id]}} = %var{key.id}
|
||||||
|
{% end %}
|
||||||
|
{% elsif value[:default] != nil %}
|
||||||
|
@{{value[:key_id]}} = %var{key.id}.nil? ? (default.responds_to?(:{{value[:key_id]}}) ? default.{{value[:key_id]}} : {{value[:default]}}) : %var{key.id}
|
||||||
|
{% else %}
|
||||||
|
@{{value[:key_id]}} = (%var{key.id}).as({{value[:type]}})
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:presence] %}
|
||||||
|
@{{value[:key_id]}}_present = %found{key.id}
|
||||||
|
{% end %}
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_json(json : ::JSON::Builder)
|
||||||
|
json.object do
|
||||||
|
{% for key, value in _properties_ %}
|
||||||
|
_{{value[:key_id]}} = @{{value[:key_id]}}
|
||||||
|
|
||||||
|
{% unless value[:emit_null] %}
|
||||||
|
unless _{{value[:key_id]}}.nil?
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
json.field({{value[:key] || value[:key_id].stringify}}) do
|
||||||
|
{% if value[:root] %}
|
||||||
|
{% if value[:emit_null] %}
|
||||||
|
if _{{value[:key_id]}}.nil?
|
||||||
|
nil.to_json(json)
|
||||||
|
else
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
json.object do
|
||||||
|
json.field({{value[:root]}}) do
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:converter] %}
|
||||||
|
if _{{value[:key_id]}}
|
||||||
|
{{ value[:converter] }}.to_json(_{{value[:key_id]}}, json)
|
||||||
|
else
|
||||||
|
nil.to_json(json)
|
||||||
|
end
|
||||||
|
{% else %}
|
||||||
|
_{{value[:key_id]}}.to_json(json)
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
{% if value[:root] %}
|
||||||
|
{% if value[:emit_null] %}
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
|
||||||
|
{% unless value[:emit_null] %}
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -31,10 +31,10 @@ class HTTPProxy
|
|||||||
|
|
||||||
if resp[:code]? == 200
|
if resp[:code]? == 200
|
||||||
{% if !flag?(:without_openssl) %}
|
{% if !flag?(:without_openssl) %}
|
||||||
if tls
|
if tls
|
||||||
tls_socket = OpenSSL::SSL::Socket::Client.new(socket, context: tls, sync_close: true, hostname: host)
|
tls_socket = OpenSSL::SSL::Socket::Client.new(socket, context: tls, sync_close: true, hostname: host)
|
||||||
socket = tls_socket
|
socket = tls_socket
|
||||||
end
|
end
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
||||||
return socket
|
return socket
|
||||||
@@ -77,6 +77,10 @@ class HTTPClient < HTTP::Client
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def unset_proxy
|
||||||
|
@socket = nil
|
||||||
|
end
|
||||||
|
|
||||||
def proxy_connection_options
|
def proxy_connection_options
|
||||||
opts = {} of Symbol => Float64 | Nil
|
opts = {} of Symbol => Float64 | Nil
|
||||||
|
|
||||||
@@ -97,6 +101,7 @@ def filter_proxies(proxies)
|
|||||||
proxies.select! do |proxy|
|
proxies.select! do |proxy|
|
||||||
begin
|
begin
|
||||||
client = HTTPClient.new(YT_URL)
|
client = HTTPClient.new(YT_URL)
|
||||||
|
client.before_request { |r| add_yt_headers(r) } if url.host == "www.youtube.com"
|
||||||
client.read_timeout = 10.seconds
|
client.read_timeout = 10.seconds
|
||||||
client.connect_timeout = 10.seconds
|
client.connect_timeout = 10.seconds
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
def fetch_decrypt_function(id = "CvFH_6DNRCY")
|
def fetch_decrypt_function(id = "CvFH_6DNRCY")
|
||||||
client = make_client(YT_URL)
|
document = YT_POOL.client &.get("/watch?v=#{id}&gl=US&hl=en&disable_polymer=1").body
|
||||||
document = client.get("/watch?v=#{id}&gl=US&hl=en&disable_polymer=1").body
|
|
||||||
url = document.match(/src="(?<url>\/yts\/jsbin\/player_ias-.{9}\/en_US\/base.js)"/).not_nil!["url"]
|
url = document.match(/src="(?<url>\/yts\/jsbin\/player_ias-.{9}\/en_US\/base.js)"/).not_nil!["url"]
|
||||||
player = client.get(url).body
|
player = YT_POOL.client &.get(url).body
|
||||||
|
|
||||||
function_name = player.match(/^(?<name>[^=]+)=function\(a\){a=a\.split\(""\)/m).not_nil!["name"]
|
function_name = player.match(/^(?<name>[^=]+)=function\(a\){a=a\.split\(""\)/m).not_nil!["name"]
|
||||||
function_body = player.match(/^#{Regex.escape(function_name)}=function\(a\){(?<body>[^}]+)}/m).not_nil!["body"]
|
function_body = player.match(/^#{Regex.escape(function_name)}=function\(a\){(?<body>[^}]+)}/m).not_nil!["body"]
|
||||||
194
src/invidious/helpers/static_file_handler.cr
Normal file
194
src/invidious/helpers/static_file_handler.cr
Normal file
@@ -0,0 +1,194 @@
|
|||||||
|
# Since systems have a limit on number of open files (`ulimit -a`),
|
||||||
|
# we serve them from memory to avoid 'Too many open files' without needing
|
||||||
|
# to modify ulimit.
|
||||||
|
#
|
||||||
|
# Very heavily re-used:
|
||||||
|
# https://github.com/kemalcr/kemal/blob/master/src/kemal/helpers/helpers.cr
|
||||||
|
# https://github.com/kemalcr/kemal/blob/master/src/kemal/static_file_handler.cr
|
||||||
|
#
|
||||||
|
# Changes:
|
||||||
|
# - A `send_file` overload is added which supports sending a Slice, file_path, filestat
|
||||||
|
# - `StaticFileHandler` is patched to cache to and serve from @cached_files
|
||||||
|
|
||||||
|
private def multipart(file, env : HTTP::Server::Context)
|
||||||
|
# See http://httpwg.org/specs/rfc7233.html
|
||||||
|
fileb = file.size
|
||||||
|
startb = endb = 0
|
||||||
|
|
||||||
|
if match = env.request.headers["Range"].match /bytes=(\d{1,})-(\d{0,})/
|
||||||
|
startb = match[1].to_i { 0 } if match.size >= 2
|
||||||
|
endb = match[2].to_i { 0 } if match.size >= 3
|
||||||
|
end
|
||||||
|
|
||||||
|
endb = fileb - 1 if endb == 0
|
||||||
|
|
||||||
|
if startb < endb < fileb
|
||||||
|
content_length = 1 + endb - startb
|
||||||
|
env.response.status_code = 206
|
||||||
|
env.response.content_length = content_length
|
||||||
|
env.response.headers["Accept-Ranges"] = "bytes"
|
||||||
|
env.response.headers["Content-Range"] = "bytes #{startb}-#{endb}/#{fileb}" # MUST
|
||||||
|
|
||||||
|
if startb > 1024
|
||||||
|
skipped = 0
|
||||||
|
# file.skip only accepts values less or equal to 1024 (buffer size, undocumented)
|
||||||
|
until (increase_skipped = skipped + 1024) > startb
|
||||||
|
file.skip(1024)
|
||||||
|
skipped = increase_skipped
|
||||||
|
end
|
||||||
|
if (skipped_minus_startb = skipped - startb) > 0
|
||||||
|
file.skip skipped_minus_startb
|
||||||
|
end
|
||||||
|
else
|
||||||
|
file.skip(startb)
|
||||||
|
end
|
||||||
|
|
||||||
|
IO.copy(file, env.response, content_length)
|
||||||
|
else
|
||||||
|
env.response.content_length = fileb
|
||||||
|
env.response.status_code = 200 # Range not satisfable, see 4.4 Note
|
||||||
|
IO.copy(file, env.response)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Set the Content-Disposition to "attachment" with the specified filename,
|
||||||
|
# instructing the user agents to prompt to save.
|
||||||
|
private def attachment(env : HTTP::Server::Context, filename : String? = nil, disposition : String? = nil)
|
||||||
|
disposition = "attachment" if disposition.nil? && filename
|
||||||
|
if disposition && filename
|
||||||
|
env.response.headers["Content-Disposition"] = "#{disposition}; filename=\"#{File.basename(filename)}\""
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def send_file(env : HTTP::Server::Context, file_path : String, data : Slice(UInt8), filestat : File::Info, filename : String? = nil, disposition : String? = nil)
|
||||||
|
config = Kemal.config.serve_static
|
||||||
|
mime_type = MIME.from_filename(file_path, "application/octet-stream")
|
||||||
|
env.response.content_type = mime_type
|
||||||
|
env.response.headers["Accept-Ranges"] = "bytes"
|
||||||
|
env.response.headers["X-Content-Type-Options"] = "nosniff"
|
||||||
|
minsize = 860 # http://webmasters.stackexchange.com/questions/31750/what-is-recommended-minimum-object-size-for-gzip-performance-benefits ??
|
||||||
|
request_headers = env.request.headers
|
||||||
|
filesize = data.bytesize
|
||||||
|
attachment(env, filename, disposition)
|
||||||
|
|
||||||
|
Kemal.config.static_headers.try(&.call(env.response, file_path, filestat))
|
||||||
|
|
||||||
|
file = IO::Memory.new(data)
|
||||||
|
if env.request.method == "GET" && env.request.headers.has_key?("Range")
|
||||||
|
return multipart(file, env)
|
||||||
|
end
|
||||||
|
|
||||||
|
condition = config.is_a?(Hash) && config["gzip"]? == true && filesize > minsize && Kemal::Utils.zip_types(file_path)
|
||||||
|
if condition && request_headers.includes_word?("Accept-Encoding", "gzip")
|
||||||
|
env.response.headers["Content-Encoding"] = "gzip"
|
||||||
|
Gzip::Writer.open(env.response) do |deflate|
|
||||||
|
IO.copy(file, deflate)
|
||||||
|
end
|
||||||
|
elsif condition && request_headers.includes_word?("Accept-Encoding", "deflate")
|
||||||
|
env.response.headers["Content-Encoding"] = "deflate"
|
||||||
|
Flate::Writer.open(env.response) do |deflate|
|
||||||
|
IO.copy(file, deflate)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
env.response.content_length = filesize
|
||||||
|
IO.copy(file, env.response)
|
||||||
|
end
|
||||||
|
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
module Kemal
|
||||||
|
class StaticFileHandler < HTTP::StaticFileHandler
|
||||||
|
CACHE_LIMIT = 5_000_000 # 5MB
|
||||||
|
@cached_files = {} of String => {data: Bytes, filestat: File::Info}
|
||||||
|
|
||||||
|
def call(context : HTTP::Server::Context)
|
||||||
|
return call_next(context) if context.request.path.not_nil! == "/"
|
||||||
|
|
||||||
|
case context.request.method
|
||||||
|
when "GET", "HEAD"
|
||||||
|
else
|
||||||
|
if @fallthrough
|
||||||
|
call_next(context)
|
||||||
|
else
|
||||||
|
context.response.status_code = 405
|
||||||
|
context.response.headers.add("Allow", "GET, HEAD")
|
||||||
|
end
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
config = Kemal.config.serve_static
|
||||||
|
original_path = context.request.path.not_nil!
|
||||||
|
request_path = URI.decode_www_form(original_path)
|
||||||
|
|
||||||
|
# File path cannot contains '\0' (NUL) because all filesystem I know
|
||||||
|
# don't accept '\0' character as file name.
|
||||||
|
if request_path.includes? '\0'
|
||||||
|
context.response.status_code = 400
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
expanded_path = File.expand_path(request_path, "/")
|
||||||
|
is_dir_path = if original_path.ends_with?('/') && !expanded_path.ends_with? '/'
|
||||||
|
expanded_path = expanded_path + '/'
|
||||||
|
true
|
||||||
|
else
|
||||||
|
expanded_path.ends_with? '/'
|
||||||
|
end
|
||||||
|
|
||||||
|
file_path = File.join(@public_dir, expanded_path)
|
||||||
|
|
||||||
|
if file = @cached_files[file_path]?
|
||||||
|
last_modified = file[:filestat].modification_time
|
||||||
|
add_cache_headers(context.response.headers, last_modified)
|
||||||
|
|
||||||
|
if cache_request?(context, last_modified)
|
||||||
|
context.response.status_code = 304
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
send_file(context, file_path, file[:data], file[:filestat])
|
||||||
|
else
|
||||||
|
is_dir = Dir.exists? file_path
|
||||||
|
|
||||||
|
if request_path != expanded_path
|
||||||
|
redirect_to context, expanded_path
|
||||||
|
elsif is_dir && !is_dir_path
|
||||||
|
redirect_to context, expanded_path + '/'
|
||||||
|
end
|
||||||
|
|
||||||
|
if Dir.exists?(file_path)
|
||||||
|
if config.is_a?(Hash) && config["dir_listing"] == true
|
||||||
|
context.response.content_type = "text/html"
|
||||||
|
directory_listing(context.response, request_path, file_path)
|
||||||
|
else
|
||||||
|
call_next(context)
|
||||||
|
end
|
||||||
|
elsif File.exists?(file_path)
|
||||||
|
last_modified = modification_time(file_path)
|
||||||
|
add_cache_headers(context.response.headers, last_modified)
|
||||||
|
|
||||||
|
if cache_request?(context, last_modified)
|
||||||
|
context.response.status_code = 304
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
if @cached_files.sum { |element| element[1][:data].bytesize } + (size = File.size(file_path)) < CACHE_LIMIT
|
||||||
|
data = Bytes.new(size)
|
||||||
|
File.open(file_path) do |file|
|
||||||
|
file.read(data)
|
||||||
|
end
|
||||||
|
filestat = File.info(file_path)
|
||||||
|
|
||||||
|
@cached_files[file_path] = {data: data, filestat: filestat}
|
||||||
|
send_file(context, file_path, data, filestat)
|
||||||
|
else
|
||||||
|
send_file(context, file_path)
|
||||||
|
end
|
||||||
|
else
|
||||||
|
call_next(context)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
146
src/invidious/helpers/tokens.cr
Normal file
146
src/invidious/helpers/tokens.cr
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
def generate_token(email, scopes, expire, key, db)
|
||||||
|
session = "v1:#{Base64.urlsafe_encode(Random::Secure.random_bytes(32))}"
|
||||||
|
PG_DB.exec("INSERT INTO session_ids VALUES ($1, $2, $3)", session, email, Time.utc)
|
||||||
|
|
||||||
|
token = {
|
||||||
|
"session" => session,
|
||||||
|
"scopes" => scopes,
|
||||||
|
"expire" => expire,
|
||||||
|
}
|
||||||
|
|
||||||
|
if !expire
|
||||||
|
token.delete("expire")
|
||||||
|
end
|
||||||
|
|
||||||
|
token["signature"] = sign_token(key, token)
|
||||||
|
|
||||||
|
return token.to_json
|
||||||
|
end
|
||||||
|
|
||||||
|
def generate_response(session, scopes, key, db, expire = 6.hours, use_nonce = false)
|
||||||
|
expire = Time.utc + expire
|
||||||
|
|
||||||
|
token = {
|
||||||
|
"session" => session,
|
||||||
|
"expire" => expire.to_unix,
|
||||||
|
"scopes" => scopes,
|
||||||
|
}
|
||||||
|
|
||||||
|
if use_nonce
|
||||||
|
nonce = Random::Secure.hex(16)
|
||||||
|
db.exec("INSERT INTO nonces VALUES ($1, $2) ON CONFLICT DO NOTHING", nonce, expire)
|
||||||
|
token["nonce"] = nonce
|
||||||
|
end
|
||||||
|
|
||||||
|
token["signature"] = sign_token(key, token)
|
||||||
|
|
||||||
|
return token.to_json
|
||||||
|
end
|
||||||
|
|
||||||
|
def sign_token(key, hash)
|
||||||
|
string_to_sign = [] of String
|
||||||
|
|
||||||
|
hash.each do |key, value|
|
||||||
|
if key == "signature"
|
||||||
|
next
|
||||||
|
end
|
||||||
|
|
||||||
|
if value.is_a?(JSON::Any)
|
||||||
|
case value
|
||||||
|
when .as_a?
|
||||||
|
value = value.as_a.map { |item| item.as_s }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
case value
|
||||||
|
when Array
|
||||||
|
string_to_sign << "#{key}=#{value.sort.join(",")}"
|
||||||
|
when Tuple
|
||||||
|
string_to_sign << "#{key}=#{value.to_a.sort.join(",")}"
|
||||||
|
else
|
||||||
|
string_to_sign << "#{key}=#{value}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
string_to_sign = string_to_sign.sort.join("\n")
|
||||||
|
return Base64.urlsafe_encode(OpenSSL::HMAC.digest(:sha256, key, string_to_sign)).strip
|
||||||
|
end
|
||||||
|
|
||||||
|
def validate_request(token, session, request, key, db, locale = nil)
|
||||||
|
case token
|
||||||
|
when String
|
||||||
|
token = JSON.parse(URI.decode_www_form(token)).as_h
|
||||||
|
when JSON::Any
|
||||||
|
token = token.as_h
|
||||||
|
when Nil
|
||||||
|
raise translate(locale, "Hidden field \"token\" is a required field")
|
||||||
|
end
|
||||||
|
|
||||||
|
if token["signature"] != sign_token(key, token)
|
||||||
|
raise translate(locale, "Invalid signature")
|
||||||
|
end
|
||||||
|
|
||||||
|
if token["session"] != session
|
||||||
|
raise translate(locale, "Erroneous token")
|
||||||
|
end
|
||||||
|
|
||||||
|
if token["nonce"]? && (nonce = db.query_one?("SELECT * FROM nonces WHERE nonce = $1", token["nonce"], as: {String, Time}))
|
||||||
|
if nonce[1] > Time.utc
|
||||||
|
db.exec("UPDATE nonces SET expire = $1 WHERE nonce = $2", Time.utc(1990, 1, 1), nonce[0])
|
||||||
|
else
|
||||||
|
raise translate(locale, "Erroneous token")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
scopes = token["scopes"].as_a.map { |v| v.as_s }
|
||||||
|
scope = "#{request.method}:#{request.path.lchop("/api/v1/auth/").lstrip("/")}"
|
||||||
|
|
||||||
|
if !scopes_include_scope(scopes, scope)
|
||||||
|
raise translate(locale, "Invalid scope")
|
||||||
|
end
|
||||||
|
|
||||||
|
expire = token["expire"]?.try &.as_i
|
||||||
|
if expire.try &.< Time.utc.to_unix
|
||||||
|
raise translate(locale, "Token is expired, please try again")
|
||||||
|
end
|
||||||
|
|
||||||
|
return {scopes, expire, token["signature"].as_s}
|
||||||
|
end
|
||||||
|
|
||||||
|
def scope_includes_scope(scope, subset)
|
||||||
|
methods, endpoint = scope.split(":")
|
||||||
|
methods = methods.split(";").map { |method| method.upcase }.reject { |method| method.empty? }.sort
|
||||||
|
endpoint = endpoint.downcase
|
||||||
|
|
||||||
|
subset_methods, subset_endpoint = subset.split(":")
|
||||||
|
subset_methods = subset_methods.split(";").map { |method| method.upcase }.sort
|
||||||
|
subset_endpoint = subset_endpoint.downcase
|
||||||
|
|
||||||
|
if methods.empty?
|
||||||
|
methods = %w(GET POST PUT HEAD DELETE PATCH OPTIONS)
|
||||||
|
end
|
||||||
|
|
||||||
|
if methods & subset_methods != subset_methods
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
if endpoint.ends_with?("*") && !subset_endpoint.starts_with? endpoint.rchop("*")
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
if !endpoint.ends_with?("*") && subset_endpoint != endpoint
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
|
def scopes_include_scope(scopes, subset)
|
||||||
|
scopes.each do |scope|
|
||||||
|
if scope_includes_scope(scope, subset)
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return false
|
||||||
|
end
|
||||||
@@ -1,3 +1,113 @@
|
|||||||
|
require "lsquic"
|
||||||
|
require "pool/connection"
|
||||||
|
|
||||||
|
def add_yt_headers(request)
|
||||||
|
request.headers["x-youtube-client-name"] ||= "1"
|
||||||
|
request.headers["x-youtube-client-version"] ||= "1.20180719"
|
||||||
|
request.headers["user-agent"] ||= "Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/78.0.3904.97 Safari/537.36"
|
||||||
|
request.headers["accept-charset"] ||= "ISO-8859-1,utf-8;q=0.7,*;q=0.7"
|
||||||
|
request.headers["accept"] ||= "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"
|
||||||
|
request.headers["accept-language"] ||= "en-us,en;q=0.5"
|
||||||
|
request.headers["cookie"] = "#{(CONFIG.cookies.map { |c| "#{c.name}=#{c.value}" }).join("; ")}; #{request.headers["cookie"]?}"
|
||||||
|
end
|
||||||
|
|
||||||
|
struct HTTPPool
|
||||||
|
property! url : URI
|
||||||
|
property! capacity : Int32
|
||||||
|
property! timeout : Float64
|
||||||
|
property pool : ConnectionPool(HTTPClient)
|
||||||
|
|
||||||
|
def initialize(url : URI, @capacity = 5, @timeout = 5.0)
|
||||||
|
@url = url
|
||||||
|
@pool = build_pool
|
||||||
|
end
|
||||||
|
|
||||||
|
def client(region = nil, &block)
|
||||||
|
conn = pool.checkout
|
||||||
|
|
||||||
|
begin
|
||||||
|
if region
|
||||||
|
PROXY_LIST[region]?.try &.sample(40).each do |proxy|
|
||||||
|
begin
|
||||||
|
proxy = HTTPProxy.new(proxy_host: proxy[:ip], proxy_port: proxy[:port])
|
||||||
|
conn.set_proxy(proxy)
|
||||||
|
break
|
||||||
|
rescue ex
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
response = yield conn
|
||||||
|
|
||||||
|
if region
|
||||||
|
conn.unset_proxy
|
||||||
|
end
|
||||||
|
|
||||||
|
response
|
||||||
|
rescue ex
|
||||||
|
conn = HTTPClient.new(url)
|
||||||
|
conn.before_request { |r| add_yt_headers(r) } if url.host == "www.youtube.com"
|
||||||
|
conn.family = (url.host == "www.youtube.com" || url.host == "suggestqueries.google.com") ? CONFIG.force_resolve : Socket::Family::UNSPEC
|
||||||
|
conn.read_timeout = 10.seconds
|
||||||
|
conn.connect_timeout = 10.seconds
|
||||||
|
yield conn
|
||||||
|
ensure
|
||||||
|
pool.checkin(conn)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private def build_pool
|
||||||
|
ConnectionPool(HTTPClient).new(capacity: capacity, timeout: timeout) do
|
||||||
|
client = HTTPClient.new(url)
|
||||||
|
client.before_request { |r| add_yt_headers(r) } if url.host == "www.youtube.com"
|
||||||
|
client.family = (url.host == "www.youtube.com" || url.host == "suggestqueries.google.com") ? CONFIG.force_resolve : Socket::Family::UNSPEC
|
||||||
|
client.read_timeout = 10.seconds
|
||||||
|
client.connect_timeout = 10.seconds
|
||||||
|
client
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
struct QUICPool
|
||||||
|
property! url : URI
|
||||||
|
property! capacity : Int32
|
||||||
|
property! timeout : Float64
|
||||||
|
|
||||||
|
def initialize(url : URI, @capacity = 5, @timeout = 5.0)
|
||||||
|
@url = url
|
||||||
|
end
|
||||||
|
|
||||||
|
def client(region = nil, &block)
|
||||||
|
begin
|
||||||
|
if region
|
||||||
|
client = HTTPClient.new(url)
|
||||||
|
client.before_request { |r| add_yt_headers(r) } if url.host == "www.youtube.com"
|
||||||
|
client.read_timeout = 10.seconds
|
||||||
|
client.connect_timeout = 10.seconds
|
||||||
|
|
||||||
|
PROXY_LIST[region]?.try &.sample(40).each do |proxy|
|
||||||
|
begin
|
||||||
|
proxy = HTTPProxy.new(proxy_host: proxy[:ip], proxy_port: proxy[:port])
|
||||||
|
client.set_proxy(proxy)
|
||||||
|
break
|
||||||
|
rescue ex
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
yield client
|
||||||
|
else
|
||||||
|
conn = QUIC::Client.new(url)
|
||||||
|
conn.before_request { |r| add_yt_headers(r) } if url.host == "www.youtube.com"
|
||||||
|
yield conn
|
||||||
|
end
|
||||||
|
rescue ex
|
||||||
|
conn = QUIC::Client.new(url)
|
||||||
|
conn.before_request { |r| add_yt_headers(r) } if url.host == "www.youtube.com"
|
||||||
|
yield conn
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
# See http://www.evanmiller.org/how-not-to-sort-by-average-rating.html
|
# See http://www.evanmiller.org/how-not-to-sort-by-average-rating.html
|
||||||
def ci_lower_bound(pos, n)
|
def ci_lower_bound(pos, n)
|
||||||
if n == 0
|
if n == 0
|
||||||
@@ -18,19 +128,14 @@ def elapsed_text(elapsed)
|
|||||||
"#{(millis * 1000).round(2)}µs"
|
"#{(millis * 1000).round(2)}µs"
|
||||||
end
|
end
|
||||||
|
|
||||||
def make_client(url, proxies = {} of String => Array({ip: String, port: Int32}), region = nil)
|
def make_client(url : URI, region = nil)
|
||||||
context = OpenSSL::SSL::Context::Client.new
|
client = HTTPClient.new(url)
|
||||||
context.add_options(
|
client.family = (url.host == "www.youtube.com") ? CONFIG.force_resolve : Socket::Family::UNSPEC
|
||||||
OpenSSL::SSL::Options::ALL |
|
|
||||||
OpenSSL::SSL::Options::NO_SSL_V2 |
|
|
||||||
OpenSSL::SSL::Options::NO_SSL_V3
|
|
||||||
)
|
|
||||||
client = HTTPClient.new(url, context)
|
|
||||||
client.read_timeout = 10.seconds
|
client.read_timeout = 10.seconds
|
||||||
client.connect_timeout = 10.seconds
|
client.connect_timeout = 10.seconds
|
||||||
|
|
||||||
if region
|
if region
|
||||||
proxies[region]?.try &.sample(40).each do |proxy|
|
PROXY_LIST[region]?.try &.sample(40).each do |proxy|
|
||||||
begin
|
begin
|
||||||
proxy = HTTPProxy.new(proxy_host: proxy[:ip], proxy_port: proxy[:port])
|
proxy = HTTPProxy.new(proxy_host: proxy[:ip], proxy_port: proxy[:port])
|
||||||
client.set_proxy(proxy)
|
client.set_proxy(proxy)
|
||||||
@@ -44,7 +149,7 @@ def make_client(url, proxies = {} of String => Array({ip: String, port: Int32}),
|
|||||||
end
|
end
|
||||||
|
|
||||||
def decode_length_seconds(string)
|
def decode_length_seconds(string)
|
||||||
length_seconds = string.split(":").map { |a| a.to_i }
|
length_seconds = string.gsub(/[^0-9:]/, "").split(":").map &.to_i
|
||||||
length_seconds = [0] * (3 - length_seconds.size) + length_seconds
|
length_seconds = [0] * (3 - length_seconds.size) + length_seconds
|
||||||
length_seconds = Time::Span.new(length_seconds[0], length_seconds[1], length_seconds[2])
|
length_seconds = Time::Span.new(length_seconds[0], length_seconds[1], length_seconds[2])
|
||||||
length_seconds = length_seconds.total_seconds.to_i
|
length_seconds = length_seconds.total_seconds.to_i
|
||||||
@@ -59,8 +164,8 @@ def recode_length_seconds(time)
|
|||||||
time = time.seconds
|
time = time.seconds
|
||||||
text = "#{time.minutes.to_s.rjust(2, '0')}:#{time.seconds.to_s.rjust(2, '0')}"
|
text = "#{time.minutes.to_s.rjust(2, '0')}:#{time.seconds.to_s.rjust(2, '0')}"
|
||||||
|
|
||||||
if time.hours > 0
|
if time.total_hours.to_i > 0
|
||||||
text = "#{time.hours.to_s.rjust(2, '0')}:#{text}"
|
text = "#{time.total_hours.to_i.to_s.rjust(2, '0')}:#{text}"
|
||||||
end
|
end
|
||||||
|
|
||||||
text = text.lchop('0')
|
text = text.lchop('0')
|
||||||
@@ -85,7 +190,7 @@ def decode_time(string)
|
|||||||
millis = /(?<millis>\d+)ms/.match(string).try &.["millis"].try &.to_f
|
millis = /(?<millis>\d+)ms/.match(string).try &.["millis"].try &.to_f
|
||||||
millis ||= 0
|
millis ||= 0
|
||||||
|
|
||||||
time = hours * 3600 + minutes * 60 + seconds + millis / 1000
|
time = hours * 3600 + minutes * 60 + seconds + millis // 1000
|
||||||
end
|
end
|
||||||
|
|
||||||
return time
|
return time
|
||||||
@@ -94,7 +199,7 @@ end
|
|||||||
def decode_date(string : String)
|
def decode_date(string : String)
|
||||||
# String matches 'YYYY'
|
# String matches 'YYYY'
|
||||||
if string.match(/^\d{4}/)
|
if string.match(/^\d{4}/)
|
||||||
return Time.new(string.to_i, 1, 1)
|
return Time.utc(string.to_i, 1, 1)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Try to parse as format Jul 10, 2000
|
# Try to parse as format Jul 10, 2000
|
||||||
@@ -105,9 +210,9 @@ def decode_date(string : String)
|
|||||||
|
|
||||||
case string
|
case string
|
||||||
when "today"
|
when "today"
|
||||||
return Time.now
|
return Time.utc
|
||||||
when "yesterday"
|
when "yesterday"
|
||||||
return Time.now - 1.day
|
return Time.utc - 1.day
|
||||||
end
|
end
|
||||||
|
|
||||||
# String matches format "20 hours ago", "4 months ago"...
|
# String matches format "20 hours ago", "4 months ago"...
|
||||||
@@ -133,18 +238,18 @@ def decode_date(string : String)
|
|||||||
raise "Could not parse #{string}"
|
raise "Could not parse #{string}"
|
||||||
end
|
end
|
||||||
|
|
||||||
return Time.now - delta
|
return Time.utc - delta
|
||||||
end
|
end
|
||||||
|
|
||||||
def recode_date(time : Time, locale)
|
def recode_date(time : Time, locale)
|
||||||
span = Time.now - time
|
span = Time.utc - time
|
||||||
|
|
||||||
if span.total_days > 365.0
|
if span.total_days > 365.0
|
||||||
span = translate(locale, "`x` years", (span.total_days.to_i / 365).to_s)
|
span = translate(locale, "`x` years", (span.total_days.to_i // 365).to_s)
|
||||||
elsif span.total_days > 30.0
|
elsif span.total_days > 30.0
|
||||||
span = translate(locale, "`x` months", (span.total_days.to_i / 30).to_s)
|
span = translate(locale, "`x` months", (span.total_days.to_i // 30).to_s)
|
||||||
elsif span.total_days > 7.0
|
elsif span.total_days > 7.0
|
||||||
span = translate(locale, "`x` weeks", (span.total_days.to_i / 7).to_s)
|
span = translate(locale, "`x` weeks", (span.total_days.to_i // 7).to_s)
|
||||||
elsif span.total_hours > 24.0
|
elsif span.total_hours > 24.0
|
||||||
span = translate(locale, "`x` days", (span.total_days.to_i).to_s)
|
span = translate(locale, "`x` days", (span.total_days.to_i).to_s)
|
||||||
elsif span.total_minutes > 60.0
|
elsif span.total_minutes > 60.0
|
||||||
@@ -162,7 +267,7 @@ def number_with_separator(number)
|
|||||||
number.to_s.reverse.gsub(/(\d{3})(?=\d)/, "\\1,").reverse
|
number.to_s.reverse.gsub(/(\d{3})(?=\d)/, "\\1,").reverse
|
||||||
end
|
end
|
||||||
|
|
||||||
def short_text_to_number(short_text)
|
def short_text_to_number(short_text : String) : Int32
|
||||||
case short_text
|
case short_text
|
||||||
when .ends_with? "M"
|
when .ends_with? "M"
|
||||||
number = short_text.rstrip(" mM").to_f
|
number = short_text.rstrip(" mM").to_f
|
||||||
@@ -189,9 +294,11 @@ def number_to_short_text(number)
|
|||||||
|
|
||||||
text = text.rchop(".0")
|
text = text.rchop(".0")
|
||||||
|
|
||||||
if number / 1000000 != 0
|
if number // 1_000_000_000 != 0
|
||||||
|
text += "B"
|
||||||
|
elsif number // 1_000_000 != 0
|
||||||
text += "M"
|
text += "M"
|
||||||
elsif number / 1000 != 0
|
elsif number // 1000 != 0
|
||||||
text += "K"
|
text += "K"
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -236,7 +343,7 @@ def make_host_url(config, kemal_config)
|
|||||||
return "#{scheme}#{host}#{port}"
|
return "#{scheme}#{host}#{port}"
|
||||||
end
|
end
|
||||||
|
|
||||||
def get_referer(env, fallback = "/")
|
def get_referer(env, fallback = "/", unroll = true)
|
||||||
referer = env.params.query["referer"]?
|
referer = env.params.query["referer"]?
|
||||||
referer ||= env.request.headers["referer"]?
|
referer ||= env.request.headers["referer"]?
|
||||||
referer ||= fallback
|
referer ||= fallback
|
||||||
@@ -244,16 +351,18 @@ def get_referer(env, fallback = "/")
|
|||||||
referer = URI.parse(referer)
|
referer = URI.parse(referer)
|
||||||
|
|
||||||
# "Unroll" nested referrers
|
# "Unroll" nested referrers
|
||||||
loop do
|
if unroll
|
||||||
if referer.query
|
loop do
|
||||||
params = HTTP::Params.parse(referer.query.not_nil!)
|
if referer.query
|
||||||
if params["referer"]?
|
params = HTTP::Params.parse(referer.query.not_nil!)
|
||||||
referer = URI.parse(URI.unescape(params["referer"]))
|
if params["referer"]?
|
||||||
|
referer = URI.parse(URI.decode_www_form(params["referer"]))
|
||||||
|
else
|
||||||
|
break
|
||||||
|
end
|
||||||
else
|
else
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
else
|
|
||||||
break
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -267,50 +376,41 @@ def get_referer(env, fallback = "/")
|
|||||||
return referer
|
return referer
|
||||||
end
|
end
|
||||||
|
|
||||||
def read_var_int(bytes)
|
struct VarInt
|
||||||
num_read = 0
|
def self.from_io(io : IO, format = IO::ByteFormat::NetworkEndian) : Int32
|
||||||
result = 0
|
result = 0_u32
|
||||||
|
num_read = 0
|
||||||
|
|
||||||
read = bytes[num_read]
|
loop do
|
||||||
|
byte = io.read_byte
|
||||||
if bytes.size == 1
|
raise "Invalid VarInt" if !byte
|
||||||
result = bytes[0].to_i32
|
value = byte & 0x7f
|
||||||
else
|
|
||||||
while ((read & 0b10000000) != 0)
|
|
||||||
read = bytes[num_read].to_u64
|
|
||||||
value = (read & 0b01111111)
|
|
||||||
result |= (value << (7 * num_read))
|
|
||||||
|
|
||||||
|
result |= value.to_u32 << (7 * num_read)
|
||||||
num_read += 1
|
num_read += 1
|
||||||
if num_read > 5
|
|
||||||
raise "VarInt is too big"
|
break if byte & 0x80 == 0
|
||||||
end
|
raise "Invalid VarInt" if num_read > 5
|
||||||
end
|
end
|
||||||
|
|
||||||
|
result.to_i32
|
||||||
end
|
end
|
||||||
|
|
||||||
return result
|
def self.to_io(io : IO, value : Int32)
|
||||||
end
|
io.write_byte 0x00 if value == 0x00
|
||||||
|
value = value.to_u32
|
||||||
|
|
||||||
def write_var_int(value : Int)
|
|
||||||
bytes = [] of UInt8
|
|
||||||
value = value.to_u32
|
|
||||||
|
|
||||||
if value == 0
|
|
||||||
bytes = [0_u8]
|
|
||||||
else
|
|
||||||
while value != 0
|
while value != 0
|
||||||
temp = (value & 0b01111111).to_u8
|
byte = (value & 0x7f).to_u8
|
||||||
value = value >> 7
|
value >>= 7
|
||||||
|
|
||||||
if value != 0
|
if value != 0
|
||||||
temp |= 0b10000000
|
byte |= 0x80
|
||||||
end
|
end
|
||||||
|
|
||||||
bytes << temp
|
io.write_byte byte
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
return Slice.new(bytes.to_unsafe, bytes.size)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def sha256(text)
|
def sha256(text)
|
||||||
@@ -318,3 +418,64 @@ def sha256(text)
|
|||||||
digest << text
|
digest << text
|
||||||
return digest.hexdigest
|
return digest.hexdigest
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def subscribe_pubsub(topic, key, config, client_pool)
|
||||||
|
case topic
|
||||||
|
when .match(/^UC[A-Za-z0-9_-]{22}$/)
|
||||||
|
topic = "channel_id=#{topic}"
|
||||||
|
when .match(/^(PL|LL|EC|UU|FL|UL|OLAK5uy_)[0-9A-Za-z-_]{10,}$/)
|
||||||
|
# There's a couple missing from the above regex, namely TL and RD, which
|
||||||
|
# don't have feeds
|
||||||
|
topic = "playlist_id=#{topic}"
|
||||||
|
else
|
||||||
|
# TODO
|
||||||
|
end
|
||||||
|
|
||||||
|
time = Time.utc.to_unix.to_s
|
||||||
|
nonce = Random::Secure.hex(4)
|
||||||
|
signature = "#{time}:#{nonce}"
|
||||||
|
|
||||||
|
host_url = make_host_url(config, Kemal.config)
|
||||||
|
|
||||||
|
body = {
|
||||||
|
"hub.callback" => "#{host_url}/feed/webhook/v1:#{time}:#{nonce}:#{OpenSSL::HMAC.hexdigest(:sha1, key, signature)}",
|
||||||
|
"hub.topic" => "https://www.youtube.com/xml/feeds/videos.xml?#{topic}",
|
||||||
|
"hub.verify" => "async",
|
||||||
|
"hub.mode" => "subscribe",
|
||||||
|
"hub.lease_seconds" => "432000",
|
||||||
|
"hub.secret" => key.to_s,
|
||||||
|
}
|
||||||
|
|
||||||
|
return client_pool.client &.post("/subscribe", form: body)
|
||||||
|
end
|
||||||
|
|
||||||
|
def parse_range(range)
|
||||||
|
if !range
|
||||||
|
return 0_i64, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
ranges = range.lchop("bytes=").split(',')
|
||||||
|
ranges.each do |range|
|
||||||
|
start_range, end_range = range.split('-')
|
||||||
|
|
||||||
|
start_range = start_range.to_i64? || 0_i64
|
||||||
|
end_range = end_range.to_i64?
|
||||||
|
|
||||||
|
return start_range, end_range
|
||||||
|
end
|
||||||
|
|
||||||
|
return 0_i64, nil
|
||||||
|
end
|
||||||
|
|
||||||
|
def convert_theme(theme)
|
||||||
|
case theme
|
||||||
|
when "true"
|
||||||
|
"dark"
|
||||||
|
when "false"
|
||||||
|
"light"
|
||||||
|
when "", nil
|
||||||
|
nil
|
||||||
|
else
|
||||||
|
theme
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|||||||
@@ -1,225 +0,0 @@
|
|||||||
def refresh_channels(db, logger, max_threads = 1, full_refresh = false)
|
|
||||||
max_channel = Channel(Int32).new
|
|
||||||
|
|
||||||
spawn do
|
|
||||||
max_threads = max_channel.receive
|
|
||||||
active_threads = 0
|
|
||||||
active_channel = Channel(Bool).new
|
|
||||||
|
|
||||||
loop do
|
|
||||||
db.query("SELECT id FROM channels ORDER BY updated") do |rs|
|
|
||||||
rs.each do
|
|
||||||
id = rs.read(String)
|
|
||||||
|
|
||||||
if active_threads >= max_threads
|
|
||||||
if active_channel.receive
|
|
||||||
active_threads -= 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
active_threads += 1
|
|
||||||
spawn do
|
|
||||||
begin
|
|
||||||
channel = fetch_channel(id, db, full_refresh)
|
|
||||||
|
|
||||||
db.exec("UPDATE channels SET updated = $1, author = $2, deleted = false WHERE id = $3", Time.now, channel.author, id)
|
|
||||||
rescue ex
|
|
||||||
if ex.message == "Deleted or invalid channel"
|
|
||||||
db.exec("UPDATE channels SET updated = $1, deleted = true WHERE id = $2", Time.now, id)
|
|
||||||
end
|
|
||||||
logger.write("#{id} : #{ex.message}\n")
|
|
||||||
end
|
|
||||||
|
|
||||||
active_channel.send(true)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
max_channel.send(max_threads)
|
|
||||||
end
|
|
||||||
|
|
||||||
def refresh_feeds(db, logger, max_threads = 1)
|
|
||||||
max_channel = Channel(Int32).new
|
|
||||||
|
|
||||||
spawn do
|
|
||||||
max_threads = max_channel.receive
|
|
||||||
active_threads = 0
|
|
||||||
active_channel = Channel(Bool).new
|
|
||||||
|
|
||||||
loop do
|
|
||||||
db.query("SELECT email FROM users") do |rs|
|
|
||||||
rs.each do
|
|
||||||
email = rs.read(String)
|
|
||||||
view_name = "subscriptions_#{sha256(email)[0..7]}"
|
|
||||||
|
|
||||||
if active_threads >= max_threads
|
|
||||||
if active_channel.receive
|
|
||||||
active_threads -= 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
active_threads += 1
|
|
||||||
spawn do
|
|
||||||
begin
|
|
||||||
db.query("SELECT * FROM #{view_name} LIMIT 1") do |rs|
|
|
||||||
# View doesn't contain same number of rows as ChannelVideo
|
|
||||||
if ChannelVideo.from_rs(rs)[0]?.try &.to_a.size.try &.!= rs.column_count
|
|
||||||
db.exec("DROP MATERIALIZED VIEW #{view_name}")
|
|
||||||
raise "valid schema does not exist"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
db.exec("REFRESH MATERIALIZED VIEW #{view_name}")
|
|
||||||
rescue ex
|
|
||||||
# Create view if it doesn't exist
|
|
||||||
if ex.message.try &.ends_with?("does not exist")
|
|
||||||
# While iterating through, we may have an email stored from a deleted account
|
|
||||||
if db.query_one?("SELECT true FROM users WHERE email = $1", email, as: Bool)
|
|
||||||
db.exec("CREATE MATERIALIZED VIEW #{view_name} AS \
|
|
||||||
SELECT * FROM channel_videos WHERE \
|
|
||||||
ucid = ANY ((SELECT subscriptions FROM users WHERE email = E'#{email.gsub("'", "\\'")}')::text[]) \
|
|
||||||
ORDER BY published DESC;")
|
|
||||||
logger.write("CREATE #{view_name}\n")
|
|
||||||
end
|
|
||||||
else
|
|
||||||
logger.write("REFRESH #{email} : #{ex.message}\n")
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
active_channel.send(true)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
max_channel.send(max_threads)
|
|
||||||
end
|
|
||||||
|
|
||||||
def subscribe_to_feeds(db, logger, key, config)
|
|
||||||
if config.use_pubsub_feeds
|
|
||||||
case config.use_pubsub_feeds
|
|
||||||
when Bool
|
|
||||||
max_threads = config.use_pubsub_feeds.as(Bool).to_unsafe
|
|
||||||
when Int32
|
|
||||||
max_threads = config.use_pubsub_feeds.as(Int32)
|
|
||||||
end
|
|
||||||
max_channel = Channel(Int32).new
|
|
||||||
|
|
||||||
spawn do
|
|
||||||
max_threads = max_channel.receive
|
|
||||||
active_threads = 0
|
|
||||||
active_channel = Channel(Bool).new
|
|
||||||
|
|
||||||
loop do
|
|
||||||
db.query_all("SELECT id FROM channels WHERE CURRENT_TIMESTAMP - subscribed > interval '4 days' OR subscribed IS NULL") do |rs|
|
|
||||||
rs.each do
|
|
||||||
ucid = rs.read(String)
|
|
||||||
|
|
||||||
if active_threads >= max_threads.as(Int32)
|
|
||||||
if active_channel.receive
|
|
||||||
active_threads -= 1
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
active_threads += 1
|
|
||||||
|
|
||||||
spawn do
|
|
||||||
begin
|
|
||||||
response = subscribe_pubsub(ucid, key, config)
|
|
||||||
|
|
||||||
if response.status_code >= 400
|
|
||||||
logger.write("#{ucid} : #{response.body}\n")
|
|
||||||
end
|
|
||||||
rescue ex
|
|
||||||
end
|
|
||||||
|
|
||||||
active_channel.send(true)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
max_channel.send(max_threads.as(Int32))
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def pull_top_videos(config, db)
|
|
||||||
loop do
|
|
||||||
begin
|
|
||||||
top = rank_videos(db, 40)
|
|
||||||
rescue ex
|
|
||||||
next
|
|
||||||
end
|
|
||||||
|
|
||||||
if top.size > 0
|
|
||||||
args = arg_array(top)
|
|
||||||
else
|
|
||||||
next
|
|
||||||
end
|
|
||||||
|
|
||||||
videos = [] of Video
|
|
||||||
|
|
||||||
top.each do |id|
|
|
||||||
begin
|
|
||||||
videos << get_video(id, db)
|
|
||||||
rescue ex
|
|
||||||
next
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
yield videos
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def pull_popular_videos(db)
|
|
||||||
loop do
|
|
||||||
subscriptions = db.query_all("SELECT channel FROM \
|
|
||||||
(SELECT UNNEST(subscriptions) AS channel FROM users) AS d \
|
|
||||||
GROUP BY channel ORDER BY COUNT(channel) DESC LIMIT 40", as: String)
|
|
||||||
|
|
||||||
videos = db.query_all("SELECT DISTINCT ON (ucid) * FROM \
|
|
||||||
channel_videos WHERE ucid IN (#{arg_array(subscriptions)}) \
|
|
||||||
ORDER BY ucid, published DESC", subscriptions, as: ChannelVideo).sort_by { |video| video.published }.reverse
|
|
||||||
|
|
||||||
yield videos
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def update_decrypt_function
|
|
||||||
loop do
|
|
||||||
begin
|
|
||||||
decrypt_function = fetch_decrypt_function
|
|
||||||
rescue ex
|
|
||||||
next
|
|
||||||
end
|
|
||||||
|
|
||||||
yield decrypt_function
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def find_working_proxies(regions)
|
|
||||||
loop do
|
|
||||||
regions.each do |region|
|
|
||||||
proxies = get_proxies(region).first(20)
|
|
||||||
proxies = proxies.map { |proxy| {ip: proxy[:ip], port: proxy[:port]} }
|
|
||||||
# proxies = filter_proxies(proxies)
|
|
||||||
|
|
||||||
yield region, proxies
|
|
||||||
end
|
|
||||||
|
|
||||||
sleep 1.minute
|
|
||||||
end
|
|
||||||
end
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user