Compare commits
373 commits
Author | SHA1 | Date | |
---|---|---|---|
8d7ca9a4e2 | |||
9a66a7bd51 | |||
66b481713d | |||
6587528ed9 | |||
917cede8b7 | |||
fc0a3ab307 | |||
62d64ca814 | |||
e78f7e5430 | |||
|
b551fcf96a | ||
7689251158 | |||
b3a8866022 | |||
eac85f111c | |||
5dd37bfee7 | |||
ab32c38719 | |||
cf6c3a7b5b | |||
2f5a555ea7 | |||
472dd8663d | |||
dc2aba106c | |||
eff8673efc | |||
b1f25a69ad | |||
d5b8b0b19c | |||
|
b3e6aaddab | ||
33ffafb9e3 | |||
|
a88a723de3 | ||
|
d5f5490aee | ||
|
82d797b74e | ||
|
97895a491a | ||
|
0ac9367322 | ||
|
d3830f7870 | ||
|
3cfcc16403 | ||
|
171c0a0814 | ||
|
82ac9a8609 | ||
|
7c79ee7cc2 | ||
|
f6e09250cd | ||
|
0fecde6917 | ||
|
66f5b12ecd | ||
|
77f57714ea | ||
|
d9afe38504 | ||
|
3af11d800c | ||
|
d72531d843 | ||
|
ecfcad8d1c | ||
|
d63b15dc1c | ||
|
edb69d601e | ||
|
51562f4b24 | ||
|
76f045b8d7 | ||
|
46eaa0f9b8 | ||
|
56bccaba77 | ||
|
4e8d03221b | ||
|
5d46eba6f2 | ||
|
d3eedab545 | ||
|
cd43997bba | ||
|
fead7603e6 | ||
|
486b5b363c | ||
|
2b3619e489 | ||
|
7a95cb43ef | ||
|
e09a7de5c7 | ||
|
79d1aaff1a | ||
|
d7a5ca8fff | ||
|
542d4fe553 | ||
|
33df8249f1 | ||
|
4e7fd7ac3b | ||
|
8912e2448d | ||
|
98f1e4170b | ||
|
b384133dc9 | ||
|
1961fc3b11 | ||
|
2e649363d2 | ||
|
53e8a5d62d | ||
|
a021b93063 | ||
|
d9df90b5e3 | ||
|
cec3cfba77 | ||
|
de918b9234 | ||
8f46bd5751 | |||
cae8cbdda8 | |||
100ecff0b3 | |||
|
5e899d73a9 | ||
|
f247b2f862 | ||
|
bd34659ff6 | ||
|
f1baeef4bc | ||
|
157c4c3e98 | ||
|
4782a67038 | ||
|
5baaedfa39 | ||
|
4f066e880c | ||
|
3e17d04875 | ||
|
cec905e95e | ||
|
80958aa0d8 | ||
5ced7694fe | |||
|
c5fdd9ea65 | ||
|
2876ee0f9f | ||
|
0699e5fc27 | ||
|
15669acccf | ||
|
cd2daf4adb | ||
|
ccecc6d318 | ||
|
3c6a662aaf | ||
|
9e55799269 | ||
|
da70c9b7b0 | ||
|
828da3c6ce | ||
|
febf18cbf7 | ||
|
b2133c6b2c | ||
ae937d8339 | |||
|
21ab5dc668 | ||
|
b200ebfb6b | ||
|
ecbea0b67b | ||
|
d1cd790388 | ||
|
f66068976e | ||
|
22b35c453e | ||
4a2877f28b | |||
|
c606465708 | ||
|
85deea5aca | ||
|
78c5ba93c7 | ||
|
31a80420ec | ||
|
4c0b5c314d | ||
|
eb0f651812 | ||
|
764965c441 | ||
|
b795bdf2a4 | ||
|
5b05f3bd14 | ||
|
a335bc0814 | ||
|
7b50388eaf | ||
|
da3d58f03c | ||
|
8327862697 | ||
|
6878822c4d | ||
|
0b28054f8a | ||
|
cc33d3f074 | ||
|
acbb625866 | ||
|
466bfbb306 | ||
|
e476dbe25b | ||
2f8ef155c8 | |||
|
e31053e812 | ||
|
96ade642fa | ||
|
e319c35f09 | ||
|
2d18ff1f80 | ||
|
2d7869b48b | ||
|
88b9f17388 | ||
|
7a7d1137d6 | ||
|
3add83c49e | ||
|
a8825a27d4 | ||
|
bedcf97fbf | ||
|
89c17f2127 | ||
|
905fed66d1 | ||
|
f837d99eab | ||
|
86ec5ad6e0 | ||
|
ae93146f47 | ||
|
e538410262 | ||
|
8ad19f06ee | ||
|
366732b4fd | ||
|
32ea9cfe16 | ||
|
53a60bf7bd | ||
|
2d485b18a4 | ||
|
5cb1688c78 | ||
|
456b00a699 | ||
|
84aded85c5 | ||
|
e99b591855 | ||
|
7cf7cce0b2 | ||
|
f842033eb5 | ||
|
c9fb19431d | ||
|
4b8bfe1201 | ||
|
e6c39f9e3a | ||
|
5e38ef59da | ||
|
80ffc442f2 | ||
|
9bf754ed4f | ||
|
fa6c5158c5 | ||
|
b45310c7d4 | ||
|
eb2dfe0ab1 | ||
6c2626cf05 | |||
72150ae676 | |||
b7430c5a5a | |||
|
cc36a82933 | ||
|
7798faf234 | ||
|
ec1bb5db87 | ||
|
3b7e45b7bc | ||
|
e098c27a45 | ||
|
90e94d4e6c | ||
|
6506b8dbfc | ||
|
61d75050e4 | ||
|
10e5788c21 | ||
|
b509aa91d5 | ||
|
ec8b7916fa | ||
|
56a7488161 | ||
|
a845752fff | ||
|
63a729998b | ||
|
205f988491 | ||
|
0db3b830b7 | ||
|
c8fb75e6fd | ||
|
636a6d0be2 | ||
|
3415507e4a | ||
|
53223f99b0 | ||
ddfb8e7d93 | |||
|
b0cd6587bd | ||
c7b8f470d8 | |||
762fa5214d | |||
d50990ea15 | |||
067dcbef5e | |||
a7e9602ccd | |||
a74057bb7a | |||
9a7b6976ff | |||
|
6e1e3e9554 | ||
ad591f3c32 | |||
|
8665a69fee | ||
|
d1051efd6e | ||
03bf4592ce | |||
d641bcbf5d | |||
2027a35e5a | |||
|
65d9468911 | ||
|
389a2a4a4d | ||
|
b60e056f96 | ||
|
532d92bb7a | ||
|
24f878e6f6 | ||
|
a9fc84bc14 | ||
|
7e680c692f | ||
|
1eb28edfb3 | ||
|
efcd94ffbe | ||
|
780f9df7d3 | ||
|
4d11c324b0 | ||
|
57f8bfb965 | ||
|
6acabc5bff | ||
|
9d0ab0a83c | ||
|
4164159057 | ||
|
30d858bc8b | ||
|
e98aafa4b5 | ||
|
325561e755 | ||
|
09bf09befe | ||
|
7fdbda612f | ||
|
4f60feee17 | ||
|
733bd27a5c | ||
|
1ff0775f4b | ||
|
e62d4db752 | ||
|
8b1da2001e | ||
|
5a12005b48 | ||
|
8575794bad | ||
|
fad0a4f52d | ||
|
fa50e0abf4 | ||
|
76ab51e219 | ||
|
8258062ec5 | ||
|
8d9723d43c | ||
|
8a90add310 | ||
|
c45e710845 | ||
|
593257a750 | ||
|
bad92093bf | ||
|
436a61e3bb | ||
|
5e0f55333a | ||
|
de61b163a3 | ||
|
99c7e9e800 | ||
|
e9bab06e90 | ||
|
a56a724a55 | ||
|
0a54e26536 | ||
|
d135e5b7f7 | ||
|
911dad6935 | ||
|
220cc9bd2f | ||
|
aace30b2b4 | ||
|
64d1f26ece | ||
|
8f5c6a602b | ||
|
dd38eef41a | ||
|
848ab1e9c8 | ||
|
933802b897 | ||
|
3bac467a8c | ||
|
248df785d7 | ||
|
6b429575bf | ||
|
e0ed094cc4 | ||
|
a644d76497 | ||
|
45fd4a1968 | ||
e82c965e89 | |||
|
f466116cd7 | ||
|
5b519123a7 | ||
|
0224162ad2 | ||
|
04ca64691b | ||
|
5957523624 | ||
|
629599f940 | ||
|
31ad708206 | ||
|
1ae14cc224 | ||
|
9980c0e00f | ||
|
aa96cf3453 | ||
|
41c978d350 | ||
|
cff25a7b25 | ||
|
1ce2d10c50 | ||
|
5abafb8296 | ||
|
9cd2e93a2e | ||
|
3b773c4f77 | ||
|
57e606cb43 | ||
|
f57aac5815 | ||
|
71a821a7e6 | ||
e0d0dbde3c | |||
90fcf80a8d | |||
|
c4fec89a9b | ||
9d66676f2d | |||
|
f696f96824 | ||
|
5b11ca22d0 | ||
|
2fdb6dd644 | ||
|
470245de54 | ||
|
b0ec359028 | ||
|
eda7444ca4 | ||
|
79b342aee5 | ||
|
0ab6d9e748 | ||
|
6f889dc695 | ||
|
8eaaeb1c74 | ||
|
7c1d2714e0 | ||
|
c94c6f4b83 | ||
|
f1fd197cbc | ||
|
bd549f21e9 | ||
|
48ad864572 | ||
|
bcb679e653 | ||
|
b163afecb7 | ||
|
5855e9d188 | ||
|
33f316c864 | ||
|
be291e8f0f | ||
|
d49c762609 | ||
|
7f3ddad12e | ||
|
6db4a46c5f | ||
|
01e2a5e89d | ||
|
bff0b5c85a | ||
|
2da63bf36d | ||
|
7546cb511d | ||
|
200cfd7579 | ||
|
6ed872d72b | ||
|
58dc63671a | ||
|
eba0699c48 | ||
|
e3018e00c4 | ||
|
b54d45504f | ||
|
64eef948bd | ||
|
3f9c7b6c19 | ||
|
c60d2561d1 | ||
|
0de3b0a96d | ||
|
5551b613d3 | ||
|
ef7f3f5bd4 | ||
|
c8369f9dbb | ||
|
8d75d6431a | ||
|
a6bcf0280c | ||
|
97c4263530 | ||
|
dd01b0f5eb | ||
|
197b3972a9 | ||
|
cbbaded209 | ||
|
066b1c35cc | ||
|
25cbfd0681 | ||
|
a2f9707b3f | ||
|
89c008211d | ||
|
e92d250a1c | ||
|
668c130f01 | ||
|
f7ae680c25 | ||
|
24de19d06f | ||
|
f313162fa1 | ||
|
b90cf286fc | ||
|
2b6e71b553 | ||
|
d1eae10147 | ||
|
fbf07e18aa | ||
|
de2287963f | ||
|
039212ed91 | ||
|
a9f55aa310 | ||
|
c27bb90e4d | ||
|
b673695aa2 | ||
|
990931ff67 | ||
|
bfd9c9876e | ||
|
2a029b4d8c | ||
|
170eef58fd | ||
|
a9e8aabe1f | ||
|
b0c6bdf44c | ||
|
c5eb10b21f | ||
|
08390acd0c | ||
|
1a2d408d38 | ||
|
72fe8af850 | ||
|
0aaa3e6a08 | ||
|
499aed37dd | ||
|
5ceeefa236 | ||
|
4adb4c00d2 | ||
|
72bcd3cc72 | ||
|
cf61af67ab | ||
|
1363fb8094 | ||
|
5f2b43d653 | ||
|
6251d8d43f | ||
|
162b89d942 | ||
|
0d63ad5a7f | ||
|
63e5d72466 | ||
|
c251c66748 | ||
|
b0df3774db | ||
|
824cc1a5aa | ||
|
bf470704a5 |
152 changed files with 5806 additions and 3206 deletions
78
.ameba.yml
78
.ameba.yml
|
@ -20,6 +20,13 @@ Lint/ShadowingOuterLocalVar:
|
||||||
Excluded:
|
Excluded:
|
||||||
- src/invidious/helpers/tokens.cr
|
- src/invidious/helpers/tokens.cr
|
||||||
|
|
||||||
|
Lint/NotNil:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
Lint/SpecFilename:
|
||||||
|
Excluded:
|
||||||
|
- spec/parsers_helper.cr
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# Style
|
# Style
|
||||||
|
@ -31,6 +38,29 @@ Style/RedundantBegin:
|
||||||
Style/RedundantReturn:
|
Style/RedundantReturn:
|
||||||
Enabled: false
|
Enabled: false
|
||||||
|
|
||||||
|
Style/RedundantNext:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
Style/ParenthesesAroundCondition:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
# This requires a rewrite of most data structs (and their usage) in Invidious.
|
||||||
|
Naming/QueryBoolMethods:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
Naming/AccessorMethodName:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
Naming/BlockParameterName:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
# Hides TODO comment warnings.
|
||||||
|
#
|
||||||
|
# Call `bin/ameba --only Documentation/DocumentationAdmonition` to
|
||||||
|
# list them
|
||||||
|
Documentation/DocumentationAdmonition:
|
||||||
|
Enabled: false
|
||||||
|
|
||||||
|
|
||||||
#
|
#
|
||||||
# Metrics
|
# Metrics
|
||||||
|
@ -39,50 +69,4 @@ Style/RedundantReturn:
|
||||||
# Ignore function complexity (number of if/else & case/when branches)
|
# Ignore function complexity (number of if/else & case/when branches)
|
||||||
# For some functions that can hardly be simplified for now
|
# For some functions that can hardly be simplified for now
|
||||||
Metrics/CyclomaticComplexity:
|
Metrics/CyclomaticComplexity:
|
||||||
Excluded:
|
Enabled: false
|
||||||
# get_about_info(ucid, locale) => [17/10]
|
|
||||||
- src/invidious/channels/about.cr
|
|
||||||
|
|
||||||
# fetch_channel_community(ucid, continuation, ...) => [34/10]
|
|
||||||
- src/invidious/channels/community.cr
|
|
||||||
|
|
||||||
# create_notification_stream(env, topics, connection_channel) => [14/10]
|
|
||||||
- src/invidious/helpers/helpers.cr:84:5
|
|
||||||
|
|
||||||
# get_index(plural_form, count) => [25/10]
|
|
||||||
- src/invidious/helpers/i18next.cr
|
|
||||||
|
|
||||||
# call(context) => [18/10]
|
|
||||||
- src/invidious/helpers/static_file_handler.cr
|
|
||||||
|
|
||||||
# show(env) => [38/10]
|
|
||||||
- src/invidious/routes/embed.cr
|
|
||||||
|
|
||||||
# get_video_playback(env) => [45/10]
|
|
||||||
- src/invidious/routes/video_playback.cr
|
|
||||||
|
|
||||||
# handle(env) => [40/10]
|
|
||||||
- src/invidious/routes/watch.cr
|
|
||||||
|
|
||||||
# playlist_ajax(env) => [24/10]
|
|
||||||
- src/invidious/routes/playlists.cr
|
|
||||||
|
|
||||||
# fetch_youtube_comments(id, cursor, ....) => [40/10]
|
|
||||||
# template_youtube_comments(comments, locale, ...) => [16/10]
|
|
||||||
# content_to_comment_html(content) => [14/10]
|
|
||||||
- src/invidious/comments.cr
|
|
||||||
|
|
||||||
# to_json(locale, json) => [21/10]
|
|
||||||
# extract_video_info(video_id, ...) => [44/10]
|
|
||||||
# process_video_params(query, preferences) => [20/10]
|
|
||||||
- src/invidious/videos.cr
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#src/invidious/playlists.cr:327:5
|
|
||||||
#[C] Metrics/CyclomaticComplexity: Cyclomatic complexity too high [19/10]
|
|
||||||
# fetch_playlist(plid : String)
|
|
||||||
|
|
||||||
#src/invidious/playlists.cr:436:5
|
|
||||||
#[C] Metrics/CyclomaticComplexity: Cyclomatic complexity too high [11/10]
|
|
||||||
# extract_playlist_videos(initial_data : Hash(String, JSON::Any))
|
|
||||||
|
|
50
.forgejo/workflows/ci.yml
Normal file
50
.forgejo/workflows/ci.yml
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
name: 'Invidious CI'
|
||||||
|
|
||||||
|
on:
|
||||||
|
# workflow_dispatch:
|
||||||
|
# inputs: {}
|
||||||
|
schedule:
|
||||||
|
- cron: '0 7 * * 0'
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- "master"
|
||||||
|
- "experimental"
|
||||||
|
- "experimental2"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: runner
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: https://code.forgejo.org/actions/checkout@v2
|
||||||
|
|
||||||
|
- uses: https://code.forgejo.org/docker/setup-buildx-action@v3
|
||||||
|
name: Setup Docker BuildX system
|
||||||
|
|
||||||
|
- name: Login to Docker Container Registry
|
||||||
|
uses: https://code.forgejo.org/docker/login-action@v3.1.0
|
||||||
|
with:
|
||||||
|
registry: git.nadeko.net
|
||||||
|
username: ${{ secrets.USERNAME }}
|
||||||
|
password: ${{ secrets.TOKEN }}
|
||||||
|
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta
|
||||||
|
uses: https://github.com/docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: git.nadeko.net/fijxu/invidious
|
||||||
|
tags: |
|
||||||
|
type=sha,format=short,prefix={{date 'YYYY.MM.DD'}}-,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
||||||
|
type=raw,value=latest,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
||||||
|
|
||||||
|
- uses: https://code.forgejo.org/docker/build-push-action@v5
|
||||||
|
name: Build images
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
file: docker/Dockerfile
|
||||||
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
|
platforms: linux/amd64
|
||||||
|
push: true
|
||||||
|
build-args: |
|
||||||
|
"release=1"
|
||||||
|
|
2
.github/CODEOWNERS
vendored
2
.github/CODEOWNERS
vendored
|
@ -6,7 +6,7 @@ docker/ @unixfox
|
||||||
kubernetes/ @unixfox
|
kubernetes/ @unixfox
|
||||||
|
|
||||||
README.md @thefrenchghosty
|
README.md @thefrenchghosty
|
||||||
config/config.example.yml @thefrenchghosty @SamantazFox @unixfox
|
config/config.example.yml @SamantazFox @unixfox
|
||||||
|
|
||||||
scripts/ @syeopite
|
scripts/ @syeopite
|
||||||
shards.lock @syeopite
|
shards.lock @syeopite
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
name: Build and release container
|
name: Build and release container directly from master
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
|
@ -24,9 +24,9 @@ jobs:
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Install Crystal
|
- name: Install Crystal
|
||||||
uses: crystal-lang/install-crystal@v1.8.0
|
uses: crystal-lang/install-crystal@v1.8.2
|
||||||
with:
|
with:
|
||||||
crystal: 1.9.2
|
crystal: 1.12.2
|
||||||
|
|
||||||
- name: Run lint
|
- name: Run lint
|
||||||
run: |
|
run: |
|
||||||
|
@ -58,7 +58,7 @@ jobs:
|
||||||
images: quay.io/invidious/invidious
|
images: quay.io/invidious/invidious
|
||||||
tags: |
|
tags: |
|
||||||
type=sha,format=short,prefix={{date 'YYYY.MM.DD'}}-,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
type=sha,format=short,prefix={{date 'YYYY.MM.DD'}}-,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
||||||
type=raw,value=latest,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
type=raw,value=master,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
||||||
labels: |
|
labels: |
|
||||||
quay.expires-after=12w
|
quay.expires-after=12w
|
||||||
|
|
||||||
|
@ -83,7 +83,7 @@ jobs:
|
||||||
suffix=-arm64
|
suffix=-arm64
|
||||||
tags: |
|
tags: |
|
||||||
type=sha,format=short,prefix={{date 'YYYY.MM.DD'}}-,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
type=sha,format=short,prefix={{date 'YYYY.MM.DD'}}-,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
||||||
type=raw,value=latest,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
type=raw,value=master,enable=${{ github.ref == format('refs/heads/{0}', 'master') }}
|
||||||
labels: |
|
labels: |
|
||||||
quay.expires-after=12w
|
quay.expires-after=12w
|
||||||
|
|
94
.github/workflows/build-stable-container.yml
vendored
Normal file
94
.github/workflows/build-stable-container.yml
vendored
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
name: Build and release container
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- "v*"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
release:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Install Crystal
|
||||||
|
uses: crystal-lang/install-crystal@v1.8.2
|
||||||
|
with:
|
||||||
|
crystal: 1.12.2
|
||||||
|
|
||||||
|
- name: Run lint
|
||||||
|
run: |
|
||||||
|
if ! crystal tool format --check; then
|
||||||
|
crystal tool format
|
||||||
|
git diff
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
with:
|
||||||
|
platforms: arm64
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to registry
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
registry: quay.io
|
||||||
|
username: ${{ secrets.QUAY_USERNAME }}
|
||||||
|
password: ${{ secrets.QUAY_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: quay.io/invidious/invidious
|
||||||
|
flavor: |
|
||||||
|
latest=false
|
||||||
|
tags: |
|
||||||
|
type=semver,pattern={{version}}
|
||||||
|
type=raw,value=latest
|
||||||
|
labels: |
|
||||||
|
quay.expires-after=12w
|
||||||
|
|
||||||
|
- name: Build and push Docker AMD64 image for Push Event
|
||||||
|
uses: docker/build-push-action@v5
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
file: docker/Dockerfile
|
||||||
|
platforms: linux/amd64
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
|
push: true
|
||||||
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
|
build-args: |
|
||||||
|
"release=1"
|
||||||
|
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta-arm64
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: quay.io/invidious/invidious
|
||||||
|
flavor: |
|
||||||
|
latest=false
|
||||||
|
suffix=-arm64
|
||||||
|
tags: |
|
||||||
|
type=semver,pattern={{version}}
|
||||||
|
type=raw,value=latest
|
||||||
|
labels: |
|
||||||
|
quay.expires-after=12w
|
||||||
|
|
||||||
|
- name: Build and push Docker ARM64 image for Push Event
|
||||||
|
uses: docker/build-push-action@v5
|
||||||
|
with:
|
||||||
|
context: .
|
||||||
|
file: docker/Dockerfile.arm64
|
||||||
|
platforms: linux/arm64/v8
|
||||||
|
labels: ${{ steps.meta-arm64.outputs.labels }}
|
||||||
|
push: true
|
||||||
|
tags: ${{ steps.meta-arm64.outputs.tags }}
|
||||||
|
build-args: |
|
||||||
|
"release=1"
|
32
.github/workflows/ci.yml
vendored
32
.github/workflows/ci.yml
vendored
|
@ -38,10 +38,10 @@ jobs:
|
||||||
matrix:
|
matrix:
|
||||||
stable: [true]
|
stable: [true]
|
||||||
crystal:
|
crystal:
|
||||||
- 1.7.3
|
|
||||||
- 1.8.2
|
|
||||||
- 1.9.2
|
- 1.9.2
|
||||||
- 1.10.1
|
- 1.10.1
|
||||||
|
- 1.11.2
|
||||||
|
- 1.12.1
|
||||||
include:
|
include:
|
||||||
- crystal: nightly
|
- crystal: nightly
|
||||||
stable: false
|
stable: false
|
||||||
|
@ -90,10 +90,10 @@ jobs:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Build Docker
|
- name: Build Docker
|
||||||
run: docker-compose build --build-arg release=0
|
run: docker compose build --build-arg release=0
|
||||||
|
|
||||||
- name: Run Docker
|
- name: Run Docker
|
||||||
run: docker-compose up -d
|
run: docker compose up -d
|
||||||
|
|
||||||
- name: Test Docker
|
- name: Test Docker
|
||||||
run: while curl -Isf http://localhost:3000; do sleep 1; done
|
run: while curl -Isf http://localhost:3000; do sleep 1; done
|
||||||
|
@ -124,4 +124,28 @@ jobs:
|
||||||
- name: Test Docker
|
- name: Test Docker
|
||||||
run: while curl -Isf http://localhost:3000; do sleep 1; done
|
run: while curl -Isf http://localhost:3000; do sleep 1; done
|
||||||
|
|
||||||
|
ameba_lint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
submodules: true
|
||||||
|
|
||||||
|
- name: Install Crystal
|
||||||
|
uses: crystal-lang/install-crystal@v1.8.0
|
||||||
|
with:
|
||||||
|
crystal: latest
|
||||||
|
|
||||||
|
- name: Cache Shards
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
./lib
|
||||||
|
./bin
|
||||||
|
key: shards-${{ hashFiles('shard.lock') }}
|
||||||
|
|
||||||
|
- name: Install Shards
|
||||||
|
run: shards install
|
||||||
|
|
||||||
|
- name: Run Ameba linter
|
||||||
|
run: bin/ameba
|
||||||
|
|
1081
CHANGELOG.md
1081
CHANGELOG.md
File diff suppressed because it is too large
Load diff
844
CHANGELOG_legacy.md
Normal file
844
CHANGELOG_legacy.md
Normal file
|
@ -0,0 +1,844 @@
|
||||||
|
# Note: This is no longer updated and links to omarroths repo, which doesn't exist anymore.
|
||||||
|
|
||||||
|
# 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)
|
||||||
|
|
||||||
|
# Version 0.16.0: API Improvements and Annotations
|
||||||
|
|
||||||
|
Hello again! This past month has seen [116 commits](https://github.com/omarroth/invidious/compare/0.15.0..0.16.0) from 13 contributors and a couple important changes I'd like to announce.
|
||||||
|
|
||||||
|
A privacy policy is now available [here](https://invidio.us/privacy). I've done my best to explain things as clearly as possible without oversimplifying, and would very much recommend reading it if you're concerned about your privacy and want to learn more about how Invidious uses your data. Please let me know if there is anything that needs clarification.
|
||||||
|
|
||||||
|
I'm also very happy to announce that a Spanish translation has been added to the site. You can use it with `?hl=es` or by setting `es` as your default locale. As always I'm extremely grateful to translators for making the site accessible to more people.
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
Invidious now supports server-to-server [push notifications](https://developers.google.com/youtube/v3/guides/push_notifications). This uses [PubSubHubbub](https://pubsubhubbub.github.io/PubSubHubbub/pubsubhubbub-core-0.4.html) to automatically handle new videos sent to an instance, which is less resource intensive and generally faster. Note that it will not pull all videos from a subscribed channel, so recommended usage is in addition to `channel_threads`. Using PubSub requires a valid `domain` that updates can be sent to, and a random string that can be used to sign updates sent to the instance. You can enable it by adding `use_pubsub_feeds: true` to your `config.yml`. See [Configuration](https://github.com/omarroth/invidious/wiki/Configuration) for more info.
|
||||||
|
|
||||||
|
Unfortunately there are a couple necessary changes to the DB to support `liveNow` and `premiereTimestamp` in subscription feeds. Migration scripts have been provided that should be used automatically if following the instructions [here](https://github.com/omarroth/invidious/wiki/Updating).
|
||||||
|
|
||||||
|
You can now configure default user preferences for your instance. This allows you to set default locale, player preferences, and more. See [#415](https://github.com/omarroth/invidious/issues/415) for more details and example usage.
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
The [fields](https://developers.google.com/youtube/v3/getting-started#fields) API has been added with [#429](https://github.com/omarroth/invidious/pull/429) and is now supported on all JSON endpoints, thanks [**@afrmtbl**](https://github.com/afrmtbl)! Synax is straight-forward and can be used to reduce data transfer and create a simpler response for debugging. You can see an example [here](https://invidio.us/api/v1/videos/CvFH_6DNRCY?pretty=1&fields=title,recommendedVideos/title). I've been quite happy using it and hope it is similarly useful for others.
|
||||||
|
|
||||||
|
An `/api/v1/annotations/:id` endpoint has been added for pulling legacy annotation data from [this](https://archive.org/details/youtubeannotations) archive, see below for more details. You can also access annotation data available on YouTube using `?source=youtube`, although this will only return card data as legacy annotations were deleted on January 15th.
|
||||||
|
|
||||||
|
A couple minor changes to existing endpoints:
|
||||||
|
|
||||||
|
- A `premiereTimestamp` field has been added to `/api/v1/videos/:id`
|
||||||
|
- A `sort_by` param has been added to `/api/v1/comments/:id`, supports `new`, `top`.
|
||||||
|
|
||||||
|
More info is available in the [documentation](https://github.com/omarroth/invidious/wiki/API).
|
||||||
|
|
||||||
|
## Annotations
|
||||||
|
|
||||||
|
I'm pleased to announce that annotation data is finally available from the roughly 1.4 billion videos archived as part of [this](https://www.reddit.com/r/DataHoarder/comments/aa6czg/youtube_annotation_archive/) project. They are accessible from the Internet Archive [here](https://archive.org/details/youtubeannotations) or as a 355GB torrent, see [here](https://www.reddit.com/r/DataHoarder/comments/b7imx9/youtube_annotation_archive_annotation_data_from/) for more details. A corresponding `/api/v1/annotations/:id` endpoint has been added to Invidious which uses the collection from IA to provide legacy annotations.
|
||||||
|
|
||||||
|
Support for them in the player is possible thanks to [this](https://github.com/afrmtbl/videojs-youtube-annotations) plugin developed by [**@afrmtbl**](https://github.com/afrmtbl). A PR for adding support to the site is available as [#303](https://github.com/omarroth/invidious/pull/303). There's also an [extension](https://github.com/afrmtbl/AnnotationsRestored) for overlaying them on top of the YouTube player (again thanks to [**@afrmtbl**](https://github.com/afrmtbl)), and an [extension](https://tech234a.bitbucket.io/AnnotationsReloaded?src=invidious) for hooking into code still present in the YouTube player itself, developed by [**@tech234a**](https://github.com/tech234a).
|
||||||
|
|
||||||
|
I would recommend reading the [official announcement](https://www.reddit.com/r/DataHoarder/comments/b7imx9/youtube_annotation_archive_annotation_data_from/) for more details. I would like to again thank everyone that helped contribute to this project.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$42.42
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$70.11
|
||||||
|
- Crypto : ~\$1.76 (converted from BCH, BTC, BSV)
|
||||||
|
- Total : \$114.29
|
||||||
|
|
||||||
|
### 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
|
||||||
|
|
||||||
|
This past month the site saw a couple abnormal peaks in traffic, so an additional webserver has been added to match the increased load. The goal on Patreon has been updated to match the above expenses.
|
||||||
|
|
||||||
|
Thanks everyone!
|
||||||
|
|
||||||
|
# 0.15.0 (2019-03-06)
|
||||||
|
|
||||||
|
## Version 0.15.0: Preferences and Channel Playlists
|
||||||
|
|
||||||
|
The project has seen quite a bit of activity this past month. Large focus has been on fixing bugs, but there's still quite a few new features I'm happy to announce. There have been [133 commits](https://github.com/omarroth/invidious/compare/0.14.0...0.15.0) from 15 contributors this past month.
|
||||||
|
|
||||||
|
As a couple miscellaneous changes, a couple [nice screenshots](https://github.com/omarroth/invidious#screenshots) have been added to the README, so folks can see more of what the site has to offer without creating an account.
|
||||||
|
|
||||||
|
The footer has also been cleaned up quite a bit, and now displays the current version, so it's easier to know what features are available from the current instance.
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
This past month there has been a minor release - `0.14.1` - which fixes a breaking change made by YouTube for their polymer redesign.
|
||||||
|
|
||||||
|
There have been several new features that unfortunately require a database migration. There are migration scripts provided in `config/migrate-scripts`, and the [wiki](https://github.com/omarroth/invidious/wiki/Updating) has instructions for automatically applying them. I'll do my best to keep those changes to a minimum, and expect to see a corresponding script to automatically apply any new changes.
|
||||||
|
|
||||||
|
Administrator preferences have been added with [#312](https://github.com/omarroth/invidious/issues/312), which allows administrators to customize their instance. Administrators can change the order of feed menus, change the default homepage, disable open registration, and several other options. There's a short 'how-to' [here](https://github.com/omarroth/invidious/issues/312#issuecomment-468831842), and the new options are documented [here](https://github.com/omarroth/invidious/wiki/Configuration).
|
||||||
|
|
||||||
|
An `/api/v1/stats` endpoint has been added with [#356](https://github.com/omarroth/invidious/issues/356), which reports the instance version and number of active users. Statistics are disabled by default, and can be enabled in administator preferences. Statistics for the official instance are available [here](https://invidio.us/api/v1/stats?pretty=1).
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
`/api/v1/channels/:ucid` now provides an `autoGenerated` tag, which returns true for topic channels, and larger genre channels generated by YouTube. These channels don't have any videos of their own, so `latestVideos` will be empty. It is recommended instead to display a list of playlists generated by YouTube.
|
||||||
|
|
||||||
|
You can now pull a list of playlists from a channel with `/api/v1/channels/playlists/:ucid`. Supported options are documented in the [wiki](https://github.com/omarroth/invidious/wiki/API#get-apiv1channelsplaylistsucid-apiv1channelsucidplaylists). Pagination is handled with a `continuation` token, which is generated on each call. Of note is that auto-generated channels currently have one page of results, and subsequent calls will be empty.
|
||||||
|
|
||||||
|
For quickly pulling the latest 30 videos from a channel, there is now `/api/v1/channels/latest/:ucid`. It is much faster than a call to `/api/v1/channels/:ucid`. It will not convert an author name to a valid ucid automatically, and will not return any extra data about a channel.
|
||||||
|
|
||||||
|
## Preferences
|
||||||
|
|
||||||
|
In addition to administrator preferences mentioned above, you can now change your preferences without an account (see [#42](https://github.com/omarroth/invidious/pull/42)). I think this is quite an improvement to the usability of the site, and is much friendlier to privacy-conscious folks that don't want to make an account. Preferences will be automatically imported to a newly created account.
|
||||||
|
|
||||||
|
Several issues with sorting subscriptions have been fixed, and `/manage_subscriptions` has been sped up significantly. The subscription feed has also seen a bump in performance. Delayed notifications have unfortunately started becoming a problem now that there are more users on the site. Some new changes are currently being tested which should mostly resolve the issue, so expect to see more in the next release.
|
||||||
|
|
||||||
|
## Channel Playlists
|
||||||
|
|
||||||
|
You can now view available playlists from a channel, and [auto-generated channels](https://invidio.us/channel/UC-9-kyTW8ZkZNDHQJ6FgpwQ) are no longer empty. You can sort as you would on YouTube, and all the same functionality should be available. I'm quite pleased to finally have it implemented, since it's currently the only data available from the above mentioned auto-generated channels, and makes it much easier to consume music on the site.
|
||||||
|
|
||||||
|
There's also more discussion on improving Invidious for streaming music in [#304](https://github.com/omarroth/invidious/issues/304), and adding support for music.youtube.com. I would appreciate any thoughts on how to improve that experience, since it's a very large and useful part of YouTube.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$42.42
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$30.97
|
||||||
|
- Crypto : ~\$0.00 (converted from BCH, BTC)
|
||||||
|
- Total : \$73.39
|
||||||
|
|
||||||
|
### 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-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$75.00
|
||||||
|
|
||||||
|
It's been very humbling to see how fast the project has grown, and I look forward to making the site even better. Thank you everyone.
|
||||||
|
|
||||||
|
# 0.14.0 (2019-02-06)
|
||||||
|
|
||||||
|
## Version 0.14.0: Community
|
||||||
|
|
||||||
|
This last month several contributors have made improvements specifically for the people using this project. New pages have been added to the wiki, and there is now a [Matrix Server](https://riot.im/app/#/room/#invidious:matrix.org) and IRC channel so it's easier and faster for people to ask questions or chat. There have been [101 commits](https://github.com/omarroth/invidious/compare/0.13.0...0.14.0) since the last major release from 8 contributors.
|
||||||
|
|
||||||
|
It has come to my attention in the past month how many people are self-hosting, and I would like to make it easier for them to do so.
|
||||||
|
|
||||||
|
With that in mind, expect future releases to have a section for For Administrators (if any relevant changes) and For Developers (if any relevant changes).
|
||||||
|
|
||||||
|
## For Administrators
|
||||||
|
|
||||||
|
This month the most notable change for administrators is releases. As always, there will be a major release each month. However, a new minor release will be made whenever there are any critical bugs that need to be fixed.
|
||||||
|
|
||||||
|
This past month is the first time there has been a minor release - `0.13.1` - which fixes a breaking change made by YouTube. Administrators using versioning for their instances will be able to rely on the latest version, and should have a system in place to upgrade their instance as soon as a new release is available.
|
||||||
|
|
||||||
|
Several new pages have been added to the [wiki](https://github.com/omarroth/invidious/wiki#for-administrators) (as mentioned below) that will help administrators better setup their own instances. Configuration, maintenance, and instructions for updating are of note, as well as several common issues that are encountered when first setting up.
|
||||||
|
|
||||||
|
## For Developers
|
||||||
|
|
||||||
|
There's now a `pretty=1` parameter for most endpoints so you can view data easily from the browser, which is convenient for debugging and casual use. You can see an example [here](https://invidio.us/api/v1/videos/CvFH_6DNRCY?pretty=1).
|
||||||
|
|
||||||
|
Unfortunately the `/api/v1/insights/:id` endpoint is no longer functional, as YouTube removed all publicly available analytics around a month ago. The YouTube endpoint now returns a 404, so it's unlikely it will be functional again.
|
||||||
|
|
||||||
|
## Wiki
|
||||||
|
|
||||||
|
There have been a sizable number of changes to the Wiki, including a [list of public Invidious instances](https://github.com/omarroth/invidious/wiki/Invidious-Instances), the [list of extensions](https://github.com/omarroth/invidious/wiki/Extensions), and documentation for administrators (as mentioned above) and developers.
|
||||||
|
|
||||||
|
The wiki is editable by anyone so feel free to add anything you think is useful.
|
||||||
|
|
||||||
|
## Matrix & IRC
|
||||||
|
|
||||||
|
Thee is now a [Matrix Server](https://riot.im/app/#/room/#invidious:matrix.org) for Invidious, so please feel free to hop on if you have any questions or want to chat. There is also a registered IRC channel: #invidious on Freenode which is bridged to Matrix.
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
Several new features have been added, including a download button, creator hearts and comment colors, and a French translation.
|
||||||
|
|
||||||
|
There have been fixes for Google logins, missing text in locales, invalid links to genre channels, and better error handling in the player, among others.
|
||||||
|
|
||||||
|
Several fixes and features are omitted for space, so I'd recommend taking a look at the [compare tab](https://github.com/omarroth/invidious/compare/0.13.0...0.14.0) for more information.
|
||||||
|
|
||||||
|
## Annotations Update
|
||||||
|
|
||||||
|
Annotations were removed January 15th, 2019 around15:00 UTC. Before they were deleted we were able to archive annotations from around 1.4 billion videos. I'd very much recommend taking a look [here](https://www.reddit.com/r/DataHoarder/comments/al7exa/youtube_annotation_archive_update_and_preview/) for more information and a list of acknowledgements. I'm extremely thankful to everyone who was able to contribute and I'm glad we were able to save such a large part of internet history.
|
||||||
|
|
||||||
|
There's been large strides in supporting them in the player as well, which you can follow in [#303](https://github.com/omarroth/invidious/pull/303). You can preview the functionality at https://dev.invidio.us . Before they are added to the main site expect to see an option to disable them, both site-wide and per video.
|
||||||
|
|
||||||
|
Organizing this project has unfortunately taken up quite a bit of my time, and I've been very grateful for everyone's patience.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth) : \$49.42
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$27.89
|
||||||
|
- Crypto : ~\$0.00 (converted from BCH, BTC)
|
||||||
|
- Total : \$77.31
|
||||||
|
|
||||||
|
### 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-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$75.00
|
||||||
|
|
||||||
|
As always I'm grateful for everyone's contributions and support. I'll see you all in March.
|
||||||
|
|
||||||
|
# 0.13.1 (2019-01-19)
|
||||||
|
|
||||||
|
##
|
||||||
|
|
||||||
|
# 0.13.0 (2019-01-06)
|
||||||
|
|
||||||
|
## Version 0.13.0: Translations, Annotations, and Tor
|
||||||
|
|
||||||
|
I hope everyone had a happy New Year! There's been a couple new additions since last release, with [44 commits](https://github.com/omarroth/invidious/compare/0.12.0...0.13.0) from 9 contributors. It's been quite a year for the project, and I hope to continue improving the project into 2019! Starting off the new year:
|
||||||
|
|
||||||
|
## Translations
|
||||||
|
|
||||||
|
I'm happy to announce support for translations has been added with [`a160c64`](https://github.com/omarroth/invidious/a160c64). Currently, there is support for:
|
||||||
|
|
||||||
|
- Arabic (`ar`)
|
||||||
|
- Dutch (`nl`)
|
||||||
|
- English (`en-US`)
|
||||||
|
- German (`de`)
|
||||||
|
- Norwegian Bokmål (`nb_NO`)
|
||||||
|
- Polish (`pl`)
|
||||||
|
- Russian (`ru`)
|
||||||
|
|
||||||
|
Which you can change in your preferences under `Language`. You can also add `&hl=LANGUAGE` to the end of any request to translate it to your preferred language, for example https://invidio.us/?hl=ru. I'd like to say thank you again to everyone who has helped translate the site! I've mentioned this before, but I'm delighted that so many people find the project useful.
|
||||||
|
|
||||||
|
## Annotations
|
||||||
|
|
||||||
|
Recently, [YouTube announced that all annotations will be deleted on January 15th, 2019](https://support.google.com/youtube/answer/7342737). I believe that annotations have a very important place in YouTube's history, and [announced a project to archive them](https://www.reddit.com/r/DataHoarder/comments/aa6czg/youtube_annotation_archive/).
|
||||||
|
|
||||||
|
I expect annotations to be supported in the Invidious player once archiving is complete (see [#110](https://github.com/omarroth/invidious/issues/110) for details), and would also like to host them for other developers to use in their projects.
|
||||||
|
|
||||||
|
The code is available [here](https://github.com/omarroth/archive), and contains instructions for running a worker if you would like to contribute. There's much more information available in the announcement as well for anyone who is interested.
|
||||||
|
|
||||||
|
## Tor
|
||||||
|
|
||||||
|
I unfortunately missed the chance to mention this in the previous release, but I'm now happy to announce that you can now view Invidious through Tor at the following links:
|
||||||
|
|
||||||
|
kgg2m7yk5aybusll.onion
|
||||||
|
axqzx4s6s54s32yentfqojs3x5i7faxza6xo3ehd4bzzsg2ii4fv2iid.onion
|
||||||
|
|
||||||
|
Invidious is well suited to use through Tor, as it does not require any JS and is fairly lightweight. I'd recommend looking [here](https://diasp.org/posts/10965196) and [here](https://www.reddit.com/r/TOR/comments/a3c1ak/you_can_now_watch_youtube_videos_anonymously_with/) for more details on how to use the onion links, and would like to say thank you to [/u/whonix-os](https://www.reddit.com/user/whonix-os) for suggesting it and providing support setting setting them up.
|
||||||
|
|
||||||
|
## Popular and Trending
|
||||||
|
|
||||||
|
You can now easily view videos trending on YouTube with [`a16f967`](https://github.com/omarroth/invidious/a16f967). It also provides support for viewing YouTube's various categories categories, such as `News`, `Gaming`, and `Music`. You can also change the `region` parameter to view trending in different countries, which should be made easier to use in the coming weeks.
|
||||||
|
|
||||||
|
A link to `/feed/popular` has also been added, which provides a list of videos sorted using the algorithm described [here](https://github.com/omarroth/invidious/issues/217#issuecomment-436503761). I think it better reflects what users watch on the site, but I'd like to hear peoples' thoughts on this and on how it could be improved.
|
||||||
|
|
||||||
|
## Finances
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth): \$64.63
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$30.05
|
||||||
|
- Crypto : ~\$28.74 (converted from BCH, BTC)
|
||||||
|
- Total : \$123.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-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$75.00
|
||||||
|
|
||||||
|
### What will happen with what's left over?
|
||||||
|
|
||||||
|
I believe this is the first month that all expenses have been fully paid for by donations. Thank you! I expect to allocate the current amount for hardware to improve performance and for hosting annotation data, as mentioned above.
|
||||||
|
|
||||||
|
Anything that is left over is kept to continue hosting the project for as long as possible. Thank you again everyone!
|
||||||
|
|
||||||
|
I think that's everything for 2018. There's lots still planned, and I'm very excited for the future of this project!
|
||||||
|
|
||||||
|
# 0.12.0 (2018-12-06)
|
||||||
|
|
||||||
|
## Version 0.12.0: Accessibility, Privacy, Transparency
|
||||||
|
|
||||||
|
Hello again, it's been a while! A lot has happened since the last release. Invidious has seen [134 commits](https://github.com/omarroth/invidious/compare/0.11.0...0.12.0) from 3 contributors, and I'm quite happy with the progress that has been made. I enjoyed this past month, and I believe having a monthly release schedule allows me to focus on more long-term improvements, and I hope people enjoy these more substantial updates as well.
|
||||||
|
|
||||||
|
## Accessability and Privacy
|
||||||
|
|
||||||
|
There have been quite a few improvements for user privacy, and improvements that improve accessibility for both people and software.
|
||||||
|
|
||||||
|
You can now view comments without JS with [`19516ea`](https://github.com/omarroth/invidious/19516ea). Currently, this functionality is limited to the first 20 comments, but expect this functionality to be improved to come as close to the JS version as possible. Folks can track progress in [#204](https://github.com/omarroth/invidious/issues/204).
|
||||||
|
|
||||||
|
Invidious is now compatible with [LibreJS](https://www.gnu.org/software/librejs/), and provides license information [here](https://invidio.us/licenses) with [`7f868ec`](https://github.com/omarroth/invidious/7f868ec). As expected, all libraries are compatible under the AGPLv3, and I'm happy to mention that no other changes were required to make Invidious compatible with LibreJS.
|
||||||
|
|
||||||
|
A DNT policy has also been added with [`9194f47`](https://github.com/omarroth/invidious/9194f47) for compatibility with [Privacy Badger](https://www.eff.org/privacybadger). I'm pleased to mention that here too no other changes had to be made in order for Invidious to be compatible with this extension. I expect a privacy policy to be added soon as well, so users can better understand how Invidious uses their data.
|
||||||
|
|
||||||
|
For users that are visually impaired, there is now a text CAPTCHA available so it's easier to register and login. Because of the simple front-end of the project, I expect screen readers and other software to be able to easily understand the site's interface. In combination with the ability to listen-only, I believe Invidious is much more accessible than YouTube. Folks can read [#244](https://github.com/omarroth/invidious/issues/244) for more details, and I would very much appreciate any feedback on how this can be improved.
|
||||||
|
|
||||||
|
## User Preferences
|
||||||
|
|
||||||
|
There have been a lot of improvements to preferences. Options for enabling audio-only by default and continuous playback (autoplay) have been added with [`e39dec9`](https://github.com/omarroth/invidious/e39dec9), with [`4b76b93`](https://github.com/omarroth/invidious/4b76b93), respectively. Users can also now mark videos as watched from their subscription feed and view watch history by going to https://invidio.us/feed/history. I expect to add more information to history so that it's easier to use. Folks can track progress with [#182](https://github.com/omarroth/invidious/issues/182). As with all data Invidious keeps, watch history can be exported [here](https://invidio.us/data_control).
|
||||||
|
|
||||||
|
Users can now delete their account with [`b9c29bf`](https://github.com/omarroth/invidious/b9c29bf). This will remove _all_ user data from Invidious, including session IDs, watch history, and subscriptions. As mentioned above, it's easy to export that data and import it to a local instance, or export subscriptions for use with other applications such as [FreeTube](https://github.com/FreeTubeApp/FreeTube) or [NewPipe](https://github.com/TeamNewPipe/NewPipe).
|
||||||
|
|
||||||
|
## Translation and Internationalis(z)ation
|
||||||
|
|
||||||
|
Invidious has been approved for hosting by Weblate, available [here](https://hosted.weblate.org/projects/invidious/translations/). At the time of writing, translations for Arabic, Dutch, German, Polish, and Russian are currently underway. I would like to say a very big thank you to everyone working on them, and I hope to fully support them within around 2 weeks. Folks can track progress with [#251](https://github.com/omarroth/invidious/issues/251).
|
||||||
|
|
||||||
|
## Transperency and Finances
|
||||||
|
|
||||||
|
For the sake of transparency, I plan on publishing each month's finances. This is currently already done on Liberapay and Patreon, but there is not a total amount currently provided anywhere, and I would also like to include expenses to provide a better explanation of how patrons' money is being spent.
|
||||||
|
|
||||||
|
### Donations
|
||||||
|
|
||||||
|
- [Patreon](https://www.patreon.com/omarroth): \$43.60 (Patreon takes roughly 9%)
|
||||||
|
- [Liberapay](https://liberapay.com/omarroth) : \$22.10
|
||||||
|
- Crypto : ~\$1.25 (converted from BCH, BTC)
|
||||||
|
- Total : \$66.95
|
||||||
|
|
||||||
|
### 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-db1 (s-4vcpu-8gb) : \$40.00 (database)
|
||||||
|
- Total : \$75.00
|
||||||
|
|
||||||
|
I'd be happy to provide any explanation where needed. I would also like to thank everyone who donates, it really helps and I can't say how happy I am to see that so many people find it valuable.
|
||||||
|
|
||||||
|
That's all for this month. I wish everyone the best for the holidays, and I'll see you all again in January!
|
||||||
|
|
||||||
|
# 0.11.0 (2018-10-23)
|
||||||
|
|
||||||
|
## Week 11: FreeTube and Styling
|
||||||
|
|
||||||
|
This past Friday I'm been very excited to see that FreeTube version [0.4.0](https://github.com/FreeTubeApp/FreeTube/tree/0.4.0) has been released! I'd recommend taking a look at the official patch notes, but to spoil a little bit here: FreeTube now uses the Invidious API for _all_ requests previously sent to YouTube, and has also seen support for playlists, keyboard shortcuts, and more default settings (speed, autoplay, and subtitles). I'm happy to see that FreeTube has reached 500 stars on Github, and I think it's very much deserved. I'd recommend keeping an eye on the newly-launched [FreeTube blog](https://freetube.writeas.com/) for updates on the project.
|
||||||
|
|
||||||
|
Quite a few styling changes have been added this past week, including channel subscriber count to the subscribe and unsubscribe buttons. The changes sound small, but they've been a very big improvement and I'm quite satisfied with how they look. Also to note is that partial support for duration in thumbnails have been added with [#202](https://github.com/omarroth/invidious/issues/202). Overall, I think the site is becoming much more pleasing visually, and I hope to continue to improve it.
|
||||||
|
|
||||||
|
I've been very pleased to see Invidious in its current state, and I believe it's many times more mature compared to even a month ago. Changes have also started slowing down a bit as it's become more mature, and therefore I'd like to transition to a monthly update schedule in order to provide more comprehensive updates for everyone. I want to thank you all for helping me reach this point. I can't say how happy I am for Invidious to be where it is now.
|
||||||
|
|
||||||
|
Enjoy the rest of your week everyone, I'll see you in November!
|
||||||
|
|
||||||
|
# 0.10.0 (2018-10-16)
|
||||||
|
|
||||||
|
## Week 10: Subscriptions
|
||||||
|
|
||||||
|
This week I'm happy to announce that subscriptions have been drastically sped up with
|
||||||
|
35e63fa. As I mentioned last week, this essentially "caches" a user's feed, meaning that operations that previously took 20 seconds or timed out, now can load in under a second. I'd take a look at [#173](https://github.com/omarroth/invidious/issues/173) for a sample benchmark. Previously features that made Invidious's feed so useful, such as filtering by unseen and by author would take too long to load, and so instead would timeout. I'm very happy that this has been fixed, and folks can get back to using these features.
|
||||||
|
|
||||||
|
Among some smaller features that have been added this week include [#118](https://github.com/omarroth/invidious/issues/118), which adds, in my opinion, some very attractive subscribe and unsubscribe buttons. I think it's also a bit of a functional improvement as well, since it doesn't require a user to reload the page in order to subscribe or unsubscribe to a channel, and also gives the opportunity to put the channel's sub count on display.
|
||||||
|
|
||||||
|
An option to swap between Reddit and YouTube comments without a page reload has been added with
|
||||||
|
5eefab6, bringing it somewhat closer in functionality to the popular [AlienTube](https://github.com/xlexi/alientube) extension, on which it is based (although the extension unfortunately appears now to be fragmented).
|
||||||
|
|
||||||
|
As always, there are a couple smaller improvements this week, including some minor fixes for geo-bypass with
|
||||||
|
e46e618 and [`245d0b5`](https://github.com/omarroth/invidious/245d0b5), playlist preferences with [`81b4477`](https://github.com/omarroth/invidious/81b4477), and YouTube comments with [`02335f3`](https://github.com/omarroth/invidious/02335f3).
|
||||||
|
|
||||||
|
This coming week I'd also recommend keeping an eye on the excellent [FreeTube](https://github.com/FreeTubeApp/FreeTube), which is looking forward to a new release. I've been very lucky to work with [**@PrestonN**](https://github.com/PrestonN) for the past few weeks to improve the Invidious API, and I'm quite looking forward to the new release.
|
||||||
|
|
||||||
|
That's all for this week folks, thank you all again for your continued interest and support.
|
||||||
|
|
||||||
|
# 0.9.0 (2018-10-08)
|
||||||
|
|
||||||
|
## Week 9: Playlists
|
||||||
|
|
||||||
|
Not as much to announce this week, but I'm still quite happy to announce a couple things, namely:
|
||||||
|
|
||||||
|
Playback support for playlists has finally been added with [`88430a6`](https://github.com/omarroth/invidious/88430a6). You can now view playlists with the `&list=` query param, as you would on YouTube. You can also view mixes with the mentioned `&list=`, although they require some extra handling that I would like to add in the coming week, as well as adding playlist looping and shuffle. I think playback support has been a roadblock for more exciting features such as [#114](https://github.com/omarroth/invidious/issues/114), and I look forward to improving the experience.
|
||||||
|
|
||||||
|
Comments have had a bit of a cosmetic upgrade with [#132](https://github.com/omarroth/invidious/issues/132), which I think helps better distinguish between Reddit and YouTube comments, as it makes them appear similarly to their respective sites. You can also now switch between YouTube and Reddit comments with a push of a button, which I think is quite an improvement, especially for newer or less popular videos with fewer comments.
|
||||||
|
|
||||||
|
I've had a small breakthrough in speeding up users' subscription feeds with PostgreSQL's [materialized views](https://www.postgresql.org/docs/current/static/rules-materializedviews.html). Without going into too much detail, materialized views essentially cache the result of a query, making it possible to run resource-intensive queries once, rather than every time a user visits their feed. In the coming week I hope to push this out to users, and hopefully close [#173](https://github.com/omarroth/invidious/issues/173).
|
||||||
|
|
||||||
|
I haven't had as much time to work on the project this week, but I'm quite happy to have added some new features. Have a great week everyone.
|
||||||
|
|
||||||
|
# 0.8.0 (2018-10-02)
|
||||||
|
|
||||||
|
## Week 8: Mixes
|
||||||
|
|
||||||
|
Hello again!
|
||||||
|
|
||||||
|
Mixes have been added with [`20130db`](https://github.com/omarroth/invidious/20130db), which makes it easy to create a playlist of related content. See [#188](https://github.com/omarroth/invidious/issues/188) for more info on how they work. Currently, they return the first 50 videos rather than a continuous feed to avoid tracking by Google/YouTube, which I think is a good trade-off between usability and privacy, and I hope other folks agree. You can create mixes by adding `RD` to the beginning of a video ID, an example is provided [here](https://www.invidio.us/mix?list=RDYE7VzlLtp-4) based on Big Buck Bunny. I've been quite happy with the results returned for the mixes I've tried, and it is not limited to music, which I think is a big plus. To emulate a continuous feed provided many are used to, using the last video of each mix as a new 'seed' has worked well for me. In the coming week I'd like to to add playback support in the player to listen to these easily.
|
||||||
|
|
||||||
|
A very big thanks to [**@flourgaz**](https://github.com/flourgaz) for Docker support with [#186](https://github.com/omarroth/invidious/pull/186). This is an enormous improvement in portability for the project, and opens the door for Heroku support (see [#162](https://github.com/omarroth/invidious/issues/162)), and seamless support on Windows. For most users, it should be as easy as running `docker-compose up`.
|
||||||
|
|
||||||
|
I've spent quite a bit of time this past week improving support for geo-bypass (see [#92](https://github.com/omarroth/invidious/issues/92)), and am happy to note that Invidious has been able to proxy ~50% of the geo-restricted videos I've tried. In addition, you can now watch geo-restricted videos if you have `dash` enabled as your `preferred quality`, for more details see [#34](https://github.com/omarroth/invidious/issues/34) and [#185](https://github.com/omarroth/invidious/issues/185), or last week's update. For folks interested in replicating these results for themselves, I'd take a look [here](https://gist.github.com/omarroth/3ce0f276c43e0c4b13e7d9cd35524688) for the script used, and [here](https://gist.github.com/omarroth/beffc4a76a7b82a422e1b36a571878ef) for a list of videos restricted in the US.
|
||||||
|
|
||||||
|
1080p has seen a fairly smooth roll-out, although there have been a couple issues reported, mainly [#193](https://github.com/omarroth/invidious/issues/193), which is likely an issue in the player. I've also encountered a couple other issues myself that I would like to investigate. Although none are major, I'd like to keep 1080p opt-in for registered users another week to better address these issues.
|
||||||
|
|
||||||
|
Have an excellent week everyone.
|
||||||
|
|
||||||
|
# 0.7.0 (2018-09-25)
|
||||||
|
|
||||||
|
## Week 7: 1080p and Search Types
|
||||||
|
|
||||||
|
Hello again everyone! I've got quite a couple announcements this week:
|
||||||
|
|
||||||
|
Experimental 1080p support has been added with [`b3ca392`](https://github.com/omarroth/invidious/b3ca392), and can be enabled by going to preferences and changing `preferred video quality` to `dash`. You can find more details [here](https://github.com/omarroth/invidious/issues/34#issuecomment-424171888). Currently quality and speed controls have not yet been integrated into the player, but I'd still appreciate feedback, mainly on any issues with buffering or DASH playback. I hope to integrate 1080p support into the player and push support site-wide in the coming weeks.
|
||||||
|
|
||||||
|
You can now filter content types in search with the `type:TYPE` filter. Supported content types are `playlist`, `channel`, and `video`. More info is available [here](https://github.com/omarroth/invidious/issues/126#issuecomment-423823148). I think this is quite an improvement in usability and I hope others find the same.
|
||||||
|
|
||||||
|
A [CHANGELOG](https://github.com/omarroth/invidious/blob/master/CHANGELOG.md) has been added to the repository, so folks will now receive a copy of all these updates when cloning. I think this is an improvement in hosting the project, as it is no longer tied to the `/releases` tab on Github or the posts on Patreon.
|
||||||
|
|
||||||
|
Recently, users have been reporting 504s when attempting to access their subscriptions, which is tracked in [#173](https://github.com/omarroth/invidious/issues/173). This is most likely caused by an uptick in usage, which I am absolutely grateful for, but unfortunately has resulted in an increase in costs for hosting the site, which is why I will be bumping my goal on Patreon from $60 to $80. I would appreciate any feedback on how subscriptions could be improved.
|
||||||
|
|
||||||
|
Other minor improvements include:
|
||||||
|
|
||||||
|
- Additional regions added to bypass geo-block with [`9a78523`](https://github.com/omarroth/invidious/9a78523)
|
||||||
|
- Fix for playlists containing less than 100 videos (previously shown as empty) with [`35ac887`](https://github.com/omarroth/invidious/35ac887)
|
||||||
|
- Fix for `published` date for Reddit comments (previously showing negative seconds) with [`6e09202`](https://github.com/omarroth/invidious/6e09202)
|
||||||
|
|
||||||
|
Thank you everyone for your support!
|
||||||
|
|
||||||
|
# 0.6.0 (2018-09-18)
|
||||||
|
|
||||||
|
## Week 6: Filters and Thumbnails
|
||||||
|
|
||||||
|
Hello again! This week I'm happy to mention a couple new features to search as well as some miscellaneous usability improvements.
|
||||||
|
|
||||||
|
You can now constrain your search query to a specific channel with the `channel:CHANNEL` filter (see [#165](https://github.com/omarroth/invidious/issues/165) for more details). Unfortunately, other search filters combined with channel search are not yet supported. I hope to add support for them in the coming weeks.
|
||||||
|
|
||||||
|
You can also now search only your subscriptions by adding `subscriptions:true` to your query (see [#30](https://github.com/omarroth/invidious/issues/30) for more details). It's not quite ready for widespread use but I would appreciate feedback as the site updates to fully support it. Other search filters are not yet supported with `subscriptions:true`, but I hope to add more functionality to this as well.
|
||||||
|
|
||||||
|
With [#153](https://github.com/omarroth/invidious/issues/153) and [#168](https://github.com/omarroth/invidious/issues/168) all images on the site are now proxied through Invidious. In addition to offering the user more protection from Google's eyes, it also allows the site to automatically pick out the highest resolution thumbnail for videos. I think this is quite a large aesthetic improvement and I hope others will find the same.
|
||||||
|
|
||||||
|
As a smaller improvement to the site, you can also now view RSS feeds for playlists with [#113](https://github.com/omarroth/invidious/issues/113).
|
||||||
|
|
||||||
|
These updates are also now listed under Github's [releases](https://github.com/omarroth/invidious/releases). I'm also planning on adding them as a `CHANGELOG.md` in the repository itself so people can receive a copy with the project's source.
|
||||||
|
|
||||||
|
That's all for this week. Thank you everyone for your support!
|
||||||
|
|
||||||
|
# 0.5.0 (2018-09-11)
|
||||||
|
|
||||||
|
## Week 5: Privacy and Security
|
||||||
|
|
||||||
|
I hope everyone had a good weekend! This past week I've been fixing some issues that have been brought to my attention to help better protect users and help them keep their anonymity.
|
||||||
|
|
||||||
|
An issue with open referers has been fixed with [`29a2186`](https://github.com/omarroth/invidious/29a2186), which prevents potential redirects to external sites on actions such as login or modifying preferences.
|
||||||
|
|
||||||
|
Additionally, X-XSS-Protection, X-Content-Type-Options, and X-Frame-Options headers have been added with [`96234e5`](https://github.com/omarroth/invidious/96234e5), which should keep users safer while using the site.
|
||||||
|
|
||||||
|
A potential XSS vector has also been fixed in YouTube comments with [`8c45694`](https://github.com/omarroth/invidious/8c45694).
|
||||||
|
|
||||||
|
All the above vulnerabilities were brought to my attention by someone who wishes to remain anonymous, but I would like to say again here how thankful I am. If anyone else would like to get in touch please feel free to email me at omarroth@hotmail.com or omarroth@protonmail.com.
|
||||||
|
|
||||||
|
This week a couple changes have been made to better protect user's privacy as well.
|
||||||
|
All CSS and JS assets are now served locally with [`3ec684a`](https://github.com/omarroth/invidious/3ec684a), which means users no longer need to whitelist unpkg.com. Although I personally have encountered few issues, I understand that many folks would like to keep their browsing activity contained to as few parties as possible. In the coming week I also hope to proxy YouTube images, so that no user data is sent to Google.
|
||||||
|
|
||||||
|
YouTube links in comments now should redirect properly to the Invidious alternate with [`1c8bd67`](https://github.com/omarroth/invidious/1c8bd67) and [`cf63c82`](https://github.com/omarroth/invidious/cf63c82), so users can more easily evade Google tracking.
|
||||||
|
|
||||||
|
I'm also happy to mention a couple quality of life features this week:
|
||||||
|
|
||||||
|
Invidious now shows a video's "license" if provided, see [#159](https://github.com/omarroth/invidious/issues/159) for more details. You can also search for videos licensed under the creative commons with "QUERY features:creative_commons".
|
||||||
|
|
||||||
|
Videos with only one source will always display the cog for changing quality, so that users can see what quality is currently playing. See [#158](https://github.com/omarroth/invidious/issues/158) for more details.
|
||||||
|
|
||||||
|
Folks have also probably noticed that the gutters on either side of the screen have been shrunk down quite significantly, so that more of the screen is filled with content. Hopefully this can be improved even more in the coming weeks.
|
||||||
|
|
||||||
|
"Music", "Sports", and "Popular on YouTube" channels now properly display their videos. You can subscribe to these channels just as you would normally.
|
||||||
|
|
||||||
|
This coming week I'm planning on spending time with my family, so I unfortunately may not be as responsive. I do still hope to add some smaller features for next week however, and I hope to continue development soon.
|
||||||
|
Thank you everyone again for your support.
|
||||||
|
|
||||||
|
# 0.4.0 (2018-09-06)
|
||||||
|
|
||||||
|
## Week 4: Genre Channels
|
||||||
|
|
||||||
|
Hello! I hope everyone enjoyed their weekend. Without further ado:
|
||||||
|
Just today genre channels have been added with [#119](https://github.com/omarroth/invidious/issues/119). More information on genre channels is available [here](https://support.google.com/youtube/answer/2579942). You can subscribe to them as normally, and view them as RSS. I think they offer an interesting alternative way to find new content and I hope people find them useful.
|
||||||
|
|
||||||
|
This past week folks have started reporting 504s on their subscription page (see [#144](https://github.com/omarroth/invidious/issues/144) for more details). Upgrading the database server appeared to fix the issue, as well as providing a smoother experience across the site. Unfortunately, that means I will be increasing the goal from $50 to $60 in order to meet the increased hosting costs.
|
||||||
|
|
||||||
|
With [#134](https://github.com/omarroth/invidious/issues/134), comments are now formatted correctly, providing support for bold, italics, and links in comments. I think this improvement makes them much easier to read, and I hope others find the same. Also to note is that links in both comments and the video description now no longer contain any of Google's tracking with [#115](https://github.com/omarroth/invidious/issues/115).
|
||||||
|
|
||||||
|
One of the major use cases for Invidious is as a stripped-down version of YouTube. In line with that, I'm happy to announce that you can now hide related videos if you're logged in, for users that prefer an even more lightweight experience.
|
||||||
|
|
||||||
|
Finally, I'm pleased to announce that Invidious has hit 100 stars on GitHub. I am very happy that Invidious has proven to be useful to so many people, and I can't say how grateful I am to everyone for their continued support.
|
||||||
|
|
||||||
|
Enjoy the rest of your week everyone!
|
||||||
|
|
||||||
|
# 0.3.0 (2018-09-06)
|
||||||
|
|
||||||
|
## Week 3: Quality of Life
|
||||||
|
|
||||||
|
Hello everyone! This week I've been working on some smaller features that will hopefully make the site more functional.
|
||||||
|
Search filters have been added with [#126](https://github.com/omarroth/invidious/issues/126). You can now specify 'sort', 'date', 'duration', and 'features' within your query using the 'operator:value' syntax. I'd recommend taking a look [here](https://github.com/omarroth/invidious/blob/master/src/invidious/search.cr#L33-L114) for a list of supported options and at [#126](https://github.com/omarroth/invidious/issues/126) for some examples. This also opens the door for features such as [#30](https://github.com/omarroth/invidious/issues/30) which can be implemented as filters. I think advanced search is a major point in which Invidious can improve on YouTube and hope to add more features soon!
|
||||||
|
|
||||||
|
This week a more advanced system for viewing fallback comments has been added (see [#84](https://github.com/omarroth/invidious/issues/84) for more details). You can now specify a comment fallback in your preferences, which Invidious will use. If, for example, no Reddit comments are available for a given video, it can choose to fallback on YouTube comments. This also makes it possible to turn comments off completely for users that prefer a more streamlined experience.
|
||||||
|
|
||||||
|
With [#98](https://github.com/omarroth/invidious/issues/98), it is now possible for users to specify preferences without creating an account. You can now change speed, volume, subtitles, autoplay, loop, and quality using query parameters. See the issue above for more details and several examples.
|
||||||
|
|
||||||
|
I'd also like to announce that I've set up an account on [Liberapay](https://liberapay.com/omarroth), for patrons that prefer a privacy-friendly alternative to Patreon. Liberapay also does not take any percentage of donations, so I'd recommend donating some to the Liberapay for their hard work. Go check it out!
|
||||||
|
|
||||||
|
[Two weeks ago](https://github.com/omarroth/invidious/releases/tag/0.1.0) I mentioned adding 1080p support into the player. Currently, the only thing blocking is [#207](https://github.com/videojs/http-streaming/pull/207) in the excellent [http-streaming](https://github.com/videojs/http-streaming) library. I hope to work with the videojs team to merge it soon and finally implement 1080p support!
|
||||||
|
|
||||||
|
That's all for this week, thank you again everyone for your support!
|
||||||
|
|
||||||
|
# 0.2.0 (2018-09-06)
|
||||||
|
|
||||||
|
## Week 2: Toward Playlists
|
||||||
|
|
||||||
|
Sorry for the late update! Not as much to announce this week, but still a couple things of note:
|
||||||
|
I'm happy to announce that a playlists page and API endpoint has been added so you can now view playlists. Currently, you cannot watch playlists through the player, but I hope to add that in the coming week as well as adding functionality to add and modify playlists. There is a good conversation on [#114](https://github.com/omarroth/invidious/issues/114) about giving playlists even more functionality, which I think is interesting and would appreciate feedback on.
|
||||||
|
|
||||||
|
As an update to the Invidious API announcement last week, I've been working with [**@PrestonN**](https://github.com/PrestonN), the developer of [FreeTube](https://github.com/FreeTubeApp/FreeTube), to help migrate his project to the Invidious API. Because of it's increasing popularity, he has had trouble keeping under the quota set by YouTube's API. I hope to improve the API to meet his and others needs and I'd recommend folks to keep an eye on his excellent project! There is a good discussion with his thoughts [here](https://github.com/FreeTubeApp/FreeTube/issues/100).
|
||||||
|
|
||||||
|
A couple of miscellaneous features and bugfixes:
|
||||||
|
|
||||||
|
- You can now login to Invidious simultaneously from multiple devices - [#109](https://github.com/omarroth/invidious/issues/109)
|
||||||
|
|
||||||
|
- Added a note for scheduled livestreams - [#124](https://github.com/omarroth/invidious/issues/124)
|
||||||
|
|
||||||
|
- Changed YouTube comment header to "View x comments" - [#120](https://github.com/omarroth/invidious/issues/120)
|
||||||
|
|
||||||
|
Enjoy your week everyone!
|
||||||
|
|
||||||
|
# 0.1.0 (2018-09-06)
|
||||||
|
|
||||||
|
## Week 1: Invidious API and Geo-Bypass
|
||||||
|
|
||||||
|
Hello everyone! This past week there have been quite a few things worthy of mention:
|
||||||
|
|
||||||
|
I'm happy to announce the [Invidious Developer API](https://github.com/omarroth/invidious/wiki/API). The Invidious API does not use any of the official YouTube APIs, and instead crawls the site to provide a JSON interface for other developers to use. It's still under development but is already powering [CloudTube](https://github.com/cloudrac3r/cadencegq). The API currently does not have a quota (compared to YouTube) which I hope to continue thanks to continued support from my Patrons. Hopefully other developers find it useful, and I hope to continue to improve it so it can better serve the community.
|
||||||
|
|
||||||
|
Just today partial support for bypassing geo-restrictions has been added with [fada57a](https://github.com/omarroth/invidious/commit/fada57a307d66d696d9286fc943c579a3fd22de6). If a video is unblocked in one of: United States, Canada, Germany, France, Japan, Russia, or United Kingdom, then Invidious will be able to serve video info. Currently you will not yet be able to access the video files themselves, but in the coming week I hope to proxy videos so that users can enjoy content across borders.
|
||||||
|
|
||||||
|
Support for generating DASH manifests has been fixed, in the coming week I hope to integrate this functionality into the watch page, so users can view videos in 1080p and above.
|
||||||
|
|
||||||
|
Thank you everyone for your continued interest and support!
|
2
Makefile
2
Makefile
|
@ -42,7 +42,7 @@ get-libs:
|
||||||
|
|
||||||
# TODO: add support for ARM64 via cross-compilation
|
# TODO: add support for ARM64 via cross-compilation
|
||||||
invidious: get-libs
|
invidious: get-libs
|
||||||
crystal build src/invidious.cr $(FLAGS) --progress --stats --error-trace
|
crystal build src/invidious.cr $(FLAGS) --progress --stats --error-trace --mcpu=x86-64-v3
|
||||||
|
|
||||||
|
|
||||||
run: invidious
|
run: invidious
|
||||||
|
|
|
@ -279,7 +279,14 @@ div.thumbnail > .bottom-right-overlay {
|
||||||
display: inline;
|
display: inline;
|
||||||
}
|
}
|
||||||
|
|
||||||
.searchbar .pure-form fieldset { padding: 0; }
|
.searchbar .pure-form {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchbar .pure-form fieldset {
|
||||||
|
padding: 0;
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
.searchbar input[type="search"] {
|
.searchbar input[type="search"] {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -311,6 +318,16 @@ input[type="search"]::-webkit-search-cancel-button {
|
||||||
background-size: 14px;
|
background-size: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.searchbar #searchbutton {
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.searchbar #searchbutton:hover {
|
||||||
|
color: rgb(0, 182, 240);
|
||||||
|
}
|
||||||
|
|
||||||
.user-field {
|
.user-field {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
|
@ -444,31 +461,68 @@ p.video-data { margin: 0; font-weight: bold; font-size: 80%; }
|
||||||
*/
|
*/
|
||||||
|
|
||||||
footer {
|
footer {
|
||||||
margin-top: auto;
|
color: #919191;
|
||||||
|
margin-top: 2.5em;
|
||||||
padding: 1.5em 0;
|
padding: 1.5em 0;
|
||||||
text-align: center;
|
|
||||||
max-height: 30vh;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.light-theme footer {
|
#footer-content-container {
|
||||||
color: #7c7c7c;
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.dark-theme footer {
|
#footer-content-container > hr {
|
||||||
color: #adadad;
|
margin: 0;
|
||||||
|
color: rgb(241, 241, 241);
|
||||||
}
|
}
|
||||||
|
|
||||||
.light-theme footer a {
|
.footer-content {
|
||||||
color: #7c7c7c !important;
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
margin-top: -10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.dark-theme footer a {
|
footer a {
|
||||||
color: #adadad !important;
|
color: #919191;
|
||||||
}
|
}
|
||||||
|
|
||||||
footer span {
|
.footer-content #footer-custom-text > b {
|
||||||
margin: 4px 0;
|
font-size: 30px;
|
||||||
display: block;
|
}
|
||||||
|
|
||||||
|
.footer-section {
|
||||||
|
margin-right: 20px;
|
||||||
|
margin-top: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-section-list {
|
||||||
|
margin-top: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-section-item {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-footer .left {
|
||||||
|
float: left
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-footer .right {
|
||||||
|
float: right;
|
||||||
|
display: flex;
|
||||||
|
gap: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.footer-right .right a {
|
||||||
|
color: #919191
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 929px) {
|
||||||
|
#footer-custom-text {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* keyframes */
|
/* keyframes */
|
||||||
|
@ -534,6 +588,23 @@ span > select {
|
||||||
color: #565d64;
|
color: #565d64;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.light-theme footer {
|
||||||
|
color: #7a7a7a;
|
||||||
|
background: #f2f2f2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.light-theme #footer-content-container > hr {
|
||||||
|
color: rgb(241, 241, 241);
|
||||||
|
}
|
||||||
|
|
||||||
|
.light-theme footer a {
|
||||||
|
color: #7c7c7c !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.light-theme footer #footer-custom-text > b {
|
||||||
|
color: #565D64;
|
||||||
|
}
|
||||||
|
|
||||||
@media (prefers-color-scheme: light) {
|
@media (prefers-color-scheme: light) {
|
||||||
.no-theme a:hover,
|
.no-theme a:hover,
|
||||||
.no-theme a:active,
|
.no-theme a:active,
|
||||||
|
@ -569,17 +640,22 @@ span > select {
|
||||||
color: #303030;
|
color: #303030;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.light-theme .pure-menu-heading {
|
||||||
|
color: #565d64;
|
||||||
|
}
|
||||||
|
|
||||||
.no-theme footer {
|
.no-theme footer {
|
||||||
|
background: #f2f2f2;
|
||||||
color: #7c7c7c;
|
color: #7c7c7c;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.no-theme footer #footer-custom-text > b {
|
||||||
|
color: #565D64;
|
||||||
|
}
|
||||||
|
|
||||||
.no-theme footer a {
|
.no-theme footer a {
|
||||||
color: #7c7c7c !important;
|
color: #7c7c7c !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.light-theme .pure-menu-heading {
|
|
||||||
color: #565d64;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -642,6 +718,20 @@ body.dark-theme {
|
||||||
color: inherit;
|
color: inherit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.dark-theme footer {
|
||||||
|
background: #16191a;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dark-theme #footer-content-container > hr {
|
||||||
|
color: #313131;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dark-theme .footer-content #footer-custom-text > b {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
@media (prefers-color-scheme: dark) {
|
@media (prefers-color-scheme: dark) {
|
||||||
.no-theme a:hover,
|
.no-theme a:hover,
|
||||||
.no-theme a:active,
|
.no-theme a:active,
|
||||||
|
@ -697,11 +787,12 @@ body.dark-theme {
|
||||||
}
|
}
|
||||||
|
|
||||||
.no-theme footer {
|
.no-theme footer {
|
||||||
color: #adadad;
|
background: #16191a;
|
||||||
|
color: #313131;
|
||||||
}
|
}
|
||||||
|
|
||||||
.no-theme footer a {
|
.no-theme footer #footer-custom-text > b {
|
||||||
color: #adadad !important;
|
color: #ccc;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -787,6 +878,17 @@ h1, h2, h3, h4, h5, p,
|
||||||
/* Center the "invidious" logo on the search page */
|
/* Center the "invidious" logo on the search page */
|
||||||
#logo > h1 { text-align: center; }
|
#logo > h1 { text-align: center; }
|
||||||
|
|
||||||
|
#footer_buffer {
|
||||||
|
margin-top: 50vh;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 450px) {
|
||||||
|
#footer_buffer {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* IE11 fixes */
|
/* IE11 fixes */
|
||||||
:-ms-input-placeholder { color: #888; }
|
:-ms-input-placeholder { color: #888; }
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,6 @@ var player_data = JSON.parse(document.getElementById('player_data').textContent)
|
||||||
var video_data = JSON.parse(document.getElementById('video_data').textContent);
|
var video_data = JSON.parse(document.getElementById('video_data').textContent);
|
||||||
|
|
||||||
var options = {
|
var options = {
|
||||||
preload: 'auto',
|
|
||||||
liveui: true,
|
liveui: true,
|
||||||
playbackRates: [0.25, 0.5, 0.75, 1.0, 1.25, 1.5, 1.75, 2.0],
|
playbackRates: [0.25, 0.5, 0.75, 1.0, 1.25, 1.5, 1.75, 2.0],
|
||||||
controlBar: {
|
controlBar: {
|
||||||
|
@ -51,6 +50,9 @@ videojs.Vhs.xhr.beforeRequest = function(options) {
|
||||||
return options;
|
return options;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
videojs.Vhs.GOAL_BUFFER_LENGTH = 40;
|
||||||
|
videojs.Vhs.MAX_GOAL_BUFFER_LENGTH = 80;
|
||||||
|
|
||||||
var player = videojs('player', options);
|
var player = videojs('player', options);
|
||||||
|
|
||||||
player.on('error', function () {
|
player.on('error', function () {
|
||||||
|
@ -352,6 +354,7 @@ if (video_data.params.save_player_pos) {
|
||||||
let lastUpdated = 0;
|
let lastUpdated = 0;
|
||||||
|
|
||||||
if(!hasTimeParam) set_seconds_after_start(rememberedTime);
|
if(!hasTimeParam) set_seconds_after_start(rememberedTime);
|
||||||
|
if (rememberedTime >= video_data.length_seconds - 20 && !hasTimeParam) set_seconds_after_start(0);
|
||||||
|
|
||||||
player.on('timeupdate', function () {
|
player.on('timeupdate', function () {
|
||||||
const raw = player.currentTime();
|
const raw = player.currentTime();
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
#########################################
|
#########################################
|
||||||
#
|
#
|
||||||
# Database configuration
|
# Database and other external servers
|
||||||
#
|
#
|
||||||
#########################################
|
#########################################
|
||||||
|
|
||||||
|
@ -41,6 +41,19 @@ db:
|
||||||
#check_tables: false
|
#check_tables: false
|
||||||
|
|
||||||
|
|
||||||
|
##
|
||||||
|
## Path to an external signature resolver, used to emulate
|
||||||
|
## the Youtube client's Javascript. If no such server is
|
||||||
|
## available, some videos will not be playable.
|
||||||
|
##
|
||||||
|
## When this setting is commented out, no external
|
||||||
|
## resolver will be used.
|
||||||
|
##
|
||||||
|
## Accepted values: a path to a UNIX socket or "<IP>:<Port>"
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
#signature_server:
|
||||||
|
|
||||||
|
|
||||||
#########################################
|
#########################################
|
||||||
#
|
#
|
||||||
|
@ -173,6 +186,18 @@ https_only: false
|
||||||
##
|
##
|
||||||
# use_innertube_for_captions: false
|
# use_innertube_for_captions: false
|
||||||
|
|
||||||
|
##
|
||||||
|
## Send Google session informations. This is useful when Invidious is blocked
|
||||||
|
## by the message "This helps protect our community."
|
||||||
|
## See https://github.com/iv-org/invidious/issues/4734.
|
||||||
|
##
|
||||||
|
## Warning: These strings gives much more identifiable information to Google!
|
||||||
|
##
|
||||||
|
## Accepted values: String
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
# po_token: ""
|
||||||
|
# visitor_data: ""
|
||||||
|
|
||||||
# -----------------------------
|
# -----------------------------
|
||||||
# Logging
|
# Logging
|
||||||
|
@ -197,6 +222,15 @@ https_only: false
|
||||||
##
|
##
|
||||||
#log_level: Info
|
#log_level: Info
|
||||||
|
|
||||||
|
##
|
||||||
|
## Enables colors in logs. Useful for debugging purposes
|
||||||
|
## This is overridden if "-k" or "--colorize"
|
||||||
|
## are passed on the command line.
|
||||||
|
##
|
||||||
|
## Accepted values: true, false
|
||||||
|
## Default: false
|
||||||
|
##
|
||||||
|
#colorize_logs: false
|
||||||
|
|
||||||
# -----------------------------
|
# -----------------------------
|
||||||
# Features
|
# Features
|
||||||
|
@ -343,21 +377,6 @@ full_refresh: false
|
||||||
##
|
##
|
||||||
feed_threads: 1
|
feed_threads: 1
|
||||||
|
|
||||||
##
|
|
||||||
## Enable/Disable the polling job that keeps the decryption
|
|
||||||
## function (for "secured" videos) up to date.
|
|
||||||
##
|
|
||||||
## Note: This part of the code generate a small amount of data every minute.
|
|
||||||
## This may not be desired if you have bandwidth limits set by your ISP.
|
|
||||||
##
|
|
||||||
## Note 2: This part of the code is currently broken, so changing
|
|
||||||
## this setting has no impact.
|
|
||||||
##
|
|
||||||
## Accepted values: true, false
|
|
||||||
## Default: false
|
|
||||||
##
|
|
||||||
#decrypt_polling: false
|
|
||||||
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|
||||||
|
@ -391,9 +410,8 @@ jobs:
|
||||||
##
|
##
|
||||||
enable: true
|
enable: true
|
||||||
|
|
||||||
|
|
||||||
# -----------------------------
|
# -----------------------------
|
||||||
# Miscellaneous
|
# Instance customization
|
||||||
# -----------------------------
|
# -----------------------------
|
||||||
|
|
||||||
##
|
##
|
||||||
|
@ -405,6 +423,78 @@ jobs:
|
||||||
##
|
##
|
||||||
#banner:
|
#banner:
|
||||||
|
|
||||||
|
##
|
||||||
|
## custom text displayed at the bottom of every page within Invidious' footer. This can
|
||||||
|
## used for instance announcements, e.g
|
||||||
|
##
|
||||||
|
## When unset Invidious defaults to some text that describes what Invidious is. See
|
||||||
|
## localization key default_invidious_footer_text
|
||||||
|
##
|
||||||
|
## Accepted values: any string. HTML is accepted.
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
#footer:
|
||||||
|
|
||||||
|
##
|
||||||
|
## Source code URL. If your instance is running a modified source
|
||||||
|
## code, you MUST publish it somewhere and set this option.
|
||||||
|
##
|
||||||
|
## Accepted values: a string
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
#modified_source_code_url: ""
|
||||||
|
|
||||||
|
##
|
||||||
|
## Email to contact the instance maintainer; used in a mailto: link within the footer.
|
||||||
|
##
|
||||||
|
## Accepted values: string
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
# instance_maintainer_email:
|
||||||
|
|
||||||
|
##
|
||||||
|
## Link to the terms of service of the instance (if any)
|
||||||
|
##
|
||||||
|
## Displayed within the instance section of the footer
|
||||||
|
##
|
||||||
|
## Accepted values: String (link)
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
# footer_instance_tos_link:
|
||||||
|
|
||||||
|
##
|
||||||
|
## Link to the privacy-policy of the instance (if any)
|
||||||
|
##
|
||||||
|
## Displayed within the instance section of the footer
|
||||||
|
##
|
||||||
|
## Accepted values: String (link)
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
# footer_instance_privacy_policy_link:
|
||||||
|
|
||||||
|
##
|
||||||
|
## Instance donation URL. If your instance has a donation option.
|
||||||
|
## you can add it here so it will be present in the footer along
|
||||||
|
## with the donation link for the project itself.
|
||||||
|
##
|
||||||
|
## Accepted values: a string
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
#footer_instance_donate_link: ""
|
||||||
|
|
||||||
|
##
|
||||||
|
## Custom fields to be displayed within the footer's instance section
|
||||||
|
##
|
||||||
|
## Accepted values: A nested array mapping field name and links together.
|
||||||
|
## IE: [ ["field1", "https://example.com/1"], ["field2", "https://example.com/2"] ]
|
||||||
|
## Default: <none>
|
||||||
|
##
|
||||||
|
# footer_instance_section_custom_fields: []
|
||||||
|
|
||||||
|
# -----------------------------
|
||||||
|
# Miscellaneous
|
||||||
|
# -----------------------------
|
||||||
|
|
||||||
##
|
##
|
||||||
## Subscribe to channels using PubSubHub (Google PubSubHubbub service).
|
## Subscribe to channels using PubSubHub (Google PubSubHubbub service).
|
||||||
## PubSubHub allows Invidious to be instantly notified when a new video
|
## PubSubHub allows Invidious to be instantly notified when a new video
|
||||||
|
@ -458,15 +548,6 @@ hmac_key: "CHANGE_ME!!"
|
||||||
##
|
##
|
||||||
#cache_annotations: false
|
#cache_annotations: false
|
||||||
|
|
||||||
##
|
|
||||||
## Source code URL. If your instance is running a modified source
|
|
||||||
## code, you MUST publish it somewhere and set this option.
|
|
||||||
##
|
|
||||||
## Accepted values: a string
|
|
||||||
## Default: <none>
|
|
||||||
##
|
|
||||||
#modified_source_code_url: ""
|
|
||||||
|
|
||||||
##
|
##
|
||||||
## Maximum custom playlist length limit.
|
## Maximum custom playlist length limit.
|
||||||
##
|
##
|
||||||
|
@ -475,6 +556,7 @@ hmac_key: "CHANGE_ME!!"
|
||||||
##
|
##
|
||||||
#playlist_length_limit: 500
|
#playlist_length_limit: 500
|
||||||
|
|
||||||
|
|
||||||
#########################################
|
#########################################
|
||||||
#
|
#
|
||||||
# Default user preferences
|
# Default user preferences
|
||||||
|
@ -697,6 +779,22 @@ default_user_preferences:
|
||||||
# Video player behavior
|
# Video player behavior
|
||||||
# -----------------------------
|
# -----------------------------
|
||||||
|
|
||||||
|
##
|
||||||
|
## This option controls the value of the HTML5 <video> element's
|
||||||
|
## "preload" attribute.
|
||||||
|
##
|
||||||
|
## If set to 'false', no video data will be loaded until the user
|
||||||
|
## explicitly starts the video by clicking the "Play" button.
|
||||||
|
## If set to 'true', the web browser will buffer some video data
|
||||||
|
## while the page is loading.
|
||||||
|
##
|
||||||
|
## See: https://www.w3schools.com/tags/att_video_preload.asp
|
||||||
|
##
|
||||||
|
## Accepted values: true, false
|
||||||
|
## Default: true
|
||||||
|
##
|
||||||
|
#preload: true
|
||||||
|
|
||||||
##
|
##
|
||||||
## Automatically play videos on page load.
|
## Automatically play videos on page load.
|
||||||
##
|
##
|
||||||
|
@ -791,7 +889,7 @@ default_user_preferences:
|
||||||
## Default: true
|
## Default: true
|
||||||
##
|
##
|
||||||
#vr_mode: true
|
#vr_mode: true
|
||||||
|
|
||||||
##
|
##
|
||||||
## Save the playback position
|
## Save the playback position
|
||||||
## Allow to continue watching at the previous position when
|
## Allow to continue watching at the previous position when
|
||||||
|
@ -899,3 +997,9 @@ default_user_preferences:
|
||||||
## Default: false
|
## Default: false
|
||||||
##
|
##
|
||||||
#extend_desc: false
|
#extend_desc: false
|
||||||
|
|
||||||
|
# redis_url: 127.0.0.1:6379
|
||||||
|
# redis_socket: /var/run/valkey/valkey.sock
|
||||||
|
# donation_url: "https://example.com/donate"
|
||||||
|
# contact_url: "https://example.com/contact"
|
||||||
|
# home_domain: "https://example.com/
|
||||||
|
|
6
config/migrate-scripts/migrate-db-8bc91ce.sh
Normal file
6
config/migrate-scripts/migrate-db-8bc91ce.sh
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
CREATE INDEX channel_videos_ucid_published_idx
|
||||||
|
ON public.channel_videos
|
||||||
|
USING btree
|
||||||
|
(ucid COLLATE pg_catalog."default", published);
|
||||||
|
|
||||||
|
DROP INDEX channel_videos_ucid_idx;
|
|
@ -19,12 +19,12 @@ CREATE TABLE IF NOT EXISTS public.channel_videos
|
||||||
|
|
||||||
GRANT ALL ON TABLE public.channel_videos TO current_user;
|
GRANT ALL ON TABLE public.channel_videos TO current_user;
|
||||||
|
|
||||||
-- Index: public.channel_videos_ucid_idx
|
-- Index: public.channel_videos_ucid_published_idx
|
||||||
|
|
||||||
-- DROP INDEX public.channel_videos_ucid_idx;
|
-- DROP INDEX public.channel_videos_ucid_published_idx;
|
||||||
|
|
||||||
CREATE INDEX IF NOT EXISTS channel_videos_ucid_idx
|
CREATE INDEX IF NOT EXISTS channel_videos_ucid_published_idx
|
||||||
ON public.channel_videos
|
ON public.channel_videos
|
||||||
USING btree
|
USING btree
|
||||||
(ucid COLLATE pg_catalog."default");
|
(ucid COLLATE pg_catalog."default", published);
|
||||||
|
|
||||||
|
|
|
@ -1,55 +1,56 @@
|
||||||
# Warning: This docker-compose file is made for development purposes.
|
# Docker compose file for inv.nadeko.net
|
||||||
# Using it will build an image from the locally cloned repository.
|
|
||||||
#
|
|
||||||
# If you want to use Invidious in production, see the docker-compose.yml file provided
|
|
||||||
# in the installation documentation: https://docs.invidious.io/installation/
|
|
||||||
|
|
||||||
version: "3"
|
|
||||||
services:
|
services:
|
||||||
|
valkey:
|
||||||
invidious:
|
image: valkey/valkey:7.2-alpine
|
||||||
build:
|
|
||||||
context: .
|
|
||||||
dockerfile: docker/Dockerfile
|
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
ports:
|
volumes:
|
||||||
- "127.0.0.1:3000:3000"
|
- "./valkey:/data"
|
||||||
|
|
||||||
|
invidious-refresher:
|
||||||
|
image: git.nadeko.net/fijxu/invidious:latest
|
||||||
|
restart: unless-stopped
|
||||||
|
volumes:
|
||||||
|
- ./config/config-refresher.yml:/etc/invidious/config.yml:ro
|
||||||
|
- ./logs:/var/log/invidious:rw
|
||||||
|
- /var/run/postgresql/.s.PGSQL.5432:/var/run/postgresql/.s.PGSQL.5432:rw
|
||||||
environment:
|
environment:
|
||||||
# Please read the following file for a comprehensive list of all available
|
INVIDIOUS_CONFIG_FILE: /etc/invidious/config.yml
|
||||||
# configuration options and their associated syntax:
|
depends_on:
|
||||||
# https://github.com/iv-org/invidious/blob/master/config/config.example.yml
|
- valkey
|
||||||
INVIDIOUS_CONFIG: |
|
|
||||||
db:
|
|
||||||
dbname: invidious
|
|
||||||
user: kemal
|
|
||||||
password: kemal
|
|
||||||
host: invidious-db
|
|
||||||
port: 5432
|
|
||||||
check_tables: true
|
|
||||||
# external_port:
|
|
||||||
# domain:
|
|
||||||
# https_only: false
|
|
||||||
# statistics_enabled: false
|
|
||||||
hmac_key: "CHANGE_ME!!"
|
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test: wget -nv --tries=1 --spider http://127.0.0.1:3000/api/v1/comments/jNQXAC9IVRw || exit 1
|
test: wget -nv --tries=1 --spider http://127.0.0.1:3000/api/v1/trending || exit 1
|
||||||
interval: 30s
|
interval: 30s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 2
|
retries: 2
|
||||||
|
|
||||||
invidious-db:
|
|
||||||
image: docker.io/library/postgres:14
|
invidious:
|
||||||
|
image: git.nadeko.net/fijxu/invidious:latest
|
||||||
|
restart: unless-stopped
|
||||||
|
deploy:
|
||||||
|
replicas: 8
|
||||||
|
volumes:
|
||||||
|
- ./config/config.yml:/etc/invidious/config.yml:ro
|
||||||
|
- ./logs:/var/log/invidious:rw
|
||||||
|
- /var/run/postgresql/.s.PGSQL.5432:/var/run/postgresql/.s.PGSQL.5432:rw
|
||||||
|
environment:
|
||||||
|
INVIDIOUS_CONFIG_FILE: /etc/invidious/config.yml
|
||||||
|
depends_on:
|
||||||
|
- valkey
|
||||||
|
healthcheck:
|
||||||
|
test: wget -nv --tries=1 --spider http://127.0.0.1:3000/api/v1/trending || exit 1
|
||||||
|
interval: 30s
|
||||||
|
timeout: 5s
|
||||||
|
retries: 2
|
||||||
|
|
||||||
|
|
||||||
|
invidious-nginx:
|
||||||
|
image: nginx:latest
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
volumes:
|
volumes:
|
||||||
- postgresdata:/var/lib/postgresql/data
|
- ./nginx.conf:/etc/nginx/nginx.conf:ro
|
||||||
- ./config/sql:/config/sql
|
depends_on:
|
||||||
- ./docker/init-invidious-db.sh:/docker-entrypoint-initdb.d/init-invidious-db.sh
|
- invidious
|
||||||
environment:
|
ports:
|
||||||
POSTGRES_DB: invidious
|
- "127.0.0.1:10011:3000"
|
||||||
POSTGRES_USER: kemal
|
|
||||||
POSTGRES_PASSWORD: kemal
|
|
||||||
healthcheck:
|
|
||||||
test: ["CMD-SHELL", "pg_isready -U $$POSTGRES_USER -d $$POSTGRES_DB"]
|
|
||||||
|
|
||||||
volumes:
|
|
||||||
postgresdata:
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
FROM crystallang/crystal:1.8.2-alpine AS builder
|
FROM crystallang/crystal:1.12.1-alpine AS builder
|
||||||
|
|
||||||
RUN apk add --no-cache sqlite-static yaml-static
|
RUN apk add --no-cache sqlite-static yaml-static
|
||||||
|
|
||||||
|
@ -20,10 +20,10 @@ COPY ./assets/ ./assets/
|
||||||
COPY ./videojs-dependencies.yml ./videojs-dependencies.yml
|
COPY ./videojs-dependencies.yml ./videojs-dependencies.yml
|
||||||
|
|
||||||
RUN crystal spec --warnings all \
|
RUN crystal spec --warnings all \
|
||||||
--link-flags "-lxml2 -llzma"
|
--link-flags "-lxml2 -llzma"
|
||||||
RUN if [[ "${release}" == 1 ]] ; then \
|
RUN if [[ "${release}" == 1 ]] ; then \
|
||||||
crystal build ./src/invidious.cr \
|
crystal build ./src/invidious.cr \
|
||||||
--release \
|
--release --mcpu=x86-64-v3 \
|
||||||
--static --warnings all \
|
--static --warnings all \
|
||||||
--link-flags "-lxml2 -llzma"; \
|
--link-flags "-lxml2 -llzma"; \
|
||||||
else \
|
else \
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
FROM alpine:3.18 AS builder
|
FROM alpine:3.19 AS builder
|
||||||
RUN apk add --no-cache 'crystal=1.8.2-r0' shards sqlite-static yaml-static yaml-dev libxml2-static zlib-static openssl-libs-static openssl-dev musl-dev xz-static
|
RUN apk add --no-cache 'crystal=1.10.1-r0' shards sqlite-static yaml-static yaml-dev libxml2-static zlib-static openssl-libs-static openssl-dev musl-dev xz-static
|
||||||
|
|
||||||
ARG release
|
ARG release
|
||||||
|
|
||||||
|
|
1
kubernetes/.gitignore
vendored
1
kubernetes/.gitignore
vendored
|
@ -1 +0,0 @@
|
||||||
/charts/*.tgz
|
|
|
@ -1,6 +0,0 @@
|
||||||
dependencies:
|
|
||||||
- name: postgresql
|
|
||||||
repository: https://charts.bitnami.com/bitnami/
|
|
||||||
version: 12.11.1
|
|
||||||
digest: sha256:3c10008175c4f5c1cec38782f5a7316154b89074c77ebbd9bcc4be4f5ff21122
|
|
||||||
generated: "2023-09-14T22:40:43.171275362Z"
|
|
|
@ -1,22 +0,0 @@
|
||||||
apiVersion: v2
|
|
||||||
name: invidious
|
|
||||||
description: Invidious is an alternative front-end to YouTube
|
|
||||||
version: 1.1.1
|
|
||||||
appVersion: 0.20.1
|
|
||||||
keywords:
|
|
||||||
- youtube
|
|
||||||
- proxy
|
|
||||||
- video
|
|
||||||
- privacy
|
|
||||||
home: https://invidio.us/
|
|
||||||
icon: https://raw.githubusercontent.com/iv-org/invidious/05988c1c49851b7d0094fca16aeaf6382a7f64ab/assets/favicon-32x32.png
|
|
||||||
sources:
|
|
||||||
- https://github.com/iv-org/invidious
|
|
||||||
maintainers:
|
|
||||||
- name: Leon Klingele
|
|
||||||
email: mail@leonklingele.de
|
|
||||||
dependencies:
|
|
||||||
- name: postgresql
|
|
||||||
version: ~12.11.0
|
|
||||||
repository: "https://charts.bitnami.com/bitnami/"
|
|
||||||
engine: gotpl
|
|
|
@ -1,41 +1 @@
|
||||||
# Invidious Helm chart
|
The Helm chart has moved to a dedicated GitHub repository: https://github.com/iv-org/invidious-helm-chart/tree/master/invidious
|
||||||
|
|
||||||
Easily deploy Invidious to Kubernetes.
|
|
||||||
|
|
||||||
## Installing Helm chart
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# Build Helm dependencies
|
|
||||||
$ helm dep build
|
|
||||||
|
|
||||||
# Add PostgreSQL init scripts
|
|
||||||
$ kubectl create configmap invidious-postgresql-init \
|
|
||||||
--from-file=../config/sql/channels.sql \
|
|
||||||
--from-file=../config/sql/videos.sql \
|
|
||||||
--from-file=../config/sql/channel_videos.sql \
|
|
||||||
--from-file=../config/sql/users.sql \
|
|
||||||
--from-file=../config/sql/session_ids.sql \
|
|
||||||
--from-file=../config/sql/nonces.sql \
|
|
||||||
--from-file=../config/sql/annotations.sql \
|
|
||||||
--from-file=../config/sql/playlists.sql \
|
|
||||||
--from-file=../config/sql/playlist_videos.sql
|
|
||||||
|
|
||||||
# Install Helm app to your Kubernetes cluster
|
|
||||||
$ helm install invidious ./
|
|
||||||
```
|
|
||||||
|
|
||||||
## Upgrading
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# Upgrading is easy, too!
|
|
||||||
$ helm upgrade invidious ./
|
|
||||||
```
|
|
||||||
|
|
||||||
## Uninstall
|
|
||||||
|
|
||||||
```sh
|
|
||||||
# Get rid of everything (except database)
|
|
||||||
$ helm delete invidious
|
|
||||||
|
|
||||||
# To also delete the database, remove all invidious-postgresql PVCs
|
|
||||||
```
|
|
|
@ -1,16 +0,0 @@
|
||||||
{{/* vim: set filetype=mustache: */}}
|
|
||||||
{{/*
|
|
||||||
Expand the name of the chart.
|
|
||||||
*/}}
|
|
||||||
{{- define "invidious.name" -}}
|
|
||||||
{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" -}}
|
|
||||||
{{- end -}}
|
|
||||||
|
|
||||||
{{/*
|
|
||||||
Create a default fully qualified app name.
|
|
||||||
We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
|
|
||||||
*/}}
|
|
||||||
{{- define "invidious.fullname" -}}
|
|
||||||
{{- $name := default .Chart.Name .Values.nameOverride -}}
|
|
||||||
{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" -}}
|
|
||||||
{{- end -}}
|
|
|
@ -1,11 +0,0 @@
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
metadata:
|
|
||||||
name: {{ template "invidious.fullname" . }}
|
|
||||||
labels:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
chart: "{{ .Chart.Name }}-{{ .Chart.Version }}"
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
data:
|
|
||||||
INVIDIOUS_CONFIG: |
|
|
||||||
{{ toYaml .Values.config | indent 4 }}
|
|
|
@ -1,61 +0,0 @@
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
metadata:
|
|
||||||
name: {{ template "invidious.fullname" . }}
|
|
||||||
labels:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
chart: "{{ .Chart.Name }}-{{ .Chart.Version }}"
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
spec:
|
|
||||||
replicas: {{ .Values.replicaCount }}
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
labels:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
chart: "{{ .Chart.Name }}-{{ .Chart.Version }}"
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
spec:
|
|
||||||
securityContext:
|
|
||||||
runAsUser: {{ .Values.securityContext.runAsUser }}
|
|
||||||
runAsGroup: {{ .Values.securityContext.runAsGroup }}
|
|
||||||
fsGroup: {{ .Values.securityContext.fsGroup }}
|
|
||||||
initContainers:
|
|
||||||
- name: wait-for-postgresql
|
|
||||||
image: postgres
|
|
||||||
args:
|
|
||||||
- /bin/sh
|
|
||||||
- -c
|
|
||||||
- until pg_isready -h {{ .Values.config.db.host }} -p {{ .Values.config.db.port }} -U {{ .Values.config.db.user }}; do echo waiting for database; sleep 2; done;
|
|
||||||
containers:
|
|
||||||
- name: {{ .Chart.Name }}
|
|
||||||
image: "{{ .Values.image.repository }}:{{ .Values.image.tag }}"
|
|
||||||
imagePullPolicy: {{ .Values.image.pullPolicy }}
|
|
||||||
ports:
|
|
||||||
- containerPort: 3000
|
|
||||||
env:
|
|
||||||
- name: INVIDIOUS_CONFIG
|
|
||||||
valueFrom:
|
|
||||||
configMapKeyRef:
|
|
||||||
key: INVIDIOUS_CONFIG
|
|
||||||
name: {{ template "invidious.fullname" . }}
|
|
||||||
securityContext:
|
|
||||||
allowPrivilegeEscalation: {{ .Values.securityContext.allowPrivilegeEscalation }}
|
|
||||||
capabilities:
|
|
||||||
drop:
|
|
||||||
- ALL
|
|
||||||
resources:
|
|
||||||
{{ toYaml .Values.resources | indent 10 }}
|
|
||||||
readinessProbe:
|
|
||||||
httpGet:
|
|
||||||
port: 3000
|
|
||||||
path: /
|
|
||||||
livenessProbe:
|
|
||||||
httpGet:
|
|
||||||
port: 3000
|
|
||||||
path: /
|
|
||||||
initialDelaySeconds: 15
|
|
||||||
restartPolicy: Always
|
|
|
@ -1,18 +0,0 @@
|
||||||
{{- if .Values.autoscaling.enabled }}
|
|
||||||
apiVersion: autoscaling/v1
|
|
||||||
kind: HorizontalPodAutoscaler
|
|
||||||
metadata:
|
|
||||||
name: {{ template "invidious.fullname" . }}
|
|
||||||
labels:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
chart: "{{ .Chart.Name }}-{{ .Chart.Version }}"
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
spec:
|
|
||||||
scaleTargetRef:
|
|
||||||
apiVersion: apps/v1
|
|
||||||
kind: Deployment
|
|
||||||
name: {{ template "invidious.fullname" . }}
|
|
||||||
minReplicas: {{ .Values.autoscaling.minReplicas }}
|
|
||||||
maxReplicas: {{ .Values.autoscaling.maxReplicas }}
|
|
||||||
targetCPUUtilizationPercentage: {{ .Values.autoscaling.targetCPUUtilizationPercentage }}
|
|
||||||
{{- end }}
|
|
|
@ -1,20 +0,0 @@
|
||||||
apiVersion: v1
|
|
||||||
kind: Service
|
|
||||||
metadata:
|
|
||||||
name: {{ template "invidious.fullname" . }}
|
|
||||||
labels:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
chart: {{ .Chart.Name }}
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
spec:
|
|
||||||
type: {{ .Values.service.type }}
|
|
||||||
ports:
|
|
||||||
- name: http
|
|
||||||
port: {{ .Values.service.port }}
|
|
||||||
targetPort: 3000
|
|
||||||
selector:
|
|
||||||
app: {{ template "invidious.name" . }}
|
|
||||||
release: {{ .Release.Name }}
|
|
||||||
{{- if .Values.service.loadBalancerIP }}
|
|
||||||
loadBalancerIP: {{ .Values.service.loadBalancerIP }}
|
|
||||||
{{- end }}
|
|
|
@ -49,7 +49,6 @@ postgresql:
|
||||||
# Adapted from ../config/config.yml
|
# Adapted from ../config/config.yml
|
||||||
config:
|
config:
|
||||||
channel_threads: 1
|
channel_threads: 1
|
||||||
feed_threads: 1
|
|
||||||
db:
|
db:
|
||||||
user: kemal
|
user: kemal
|
||||||
password: kemal
|
password: kemal
|
||||||
|
|
|
@ -15,13 +15,13 @@
|
||||||
"New password": "كلمة مرور جديدة",
|
"New password": "كلمة مرور جديدة",
|
||||||
"New passwords must match": "يَجبُ أن تكون كلمتا المرور متطابقتين",
|
"New passwords must match": "يَجبُ أن تكون كلمتا المرور متطابقتين",
|
||||||
"Authorize token?": "رمز التفويض؟",
|
"Authorize token?": "رمز التفويض؟",
|
||||||
"Authorize token for `x`?": "السماح بالرمز المميز ل 'x'؟",
|
"Authorize token for `x`?": "السماح بالرمز المميز ل `x`؟",
|
||||||
"Yes": "نعم",
|
"Yes": "نعم",
|
||||||
"No": "لا",
|
"No": "لا",
|
||||||
"Import and Export Data": "اِستيراد البيانات وتصديرها",
|
"Import and Export Data": "اِستيراد البيانات وتصديرها",
|
||||||
"Import": "استيراد",
|
"Import": "استيراد",
|
||||||
"Import Invidious data": "استيراد بيانات JSON Invidious",
|
"Import Invidious data": "استيراد بيانات JSON Invidious",
|
||||||
"Import YouTube subscriptions": "استيراد اشتراكات YouTube/OPML",
|
"Import YouTube subscriptions": "استيراد الاشتراكات YouTube بتنسيق CSV أو OPML",
|
||||||
"Import FreeTube subscriptions (.db)": "استيراد اشتراكات فريتيوب (.db)",
|
"Import FreeTube subscriptions (.db)": "استيراد اشتراكات فريتيوب (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "استيراد اشتراكات نيو بايب (.json)",
|
"Import NewPipe subscriptions (.json)": "استيراد اشتراكات نيو بايب (.json)",
|
||||||
"Import NewPipe data (.zip)": "استيراد بيانات نيو بايب (.zip)",
|
"Import NewPipe data (.zip)": "استيراد بيانات نيو بايب (.zip)",
|
||||||
|
@ -170,7 +170,7 @@
|
||||||
"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 log 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.": "هذه القناة غير موجودة.",
|
||||||
|
@ -382,11 +382,11 @@
|
||||||
"videoinfo_watch_on_youTube": "مشاهدة على يوتيوب",
|
"videoinfo_watch_on_youTube": "مشاهدة على يوتيوب",
|
||||||
"videoinfo_youTube_embed_link": "مضمن",
|
"videoinfo_youTube_embed_link": "مضمن",
|
||||||
"videoinfo_invidious_embed_link": "رابط مضمن",
|
"videoinfo_invidious_embed_link": "رابط مضمن",
|
||||||
"user_created_playlists": "'x' إنشاء قوائم التشغيل",
|
"user_created_playlists": "`x` إنشاء قوائم التشغيل",
|
||||||
"user_saved_playlists": "قوائم التشغيل المحفوظة 'x'",
|
"user_saved_playlists": "قوائم التشغيل المحفوظة `x`",
|
||||||
"Video unavailable": "الفيديو غير متوفر",
|
"Video unavailable": "الفيديو غير متوفر",
|
||||||
"search_filters_features_option_three_sixty": "360°",
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
"download_subtitles": "ترجمات - 'x' (.vtt)",
|
"download_subtitles": "ترجمات - `x` (.vtt)",
|
||||||
"invidious": "الخيالي",
|
"invidious": "الخيالي",
|
||||||
"preferences_save_player_pos_label": "حفظ موضع التشغيل: ",
|
"preferences_save_player_pos_label": "حفظ موضع التشغيل: ",
|
||||||
"crash_page_you_found_a_bug": "يبدو أنك قد وجدت خطأً برمجيًّا في Invidious!",
|
"crash_page_you_found_a_bug": "يبدو أنك قد وجدت خطأً برمجيًّا في Invidious!",
|
||||||
|
@ -483,7 +483,7 @@
|
||||||
"comments_view_x_replies_3": "عرض رد {{count}}",
|
"comments_view_x_replies_3": "عرض رد {{count}}",
|
||||||
"comments_view_x_replies_4": "عرض الردود {{count}}",
|
"comments_view_x_replies_4": "عرض الردود {{count}}",
|
||||||
"comments_view_x_replies_5": "عرض رد {{count}}",
|
"comments_view_x_replies_5": "عرض رد {{count}}",
|
||||||
"search_message_use_another_instance": " يمكنك أيضًا البحث عن <a href=\"`x`\"> في مثيل آخر </a>.",
|
"search_message_use_another_instance": "يمكنك أيضًا البحث عن <a href=\"`x`\"> في مثيل آخر </a>.",
|
||||||
"comments_points_count_0": "{{count}} نقطة",
|
"comments_points_count_0": "{{count}} نقطة",
|
||||||
"comments_points_count_1": "نقطة واحدة",
|
"comments_points_count_1": "نقطة واحدة",
|
||||||
"comments_points_count_2": "نقطتان",
|
"comments_points_count_2": "نقطتان",
|
||||||
|
@ -556,5 +556,13 @@
|
||||||
"generic_channels_count_4": "{{count}} قنوات",
|
"generic_channels_count_4": "{{count}} قنوات",
|
||||||
"generic_channels_count_5": "{{count}} قناة",
|
"generic_channels_count_5": "{{count}} قناة",
|
||||||
"Import YouTube watch history (.json)": "استيراد سجل مشاهدة YouTube بصيغة (.json)",
|
"Import YouTube watch history (.json)": "استيراد سجل مشاهدة YouTube بصيغة (.json)",
|
||||||
"toggle_theme": "تبديل الموضوع"
|
"toggle_theme": "تبديل الموضوع",
|
||||||
|
"Add to playlist": "أضف إلى قائمة التشغيل",
|
||||||
|
"Add to playlist: ": "أضف إلى قائمة التشغيل: ",
|
||||||
|
"Answer": "الرد",
|
||||||
|
"Search for videos": "ابحث عن مقاطع الفيديو",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "تم تعطيل الخلاصة الشائعة من قبل المسؤول.",
|
||||||
|
"carousel_slide": "الشريحة {{current}} من {{total}}",
|
||||||
|
"carousel_skip": "تخطي الكاروسيل",
|
||||||
|
"carousel_go_to": "انتقل إلى الشريحة `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -487,5 +487,11 @@
|
||||||
"generic_views_count": "{{count}} гледане",
|
"generic_views_count": "{{count}} гледане",
|
||||||
"generic_views_count_plural": "{{count}} гледания",
|
"generic_views_count_plural": "{{count}} гледания",
|
||||||
"Next page": "Следваща страница",
|
"Next page": "Следваща страница",
|
||||||
"Import YouTube watch history (.json)": "Импортиране на историята на гледане от YouTube (.json)"
|
"Import YouTube watch history (.json)": "Импортиране на историята на гледане от YouTube (.json)",
|
||||||
|
"toggle_theme": "Смени темата",
|
||||||
|
"Add to playlist": "Добави към плейлист",
|
||||||
|
"Add to playlist: ": "Добави към плейлист: ",
|
||||||
|
"Answer": "Отговор",
|
||||||
|
"Search for videos": "Търсене на видеа",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Популярната страница е деактивирана от администратора."
|
||||||
}
|
}
|
||||||
|
|
|
@ -90,5 +90,7 @@
|
||||||
"preferences_quality_option_medium": "মধ্যম",
|
"preferences_quality_option_medium": "মধ্যম",
|
||||||
"preferences_quality_option_small": "ছোট",
|
"preferences_quality_option_small": "ছোট",
|
||||||
"preferences_quality_dash_option_1080p": "১০৮০পি",
|
"preferences_quality_dash_option_1080p": "১০৮০পি",
|
||||||
"preferences_quality_dash_option_720p": "৭২০পি"
|
"preferences_quality_dash_option_720p": "৭২০পি",
|
||||||
|
"Add to playlist": "প্লেলিস্টে যোগ করুন",
|
||||||
|
"Add to playlist: ": "প্লেলিস্টে যোগ করুন: "
|
||||||
}
|
}
|
||||||
|
|
|
@ -487,5 +487,7 @@
|
||||||
"generic_button_edit": "Edita",
|
"generic_button_edit": "Edita",
|
||||||
"generic_button_rss": "RSS",
|
"generic_button_rss": "RSS",
|
||||||
"generic_button_delete": "Suprimeix",
|
"generic_button_delete": "Suprimeix",
|
||||||
"Import YouTube watch history (.json)": "Importa l'historial de visualitzacions de YouTube (.json)"
|
"Import YouTube watch history (.json)": "Importa l'historial de visualitzacions de YouTube (.json)",
|
||||||
|
"Answer": "Resposta",
|
||||||
|
"toggle_theme": "Commuta el tema"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Import a export dat",
|
"Import and Export Data": "Import a export dat",
|
||||||
"Import": "Importovat",
|
"Import": "Importovat",
|
||||||
"Import Invidious data": "Importovat JSON údaje Invidious",
|
"Import Invidious data": "Importovat JSON údaje Invidious",
|
||||||
"Import YouTube subscriptions": "Importovat odběry z YouTube/OPML",
|
"Import YouTube subscriptions": "Importovat odběry z YouTube CSV nebo OPML",
|
||||||
"Import FreeTube subscriptions (.db)": "Importovat odběry z FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importovat odběry z FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importovat odběry z NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importovat odběry z NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importovat údeje z NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importovat údeje z NewPipe (.zip)",
|
||||||
|
@ -471,7 +471,7 @@
|
||||||
"search_filters_title": "Filtry",
|
"search_filters_title": "Filtry",
|
||||||
"search_filters_duration_option_medium": "Střední (4 - 20 minut)",
|
"search_filters_duration_option_medium": "Střední (4 - 20 minut)",
|
||||||
"search_filters_duration_option_long": "Dlouhá (> 20 minut)",
|
"search_filters_duration_option_long": "Dlouhá (> 20 minut)",
|
||||||
"search_message_use_another_instance": " Můžete také <a href=\"`x`\">hledat na jiné instanci</a>.",
|
"search_message_use_another_instance": "Můžete také <a href=\"`x`\">hledat na jiné instanci</a>.",
|
||||||
"search_filters_features_label": "Vlastnosti",
|
"search_filters_features_label": "Vlastnosti",
|
||||||
"search_filters_features_option_three_sixty": "360°",
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
|
@ -505,5 +505,13 @@
|
||||||
"generic_channels_count_1": "{{count}} kanály",
|
"generic_channels_count_1": "{{count}} kanály",
|
||||||
"generic_channels_count_2": "{{count}} kanálů",
|
"generic_channels_count_2": "{{count}} kanálů",
|
||||||
"Import YouTube watch history (.json)": "Importovat historii sledování z YouTube (.json)",
|
"Import YouTube watch history (.json)": "Importovat historii sledování z YouTube (.json)",
|
||||||
"toggle_theme": "Přepnout motiv"
|
"toggle_theme": "Přepnout motiv",
|
||||||
|
"Add to playlist": "Přidat do playlistu",
|
||||||
|
"Add to playlist: ": "Přidat do playlistu: ",
|
||||||
|
"Answer": "Odpověď",
|
||||||
|
"Search for videos": "Hledat videa",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Kategorie Populární byla zakázána administrátorem.",
|
||||||
|
"carousel_slide": "Snímek {{current}} z {{total}}",
|
||||||
|
"carousel_skip": "Přeskočit galerii",
|
||||||
|
"carousel_go_to": "Přejít na snímek `x`"
|
||||||
}
|
}
|
||||||
|
|
385
locales/cy.json
Normal file
385
locales/cy.json
Normal file
|
@ -0,0 +1,385 @@
|
||||||
|
{
|
||||||
|
"Time (h:mm:ss):": "Amser (h:mm:ss):",
|
||||||
|
"Password": "Cyfrinair",
|
||||||
|
"preferences_quality_dash_option_auto": "Awtomatig",
|
||||||
|
"preferences_quality_dash_option_best": "Gorau",
|
||||||
|
"preferences_quality_dash_option_worst": "Gwaethaf",
|
||||||
|
"preferences_quality_dash_option_360p": "360p",
|
||||||
|
"published": "dyddiad cyhoeddi",
|
||||||
|
"preferences_quality_dash_option_4320p": "4320p",
|
||||||
|
"preferences_quality_dash_option_480p": "480p",
|
||||||
|
"preferences_quality_dash_option_240p": "240p",
|
||||||
|
"preferences_quality_dash_option_144p": "144p",
|
||||||
|
"preferences_comments_label": "Ffynhonnell sylwadau: ",
|
||||||
|
"preferences_captions_label": "Isdeitlau rhagosodedig: ",
|
||||||
|
"youtube": "YouTube",
|
||||||
|
"reddit": "Reddit",
|
||||||
|
"Fallback captions: ": "Isdeitlau amgen: ",
|
||||||
|
"preferences_related_videos_label": "Dangos fideos perthnasol: ",
|
||||||
|
"dark": "tywyll",
|
||||||
|
"preferences_dark_mode_label": "Thema: ",
|
||||||
|
"light": "golau",
|
||||||
|
"preferences_sort_label": "Trefnu fideo yn ôl: ",
|
||||||
|
"Import/export data": "Mewnforio/allforio data",
|
||||||
|
"Delete account": "Dileu eich cyfrif",
|
||||||
|
"preferences_category_admin": "Hoffterau gweinyddu",
|
||||||
|
"playlist_button_add_items": "Ychwanegu fideos",
|
||||||
|
"Delete playlist": "Dileu'r rhestr chwarae",
|
||||||
|
"Create playlist": "Creu rhestr chwarae",
|
||||||
|
"Show less": "Dangos llai",
|
||||||
|
"Show more": "Dangos rhagor",
|
||||||
|
"Watch on YouTube": "Gwylio ar YouTube",
|
||||||
|
"search_message_no_results": "Dim canlyniadau.",
|
||||||
|
"search_message_change_filters_or_query": "Ceisiwch ehangu eich chwiliad ac/neu newid yr hidlyddion.",
|
||||||
|
"License: ": "Trwydded: ",
|
||||||
|
"Standard YouTube license": "Trwydded safonol YouTube",
|
||||||
|
"Family friendly? ": "Addas i bawb? ",
|
||||||
|
"Wilson score: ": "Sgôr Wilson: ",
|
||||||
|
"Show replies": "Dangos ymatebion",
|
||||||
|
"Music in this video": "Cerddoriaeth yn y fideo hwn",
|
||||||
|
"Artist: ": "Artist: ",
|
||||||
|
"Erroneous CAPTCHA": "CAPTCHA anghywir",
|
||||||
|
"This channel does not exist.": "Dyw'r sianel hon ddim yn bodoli.",
|
||||||
|
"Not a playlist.": "Ddim yn rhestr chwarae.",
|
||||||
|
"Could not fetch comments": "Wedi methu llwytho sylwadau",
|
||||||
|
"Playlist does not exist.": "Dyw'r rhestr chwarae ddim yn bodoli.",
|
||||||
|
"Erroneous challenge": "Her annilys",
|
||||||
|
"channel_tab_podcasts_label": "Podlediadau",
|
||||||
|
"channel_tab_playlists_label": "Rhestrau chwarae",
|
||||||
|
"channel_tab_streams_label": "Fideos byw",
|
||||||
|
"crash_page_read_the_faq": "darllen y <a href=\"`x`\">cwestiynau cyffredin</a>",
|
||||||
|
"crash_page_switch_instance": "ceisio <a href=\"`x`\">defnyddio gweinydd arall</a>",
|
||||||
|
"crash_page_refresh": "ceisio <a href=\"`x`\">ail-lwytho'r dudalen</a>",
|
||||||
|
"search_filters_features_option_four_k": "4K",
|
||||||
|
"search_filters_features_label": "Nodweddion",
|
||||||
|
"search_filters_duration_option_medium": "Canolig (4 - 20 munud)",
|
||||||
|
"search_filters_features_option_live": "Yn fyw",
|
||||||
|
"search_filters_duration_option_long": "Hir (> 20 munud)",
|
||||||
|
"search_filters_date_option_year": "Eleni",
|
||||||
|
"search_filters_type_label": "Math",
|
||||||
|
"search_filters_date_option_month": "Y mis hwn",
|
||||||
|
"generic_views_count_0": "{{count}} o wyliadau",
|
||||||
|
"generic_views_count_1": "{{count}} gwyliad",
|
||||||
|
"generic_views_count_2": "{{count}} wyliad",
|
||||||
|
"generic_views_count_3": "{{count}} o wyliadau",
|
||||||
|
"generic_views_count_4": "{{count}} o wyliadau",
|
||||||
|
"generic_views_count_5": "{{count}} o wyliadau",
|
||||||
|
"Answer": "Ateb",
|
||||||
|
"Add to playlist: ": "Ychwanegu at y rhestr chwarae: ",
|
||||||
|
"Add to playlist": "Ychwanegu at y rhestr chwarae",
|
||||||
|
"generic_button_cancel": "Diddymu",
|
||||||
|
"generic_button_rss": "RSS",
|
||||||
|
"LIVE": "YN FYW",
|
||||||
|
"Import YouTube watch history (.json)": "Mewnforio hanes gwylio YouTube (.json)",
|
||||||
|
"generic_videos_count_0": "{{count}} fideo",
|
||||||
|
"generic_videos_count_1": "{{count}} fideo",
|
||||||
|
"generic_videos_count_2": "{{count}} fideo",
|
||||||
|
"generic_videos_count_3": "{{count}} fideo",
|
||||||
|
"generic_videos_count_4": "{{count}} fideo",
|
||||||
|
"generic_videos_count_5": "{{count}} fideo",
|
||||||
|
"generic_subscribers_count_0": "{{count}} tanysgrifiwr",
|
||||||
|
"generic_subscribers_count_1": "{{count}} tanysgrifiwr",
|
||||||
|
"generic_subscribers_count_2": "{{count}} danysgrifiwr",
|
||||||
|
"generic_subscribers_count_3": "{{count}} thanysgrifiwr",
|
||||||
|
"generic_subscribers_count_4": "{{count}} o danysgrifwyr",
|
||||||
|
"generic_subscribers_count_5": "{{count}} o danysgrifwyr",
|
||||||
|
"Authorize token?": "Awdurdodi'r tocyn?",
|
||||||
|
"Authorize token for `x`?": "Awdurdodi'r tocyn ar gyfer `x`?",
|
||||||
|
"English": "Saesneg",
|
||||||
|
"English (United Kingdom)": "Saesneg (Y Deyrnas Unedig)",
|
||||||
|
"English (United States)": "Saesneg (Yr Unol Daleithiau)",
|
||||||
|
"Afrikaans": "Affricaneg",
|
||||||
|
"English (auto-generated)": "Saesneg (awtomatig)",
|
||||||
|
"Amharic": "Amhareg",
|
||||||
|
"Albanian": "Albaneg",
|
||||||
|
"Arabic": "Arabeg",
|
||||||
|
"crash_page_report_issue": "Os nad yw'r awgrymiadau uchod wedi helpu, <a href=\"`x`\">codwch 'issue' newydd ar Github </a> (yn Saesneg, gorau oll) a chynnwys y testun canlynol yn eich neges (peidiwch â chyfieithu'r testun hwn):",
|
||||||
|
"Search for videos": "Chwilio am fideos",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Mae'r ffrwd fideos poblogaidd wedi ei hanalluogi gan y gweinyddwr.",
|
||||||
|
"generic_channels_count_0": "{{count}} sianel",
|
||||||
|
"generic_channels_count_1": "{{count}} sianel",
|
||||||
|
"generic_channels_count_2": "{{count}} sianel",
|
||||||
|
"generic_channels_count_3": "{{count}} sianel",
|
||||||
|
"generic_channels_count_4": "{{count}} sianel",
|
||||||
|
"generic_channels_count_5": "{{count}} sianel",
|
||||||
|
"generic_button_delete": "Dileu",
|
||||||
|
"generic_button_edit": "Golygu",
|
||||||
|
"generic_button_save": "Cadw",
|
||||||
|
"Shared `x` ago": "Rhannwyd `x` yn ôl",
|
||||||
|
"Unsubscribe": "Dad-danysgrifio",
|
||||||
|
"Subscribe": "Tanysgrifio",
|
||||||
|
"View channel on YouTube": "Gweld y sianel ar YouTube",
|
||||||
|
"View playlist on YouTube": "Gweld y rhestr chwarae ar YouTube",
|
||||||
|
"newest": "diweddaraf",
|
||||||
|
"oldest": "hynaf",
|
||||||
|
"popular": "poblogaidd",
|
||||||
|
"Next page": "Tudalen nesaf",
|
||||||
|
"Previous page": "Tudalen flaenorol",
|
||||||
|
"Clear watch history?": "Clirio'ch hanes gwylio?",
|
||||||
|
"New password": "Cyfrinair newydd",
|
||||||
|
"Import and Export Data": "Mewnforio ac allforio data",
|
||||||
|
"Import": "Mewnforio",
|
||||||
|
"Import Invidious data": "Mewnforio data JSON Invidious",
|
||||||
|
"Import YouTube subscriptions": "Mewnforio tanysgrifiadau YouTube ar fformat CSV neu OPML",
|
||||||
|
"Import YouTube playlist (.csv)": "Mewnforio rhestr chwarae YouTube (.csv)",
|
||||||
|
"Export": "Allforio",
|
||||||
|
"Export data as JSON": "Allforio data Invidious ar fformat JSON",
|
||||||
|
"Delete account?": "Ydych chi'n siŵr yr hoffech chi ddileu eich cyfrif?",
|
||||||
|
"History": "Hanes",
|
||||||
|
"JavaScript license information": "Gwybodaeth am y drwydded JavaScript",
|
||||||
|
"generic_subscriptions_count_0": "{{count}} tanysgrifiad",
|
||||||
|
"generic_subscriptions_count_1": "{{count}} tanysgrifiad",
|
||||||
|
"generic_subscriptions_count_2": "{{count}} danysgrifiad",
|
||||||
|
"generic_subscriptions_count_3": "{{count}} thanysgrifiad",
|
||||||
|
"generic_subscriptions_count_4": "{{count}} o danysgrifiadau",
|
||||||
|
"generic_subscriptions_count_5": "{{count}} o danysgrifiadau",
|
||||||
|
"Yes": "Iawn",
|
||||||
|
"No": "Na",
|
||||||
|
"Import FreeTube subscriptions (.db)": "Mewnforio tanysgrifiadau FreeTube (.db)",
|
||||||
|
"Import NewPipe subscriptions (.json)": "Mewnforio tanysgrifiadau NewPipe (.json)",
|
||||||
|
"Import NewPipe data (.zip)": "Mewnforio data NewPipe (.zip)",
|
||||||
|
"An alternative front-end to YouTube": "Pen blaen amgen i YouTube",
|
||||||
|
"source": "ffynhonnell",
|
||||||
|
"Log in": "Mewngofnodi",
|
||||||
|
"Log in/register": "Mewngofnodi/Cofrestru",
|
||||||
|
"User ID": "Enw defnyddiwr",
|
||||||
|
"preferences_quality_option_dash": "DASH (ansawdd addasol)",
|
||||||
|
"Sign In": "Mewngofnodi",
|
||||||
|
"Register": "Cofrestru",
|
||||||
|
"E-mail": "Ebost",
|
||||||
|
"Preferences": "Hoffterau",
|
||||||
|
"preferences_category_player": "Hoffterau'r chwaraeydd",
|
||||||
|
"preferences_autoplay_label": "Chwarae'n awtomatig: ",
|
||||||
|
"preferences_local_label": "Llwytho fideos drwy ddirprwy weinydd: ",
|
||||||
|
"preferences_watch_history_label": "Galluogi hanes gwylio: ",
|
||||||
|
"preferences_speed_label": "Cyflymder rhagosodedig: ",
|
||||||
|
"preferences_quality_label": "Ansawdd fideos: ",
|
||||||
|
"preferences_quality_option_hd720": "HD720",
|
||||||
|
"preferences_quality_option_medium": "Canolig",
|
||||||
|
"preferences_quality_option_small": "Bach",
|
||||||
|
"preferences_quality_dash_option_2160p": "2160p",
|
||||||
|
"preferences_quality_dash_option_1440p": "1440p",
|
||||||
|
"preferences_quality_dash_option_1080p": "1080p",
|
||||||
|
"preferences_quality_dash_option_720p": "720p",
|
||||||
|
"invidious": "Invidious",
|
||||||
|
"Text CAPTCHA": "CAPTCHA testun",
|
||||||
|
"Image CAPTCHA": "CAPTCHA delwedd",
|
||||||
|
"preferences_continue_label": "Chwarae'r fideo nesaf fel rhagosodiad: ",
|
||||||
|
"preferences_continue_autoplay_label": "Chwarae'r fideo nesaf yn awtomatig: ",
|
||||||
|
"preferences_listen_label": "Sain yn unig: ",
|
||||||
|
"preferences_quality_dash_label": "Ansawdd fideos DASH a ffefrir: ",
|
||||||
|
"preferences_volume_label": "Uchder sain y chwaraeydd: ",
|
||||||
|
"preferences_category_visual": "Hoffterau'r wefan",
|
||||||
|
"preferences_region_label": "Gwlad y cynnwys: ",
|
||||||
|
"preferences_player_style_label": "Arddull y chwaraeydd: ",
|
||||||
|
"Dark mode: ": "Modd tywyll: ",
|
||||||
|
"preferences_thin_mode_label": "Modd tenau: ",
|
||||||
|
"preferences_category_misc": "Hoffterau amrywiol",
|
||||||
|
"preferences_category_subscription": "Hoffterau tanysgrifio",
|
||||||
|
"preferences_max_results_label": "Nifer o fideos a ddangosir yn eich ffrwd: ",
|
||||||
|
"alphabetically": "yr wyddor",
|
||||||
|
"alphabetically - reverse": "yr wyddor - am yn ôl",
|
||||||
|
"published - reverse": "dyddiad cyhoeddi - am yn ôl",
|
||||||
|
"channel name": "enw'r sianel",
|
||||||
|
"channel name - reverse": "enw'r sianel - am yn ôl",
|
||||||
|
"Only show latest video from channel: ": "Dangos fideo diweddaraf y sianeli rydych chi'n tanysgrifio iddynt: ",
|
||||||
|
"Only show latest unwatched video from channel: ": "Dangos fideo heb ei wylio diweddaraf y sianeli rydych chi'n tanysgrifio iddynt: ",
|
||||||
|
"Enable web notifications": "Galluogi hysbysiadau gwe",
|
||||||
|
"`x` uploaded a video": "uwchlwythodd `x` fideo",
|
||||||
|
"`x` is live": "mae `x` yn darlledu'n fyw",
|
||||||
|
"preferences_category_data": "Hoffterau data",
|
||||||
|
"Clear watch history": "Clirio'ch hanes gwylio",
|
||||||
|
"Change password": "Newid eich cyfrinair",
|
||||||
|
"Manage subscriptions": "Rheoli tanysgrifiadau",
|
||||||
|
"Manage tokens": "Rheoli tocynnau",
|
||||||
|
"Watch history": "Hanes gwylio",
|
||||||
|
"preferences_default_home_label": "Hafan ragosodedig: ",
|
||||||
|
"preferences_show_nick_label": "Dangos eich enw defnyddiwr ar frig y dudalen: ",
|
||||||
|
"preferences_annotations_label": "Dangos nodiadau fel rhagosodiad: ",
|
||||||
|
"preferences_unseen_only_label": "Dangos fideos heb eu gwylio yn unig: ",
|
||||||
|
"preferences_notifications_only_label": "Dangos hysbysiadau yn unig (os oes unrhyw rai): ",
|
||||||
|
"Token manager": "Rheolydd tocynnau",
|
||||||
|
"Token": "Tocyn",
|
||||||
|
"unsubscribe": "dad-danysgrifio",
|
||||||
|
"Subscriptions": "Tanysgrifiadau",
|
||||||
|
"Import/export": "Mewngofnodi/allgofnodi",
|
||||||
|
"search": "chwilio",
|
||||||
|
"Log out": "Allgofnodi",
|
||||||
|
"View privacy policy.": "Polisi preifatrwydd",
|
||||||
|
"Trending": "Pynciau llosg",
|
||||||
|
"Public": "Cyhoeddus",
|
||||||
|
"Private": "Preifat",
|
||||||
|
"Updated `x` ago": "Diweddarwyd `x` yn ôl",
|
||||||
|
"Delete playlist `x`?": "Ydych chi'n siŵr yr hoffech chi ddileu'r rhestr chwarae `x`?",
|
||||||
|
"Title": "Teitl",
|
||||||
|
"Playlist privacy": "Preifatrwydd y rhestr chwarae",
|
||||||
|
"search_message_use_another_instance": " Gallwch hefyd <a href=\"`x`\">chwilio ar weinydd arall</a>.",
|
||||||
|
"Popular enabled: ": "Tudalen fideos poblogaidd wedi'i galluogi: ",
|
||||||
|
"CAPTCHA enabled: ": "CAPTCHA wedi'i alluogi: ",
|
||||||
|
"Registration enabled: ": "Cofrestru wedi'i alluogi: ",
|
||||||
|
"Save preferences": "Cadw'r hoffterau",
|
||||||
|
"Subscription manager": "Rheolydd tanysgrifio",
|
||||||
|
"revoke": "tynnu",
|
||||||
|
"subscriptions_unseen_notifs_count_0": "{{count}} hysbysiad heb ei weld",
|
||||||
|
"subscriptions_unseen_notifs_count_1": "{{count}} hysbysiad heb ei weld",
|
||||||
|
"subscriptions_unseen_notifs_count_2": "{{count}} hysbysiad heb eu gweld",
|
||||||
|
"subscriptions_unseen_notifs_count_3": "{{count}} hysbysiad heb eu gweld",
|
||||||
|
"subscriptions_unseen_notifs_count_4": "{{count}} hysbysiad heb eu gweld",
|
||||||
|
"subscriptions_unseen_notifs_count_5": "{{count}} hysbysiad heb eu gweld",
|
||||||
|
"Released under the AGPLv3 on Github.": "Cyhoeddwyd dan drwydded AGPLv3 ar GitHub",
|
||||||
|
"Unlisted": "Heb ei restru",
|
||||||
|
"Switch Invidious Instance": "Newid gweinydd Invidious",
|
||||||
|
"Report statistics: ": "Galluogi ystadegau'r gweinydd: ",
|
||||||
|
"View all playlists": "Gweld pob rhestr chwarae",
|
||||||
|
"Editing playlist `x`": "Yn golygu'r rhestr chwarae `x`",
|
||||||
|
"Whitelisted regions: ": "Rhanbarthau a ganiateir: ",
|
||||||
|
"Blacklisted regions: ": "Rhanbarthau a rwystrir: ",
|
||||||
|
"Song: ": "Cân: ",
|
||||||
|
"Album: ": "Albwm: ",
|
||||||
|
"Shared `x`": "Rhannwyd `x`",
|
||||||
|
"View YouTube comments": "Dangos sylwadau YouTube",
|
||||||
|
"View more comments on Reddit": "Dangos rhagor o sylwadau ar Reddit",
|
||||||
|
"View Reddit comments": "Dangos sylwadau Reddit",
|
||||||
|
"Hide replies": "Cuddio ymatebion",
|
||||||
|
"Incorrect password": "Cyfrinair anghywir",
|
||||||
|
"Wrong answer": "Ateb anghywir",
|
||||||
|
"CAPTCHA is a required field": "Rhaid rhoi'r CAPTCHA",
|
||||||
|
"User ID is a required field": "Rhaid rhoi enw defnyddiwr",
|
||||||
|
"Password is a required field": "Rhaid rhoi cyfrinair",
|
||||||
|
"Wrong username or password": "Enw defnyddiwr neu gyfrinair anghywir",
|
||||||
|
"Password cannot be empty": "All y cyfrinair ddim bod yn wag",
|
||||||
|
"Password cannot be longer than 55 characters": "All y cyfrinair ddim bod yn hirach na 55 nod",
|
||||||
|
"Please log in": "Mewngofnodwch",
|
||||||
|
"channel:`x`": "sianel: `x`",
|
||||||
|
"Deleted or invalid channel": "Sianel wedi'i dileu neu'n annilys",
|
||||||
|
"Could not get channel info.": "Wedi methu llwytho gwybodaeth y sianel.",
|
||||||
|
"`x` ago": "`x` yn ôl",
|
||||||
|
"Load more": "Llwytho rhagor",
|
||||||
|
"Empty playlist": "Rhestr chwarae wag",
|
||||||
|
"Hide annotations": "Cuddio nodiadau",
|
||||||
|
"Show annotations": "Dangos nodiadau",
|
||||||
|
"Premieres in `x`": "Yn dechrau mewn `x`",
|
||||||
|
"Premieres `x`": "Yn dechrau `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.": "Helo! Mae'n ymddangos eich bod wedi diffodd JavaScript. Cliciwch yma i weld sylwadau, ond cofiwch y gall gymryd mwy o amser i'w llwytho.",
|
||||||
|
"View `x` comments": {
|
||||||
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Gweld `x` sylw",
|
||||||
|
"": "Gweld `x` sylw"
|
||||||
|
},
|
||||||
|
"Could not create mix.": "Wedi methu creu'r cymysgiad hwn.",
|
||||||
|
"Erroneous token": "Tocyn annilys",
|
||||||
|
"No such user": "Dyw'r defnyddiwr hwn ddim yn bodoli",
|
||||||
|
"Token is expired, please try again": "Mae'r tocyn hwn wedi dod i ben, ceisiwch eto",
|
||||||
|
"Bangla": "Bangleg",
|
||||||
|
"Basque": "Basgeg",
|
||||||
|
"Bulgarian": "Bwlgareg",
|
||||||
|
"Catalan": "Catalaneg",
|
||||||
|
"Chinese": "Tsieineeg",
|
||||||
|
"Chinese (China)": "Tsieineeg (Tsieina)",
|
||||||
|
"Chinese (Hong Kong)": "Tsieineeg (Hong Kong)",
|
||||||
|
"Chinese (Taiwan)": "Tsieineeg (Taiwan)",
|
||||||
|
"Danish": "Daneg",
|
||||||
|
"Dutch": "Iseldireg",
|
||||||
|
"Esperanto": "Esperanteg",
|
||||||
|
"Finnish": "Ffinneg",
|
||||||
|
"French": "Ffrangeg",
|
||||||
|
"German": "Almaeneg",
|
||||||
|
"Greek": "Groeg",
|
||||||
|
"Could not pull trending pages.": "Wedi methu llwytho tudalennau pynciau llosg.",
|
||||||
|
"Hidden field \"challenge\" is a required field": "Mae'r maes cudd \"her\" yn ofynnol",
|
||||||
|
"Hidden field \"token\" is a required field": "Mae'r maes cudd \"tocyn\" yn ofynnol",
|
||||||
|
"Hebrew": "Hebraeg",
|
||||||
|
"Hungarian": "Hwngareg",
|
||||||
|
"Irish": "Gwyddeleg",
|
||||||
|
"Italian": "Eidaleg",
|
||||||
|
"Welsh": "Cymraeg",
|
||||||
|
"generic_count_hours_0": "{{count}} awr",
|
||||||
|
"generic_count_hours_1": "{{count}} awr",
|
||||||
|
"generic_count_hours_2": "{{count}} awr",
|
||||||
|
"generic_count_hours_3": "{{count}} awr",
|
||||||
|
"generic_count_hours_4": "{{count}} awr",
|
||||||
|
"generic_count_hours_5": "{{count}} awr",
|
||||||
|
"generic_count_minutes_0": "{{count}} munud",
|
||||||
|
"generic_count_minutes_1": "{{count}} munud",
|
||||||
|
"generic_count_minutes_2": "{{count}} funud",
|
||||||
|
"generic_count_minutes_3": "{{count}} munud",
|
||||||
|
"generic_count_minutes_4": "{{count}} o funudau",
|
||||||
|
"generic_count_minutes_5": "{{count}} o funudau",
|
||||||
|
"generic_count_weeks_0": "{{count}} wythnos",
|
||||||
|
"generic_count_weeks_1": "{{count}} wythnos",
|
||||||
|
"generic_count_weeks_2": "{{count}} wythnos",
|
||||||
|
"generic_count_weeks_3": "{{count}} wythnos",
|
||||||
|
"generic_count_weeks_4": "{{count}} wythnos",
|
||||||
|
"generic_count_weeks_5": "{{count}} wythnos",
|
||||||
|
"generic_count_seconds_0": "{{count}} eiliad",
|
||||||
|
"generic_count_seconds_1": "{{count}} eiliad",
|
||||||
|
"generic_count_seconds_2": "{{count}} eiliad",
|
||||||
|
"generic_count_seconds_3": "{{count}} eiliad",
|
||||||
|
"generic_count_seconds_4": "{{count}} o eiliadau",
|
||||||
|
"generic_count_seconds_5": "{{count}} o eiliadau",
|
||||||
|
"Fallback comments: ": "Sylwadau amgen: ",
|
||||||
|
"Popular": "Poblogaidd",
|
||||||
|
"preferences_locale_label": "Iaith: ",
|
||||||
|
"About": "Ynghylch",
|
||||||
|
"Search": "Chwilio",
|
||||||
|
"search_filters_features_option_c_commons": "Comin Creu",
|
||||||
|
"search_filters_features_option_subtitles": "Isdeitlau (CC)",
|
||||||
|
"search_filters_features_option_hd": "HD",
|
||||||
|
"permalink": "dolen barhaol",
|
||||||
|
"search_filters_duration_option_short": "Byr (< 4 munud)",
|
||||||
|
"search_filters_duration_option_none": "Unrhyw hyd",
|
||||||
|
"search_filters_duration_label": "Hyd",
|
||||||
|
"search_filters_type_option_show": "Rhaglen",
|
||||||
|
"search_filters_type_option_movie": "Ffilm",
|
||||||
|
"search_filters_type_option_playlist": "Rhestr chwarae",
|
||||||
|
"search_filters_type_option_channel": "Sianel",
|
||||||
|
"search_filters_type_option_video": "Fideo",
|
||||||
|
"search_filters_type_option_all": "Unrhyw fath",
|
||||||
|
"search_filters_date_option_week": "Yr wythnos hon",
|
||||||
|
"search_filters_date_option_today": "Heddiw",
|
||||||
|
"search_filters_date_option_hour": "Yr awr ddiwethaf",
|
||||||
|
"search_filters_date_option_none": "Unrhyw ddyddiad",
|
||||||
|
"search_filters_date_label": "Dyddiad uwchlwytho",
|
||||||
|
"search_filters_title": "Hidlyddion",
|
||||||
|
"Playlists": "Rhestrau chwarae",
|
||||||
|
"Video mode": "Modd fideo",
|
||||||
|
"Audio mode": "Modd sain",
|
||||||
|
"Channel Sponsor": "Noddwr y sianel",
|
||||||
|
"(edited)": "(golygwyd)",
|
||||||
|
"Download": "Islwytho",
|
||||||
|
"Movies": "Ffilmiau",
|
||||||
|
"News": "Newyddion",
|
||||||
|
"Gaming": "Gemau",
|
||||||
|
"Music": "Cerddoriaeth",
|
||||||
|
"Download is disabled": "Mae islwytho wedi'i analluogi",
|
||||||
|
"Download as: ": "Islwytho fel: ",
|
||||||
|
"View as playlist": "Gweld fel rhestr chwarae",
|
||||||
|
"Default": "Rhagosodiad",
|
||||||
|
"YouTube comment permalink": "Dolen barhaol i'r sylw ar YouTube",
|
||||||
|
"crash_page_before_reporting": "Cyn adrodd nam, sicrhewch eich bod wedi:",
|
||||||
|
"crash_page_search_issue": "<a href=\"`x`\">chwilio am y nam ar GitHub</a>",
|
||||||
|
"videoinfo_watch_on_youTube": "Gwylio ar YouTube",
|
||||||
|
"videoinfo_started_streaming_x_ago": "Yn ffrydio'n fyw ers `x` o funudau",
|
||||||
|
"videoinfo_invidious_embed_link": "Dolen mewnblannu",
|
||||||
|
"footer_documentation": "Dogfennaeth",
|
||||||
|
"footer_donate_page": "Rhoddi",
|
||||||
|
"Current version: ": "Fersiwn gyfredol: ",
|
||||||
|
"search_filters_apply_button": "Rhoi'r hidlyddion ar waith",
|
||||||
|
"search_filters_sort_option_date": "Dyddiad uwchlwytho",
|
||||||
|
"search_filters_sort_option_relevance": "Perthnasedd",
|
||||||
|
"search_filters_sort_label": "Trefnu yn ôl",
|
||||||
|
"search_filters_features_option_location": "Lleoliad",
|
||||||
|
"search_filters_features_option_hdr": "HDR",
|
||||||
|
"search_filters_features_option_three_d": "3D",
|
||||||
|
"search_filters_features_option_vr180": "VR180",
|
||||||
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
|
"videoinfo_youTube_embed_link": "Mewnblannu",
|
||||||
|
"download_subtitles": "Isdeitlau - `x` (.vtt)",
|
||||||
|
"user_created_playlists": "`x` rhestr chwarae wedi'u creu",
|
||||||
|
"user_saved_playlists": "`x` rhestr chwarae wedi'u cadw",
|
||||||
|
"Video unavailable": "Fideo ddim ar gael",
|
||||||
|
"crash_page_you_found_a_bug": "Mae'n debyg eich bod wedi dod o hyd i nam yn Invidious!",
|
||||||
|
"channel_tab_channels_label": "Sianeli",
|
||||||
|
"channel_tab_community_label": "Cymuned",
|
||||||
|
"channel_tab_shorts_label": "Fideos byrion",
|
||||||
|
"channel_tab_videos_label": "Fideos"
|
||||||
|
}
|
|
@ -165,12 +165,12 @@
|
||||||
"Password cannot be empty": "Adgangskoden må ikke være tom",
|
"Password cannot be empty": "Adgangskoden må ikke være tom",
|
||||||
"Password cannot be longer than 55 characters": "Adgangskoden må ikke være længere end 55 tegn",
|
"Password cannot be longer than 55 characters": "Adgangskoden må ikke være længere end 55 tegn",
|
||||||
"Please log in": "Venligst log ind",
|
"Please log in": "Venligst log ind",
|
||||||
"channel:`x`": "kanal: 'x'",
|
"channel:`x`": "kanal: `x`",
|
||||||
"Deleted or invalid channel": "Slettet eller invalid kanal",
|
"Deleted or invalid channel": "Slettet eller invalid kanal",
|
||||||
"This channel does not exist.": "Denne kanal eksisterer ikke.",
|
"This channel does not exist.": "Denne kanal eksisterer ikke.",
|
||||||
"Could not get channel info.": "Kunne ikke hente kanal info.",
|
"Could not get channel info.": "Kunne ikke hente kanal info.",
|
||||||
"Could not fetch comments": "Kunne ikke hente kommentarer",
|
"Could not fetch comments": "Kunne ikke hente kommentarer",
|
||||||
"`x` ago": "'x' siden",
|
"`x` ago": "`x` siden",
|
||||||
"Load more": "Hent flere",
|
"Load more": "Hent flere",
|
||||||
"Could not create mix.": "Kunne ikke skabe blanding.",
|
"Could not create mix.": "Kunne ikke skabe blanding.",
|
||||||
"Empty playlist": "Tom playliste",
|
"Empty playlist": "Tom playliste",
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Daten importieren und exportieren",
|
"Import and Export Data": "Daten importieren und exportieren",
|
||||||
"Import": "Importieren",
|
"Import": "Importieren",
|
||||||
"Import Invidious data": "Invidious-JSON-Daten importieren",
|
"Import Invidious data": "Invidious-JSON-Daten importieren",
|
||||||
"Import YouTube subscriptions": "YouTube-/OPML-Abonnements importieren",
|
"Import YouTube subscriptions": "YouTube-CSV/OPML-Abonnements importieren",
|
||||||
"Import FreeTube subscriptions (.db)": "FreeTube Abonnements importieren (.db)",
|
"Import FreeTube subscriptions (.db)": "FreeTube Abonnements importieren (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "NewPipe Abonnements importieren (.json)",
|
"Import NewPipe subscriptions (.json)": "NewPipe Abonnements importieren (.json)",
|
||||||
"Import NewPipe data (.zip)": "NewPipe Daten importieren (.zip)",
|
"Import NewPipe data (.zip)": "NewPipe Daten importieren (.zip)",
|
||||||
|
@ -47,6 +47,7 @@
|
||||||
"Preferences": "Einstellungen",
|
"Preferences": "Einstellungen",
|
||||||
"preferences_category_player": "Wiedergabeeinstellungen",
|
"preferences_category_player": "Wiedergabeeinstellungen",
|
||||||
"preferences_video_loop_label": "Immer wiederholen: ",
|
"preferences_video_loop_label": "Immer wiederholen: ",
|
||||||
|
"preferences_preload_label": "Videodaten vorladen: ",
|
||||||
"preferences_autoplay_label": "Automatisch abspielen: ",
|
"preferences_autoplay_label": "Automatisch abspielen: ",
|
||||||
"preferences_continue_label": "Immer automatisch nächstes Video abspielen: ",
|
"preferences_continue_label": "Immer automatisch nächstes Video abspielen: ",
|
||||||
"preferences_continue_autoplay_label": "Nächstes Video automatisch abspielen: ",
|
"preferences_continue_autoplay_label": "Nächstes Video automatisch abspielen: ",
|
||||||
|
@ -322,7 +323,7 @@
|
||||||
"channel_tab_community_label": "Gemeinschaft",
|
"channel_tab_community_label": "Gemeinschaft",
|
||||||
"search_filters_sort_option_relevance": "Relevanz",
|
"search_filters_sort_option_relevance": "Relevanz",
|
||||||
"search_filters_sort_option_rating": "Bewertung",
|
"search_filters_sort_option_rating": "Bewertung",
|
||||||
"search_filters_sort_option_date": "Datum",
|
"search_filters_sort_option_date": "Hochladedatum",
|
||||||
"search_filters_sort_option_views": "Aufrufe",
|
"search_filters_sort_option_views": "Aufrufe",
|
||||||
"search_filters_type_label": "Inhaltstyp",
|
"search_filters_type_label": "Inhaltstyp",
|
||||||
"search_filters_duration_label": "Dauer",
|
"search_filters_duration_label": "Dauer",
|
||||||
|
@ -454,7 +455,7 @@
|
||||||
"Portuguese (auto-generated)": "Portugiesisch (automatisch generiert)",
|
"Portuguese (auto-generated)": "Portugiesisch (automatisch generiert)",
|
||||||
"search_filters_title": "Filtern",
|
"search_filters_title": "Filtern",
|
||||||
"search_message_change_filters_or_query": "Versuchen Sie, Ihre Suchanfrage zu erweitern und/oder die Filter zu ändern.",
|
"search_message_change_filters_or_query": "Versuchen Sie, Ihre Suchanfrage zu erweitern und/oder die Filter zu ändern.",
|
||||||
"search_message_use_another_instance": " Sie können auch <a href=\"`x`\">auf einer anderen Instanz suchen</a>.",
|
"search_message_use_another_instance": "Sie können auch <a href=\"`x`\">auf einer anderen Instanz suchen</a>.",
|
||||||
"Popular enabled: ": "„Beliebt“-Seite aktiviert: ",
|
"Popular enabled: ": "„Beliebt“-Seite aktiviert: ",
|
||||||
"search_message_no_results": "Keine Ergebnisse gefunden.",
|
"search_message_no_results": "Keine Ergebnisse gefunden.",
|
||||||
"search_filters_duration_option_medium": "Mittel (4 - 20 Minuten)",
|
"search_filters_duration_option_medium": "Mittel (4 - 20 Minuten)",
|
||||||
|
@ -487,5 +488,14 @@
|
||||||
"channel_tab_releases_label": "Veröffentlichungen",
|
"channel_tab_releases_label": "Veröffentlichungen",
|
||||||
"generic_channels_count": "{{count}} Kanal",
|
"generic_channels_count": "{{count}} Kanal",
|
||||||
"generic_channels_count_plural": "{{count}} Kanäle",
|
"generic_channels_count_plural": "{{count}} Kanäle",
|
||||||
"Import YouTube watch history (.json)": "YouTube Wiedergabeverlauf importieren (.json)"
|
"Import YouTube watch history (.json)": "YouTube Wiedergabeverlauf importieren (.json)",
|
||||||
|
"Answer": "Antwort",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Der Angesagt-Feed wurde vom Administrator deaktiviert.",
|
||||||
|
"Add to playlist": "Einer Wiedergabeliste hinzufügen",
|
||||||
|
"Search for videos": "Nach Videos suchen",
|
||||||
|
"toggle_theme": "Thema wechseln",
|
||||||
|
"Add to playlist: ": "Einer Wiedergabeliste hinzufügen: ",
|
||||||
|
"carousel_go_to": "Zu Folie `x` gehen",
|
||||||
|
"carousel_slide": "Folie {{current}} von {{total}}",
|
||||||
|
"carousel_skip": "Karussell überspringen"
|
||||||
}
|
}
|
||||||
|
|
|
@ -486,5 +486,13 @@
|
||||||
"Switch Invidious Instance": "Αλλαγή Instance Invidious",
|
"Switch Invidious Instance": "Αλλαγή Instance Invidious",
|
||||||
"Standard YouTube license": "Τυπική άδεια YouTube",
|
"Standard YouTube license": "Τυπική άδεια YouTube",
|
||||||
"search_filters_duration_option_medium": "Μεσαία (4 - 20 λεπτά)",
|
"search_filters_duration_option_medium": "Μεσαία (4 - 20 λεπτά)",
|
||||||
"search_filters_date_label": "Ημερομηνία αναφόρτωσης"
|
"search_filters_date_label": "Ημερομηνία αναφόρτωσης",
|
||||||
|
"Search for videos": "Αναζήτηση βίντεο",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Η δημοφιλής ροή έχει απενεργοποιηθεί από τον διαχειριστή.",
|
||||||
|
"Answer": "Απάντηση",
|
||||||
|
"Add to playlist": "Λίιστα αναπαραγωγής",
|
||||||
|
"Add to playlist: ": "Λίστα αναπαραγωγής: ",
|
||||||
|
"carousel_slide": "Εικόνα {{current}}απο {{total}}",
|
||||||
|
"carousel_go_to": "Πήγαινε στην εικόνα`x`",
|
||||||
|
"toggle_theme": "Αλλαγή θέματος"
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,7 +43,7 @@
|
||||||
"Import and Export Data": "Import and Export Data",
|
"Import and Export Data": "Import and Export Data",
|
||||||
"Import": "Import",
|
"Import": "Import",
|
||||||
"Import Invidious data": "Import Invidious JSON data",
|
"Import Invidious data": "Import Invidious JSON data",
|
||||||
"Import YouTube subscriptions": "Import YouTube/OPML subscriptions",
|
"Import YouTube subscriptions": "Import YouTube CSV or OPML subscriptions",
|
||||||
"Import YouTube playlist (.csv)": "Import YouTube playlist (.csv)",
|
"Import YouTube playlist (.csv)": "Import YouTube playlist (.csv)",
|
||||||
"Import YouTube watch history (.json)": "Import YouTube watch history (.json)",
|
"Import YouTube watch history (.json)": "Import YouTube watch history (.json)",
|
||||||
"Import FreeTube subscriptions (.db)": "Import FreeTube subscriptions (.db)",
|
"Import FreeTube subscriptions (.db)": "Import FreeTube subscriptions (.db)",
|
||||||
|
@ -71,6 +71,7 @@
|
||||||
"Preferences": "Preferences",
|
"Preferences": "Preferences",
|
||||||
"preferences_category_player": "Player preferences",
|
"preferences_category_player": "Player preferences",
|
||||||
"preferences_video_loop_label": "Always loop: ",
|
"preferences_video_loop_label": "Always loop: ",
|
||||||
|
"preferences_preload_label": "Preload video data: ",
|
||||||
"preferences_autoplay_label": "Autoplay: ",
|
"preferences_autoplay_label": "Autoplay: ",
|
||||||
"preferences_continue_label": "Play next by default: ",
|
"preferences_continue_label": "Play next by default: ",
|
||||||
"preferences_continue_autoplay_label": "Autoplay next video: ",
|
"preferences_continue_autoplay_label": "Autoplay next video: ",
|
||||||
|
@ -191,7 +192,7 @@
|
||||||
"Switch Invidious Instance": "Switch Invidious Instance",
|
"Switch Invidious Instance": "Switch Invidious Instance",
|
||||||
"search_message_no_results": "No results found.",
|
"search_message_no_results": "No results found.",
|
||||||
"search_message_change_filters_or_query": "Try widening your search query and/or changing the filters.",
|
"search_message_change_filters_or_query": "Try widening your search query and/or changing the filters.",
|
||||||
"search_message_use_another_instance": " You can also <a href=\"`x`\">search on another instance</a>.",
|
"search_message_use_another_instance": "You can also <a href=\"`x`\">search on another instance</a>.",
|
||||||
"Hide annotations": "Hide annotations",
|
"Hide annotations": "Hide annotations",
|
||||||
"Show annotations": "Show annotations",
|
"Show annotations": "Show annotations",
|
||||||
"Genre: ": "Genre: ",
|
"Genre: ": "Genre: ",
|
||||||
|
@ -423,7 +424,7 @@
|
||||||
"search_filters_title": "Filters",
|
"search_filters_title": "Filters",
|
||||||
"search_filters_date_label": "Upload date",
|
"search_filters_date_label": "Upload date",
|
||||||
"search_filters_date_option_none": "Any date",
|
"search_filters_date_option_none": "Any date",
|
||||||
"search_filters_date_option_hour": "Last Hour",
|
"search_filters_date_option_hour": "Last hour",
|
||||||
"search_filters_date_option_today": "Today",
|
"search_filters_date_option_today": "Today",
|
||||||
"search_filters_date_option_week": "This week",
|
"search_filters_date_option_week": "This week",
|
||||||
"search_filters_date_option_month": "This month",
|
"search_filters_date_option_month": "This month",
|
||||||
|
@ -455,22 +456,41 @@
|
||||||
"search_filters_sort_label": "Sort By",
|
"search_filters_sort_label": "Sort By",
|
||||||
"search_filters_sort_option_relevance": "Relevance",
|
"search_filters_sort_option_relevance": "Relevance",
|
||||||
"search_filters_sort_option_rating": "Rating",
|
"search_filters_sort_option_rating": "Rating",
|
||||||
"search_filters_sort_option_date": "Upload Date",
|
"search_filters_sort_option_date": "Upload date",
|
||||||
"search_filters_sort_option_views": "View count",
|
"search_filters_sort_option_views": "View count",
|
||||||
"search_filters_apply_button": "Apply selected filters",
|
"search_filters_apply_button": "Apply selected filters",
|
||||||
"Current version: ": "Current version: ",
|
"Current version: ": "Current version: ",
|
||||||
|
"footer_current_version_modified": "Current version (modified): ",
|
||||||
"next_steps_error_message": "After which you should try to: ",
|
"next_steps_error_message": "After which you should try to: ",
|
||||||
"next_steps_error_message_refresh": "Refresh",
|
"next_steps_error_message_refresh": "Refresh",
|
||||||
"next_steps_error_message_go_to_youtube": "Go to YouTube",
|
"next_steps_error_message_go_to_youtube": "Go to YouTube",
|
||||||
"footer_donate_page": "Donate",
|
"default_invidious_footer_text": "A free and open source frontend for Youtube that that respects your privacy! Now you can watch videos (ad-free), subscribe to channels, create playlist and much more all without the prying eyes of Google!",
|
||||||
"footer_documentation": "Documentation",
|
"footer_navigation_section_header": "Navigation",
|
||||||
"footer_source_code": "Source code",
|
"footer_home_link": "Home",
|
||||||
"footer_original_source_code": "Original source code",
|
"footer_project_information_section_header": "Invidious",
|
||||||
"footer_modfied_source_code": "Modified source code",
|
"footer_project_homepage_link": "Project Homepage",
|
||||||
|
"footer_source_code_link": "Source Code",
|
||||||
|
"footer_issue_tracker_link": "Issue tracker",
|
||||||
|
"footer_public_instances_link": "Public instances",
|
||||||
|
"footer_donate_link": "Donate",
|
||||||
|
"footer_matrix_link": "Matrix",
|
||||||
|
"footer_support_section_header": "Support",
|
||||||
|
"footer_contact_link": "Contact Instance Maintainer",
|
||||||
|
"footer_report_bug_link": "Report a bug",
|
||||||
|
"footer_faq_link": "FAQ",
|
||||||
|
"footer_instance_section_header": "Instance",
|
||||||
|
"footer_instance_section_header_modified_source": "Instance (Modified)",
|
||||||
|
"footer_instance_section_modified_source_code": "Instance Source Code",
|
||||||
|
"footer_instance_section_tos": "Terms of Service",
|
||||||
|
"footer_instance_section_privacy_policy": "Privacy Policy",
|
||||||
|
"footer_instance_section_donate": "Donate (Instance)",
|
||||||
|
"footer_licences_link": "Licences",
|
||||||
|
"footer_privacy_policy_link": "Privacy",
|
||||||
"adminprefs_modified_source_code_url_label": "URL to modified source code repository",
|
"adminprefs_modified_source_code_url_label": "URL to modified source code repository",
|
||||||
"none": "none",
|
"none": "none",
|
||||||
"videoinfo_started_streaming_x_ago": "Started streaming `x` ago",
|
"videoinfo_started_streaming_x_ago": "Started streaming `x` ago",
|
||||||
"videoinfo_watch_on_youTube": "Watch on YouTube",
|
"videoinfo_watch_on_youTube": "Watch on YouTube",
|
||||||
|
"videoinfo_watch_on_materialious": "Watch on Materialious",
|
||||||
"videoinfo_youTube_embed_link": "Embed",
|
"videoinfo_youTube_embed_link": "Embed",
|
||||||
"videoinfo_invidious_embed_link": "Embed Link",
|
"videoinfo_invidious_embed_link": "Embed Link",
|
||||||
"download_subtitles": "Subtitles - `x` (.vtt)",
|
"download_subtitles": "Subtitles - `x` (.vtt)",
|
||||||
|
@ -497,5 +517,7 @@
|
||||||
"toggle_theme": "Toggle Theme",
|
"toggle_theme": "Toggle Theme",
|
||||||
"carousel_slide": "Slide {{current}} of {{total}}",
|
"carousel_slide": "Slide {{current}} of {{total}}",
|
||||||
"carousel_skip": "Skip the Carousel",
|
"carousel_skip": "Skip the Carousel",
|
||||||
"carousel_go_to": "Go to slide `x`"
|
"carousel_go_to": "Go to slide `x`",
|
||||||
|
"footer_contact_url": "Contact the Administrator"
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Importación y exportación de datos",
|
"Import and Export Data": "Importación y exportación de datos",
|
||||||
"Import": "Importar",
|
"Import": "Importar",
|
||||||
"Import Invidious data": "Importar datos JSON de Invidious",
|
"Import Invidious data": "Importar datos JSON de Invidious",
|
||||||
"Import YouTube subscriptions": "Importar suscripciones de YouTube/OPML",
|
"Import YouTube subscriptions": "Importar suscripciones CSV u OPML de YouTube",
|
||||||
"Import FreeTube subscriptions (.db)": "Importar suscripciones de FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importar suscripciones de FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importar suscripciones de NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importar suscripciones de NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importar datos de NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importar datos de NewPipe (.zip)",
|
||||||
|
@ -134,7 +134,7 @@
|
||||||
"Create playlist": "Crear lista de reproducción",
|
"Create playlist": "Crear lista de reproducción",
|
||||||
"Title": "Título",
|
"Title": "Título",
|
||||||
"Playlist privacy": "Privacidad de la lista de reproducción",
|
"Playlist privacy": "Privacidad de la lista de reproducción",
|
||||||
"Editing playlist `x`": "Editando la lista de reproducción 'x'",
|
"Editing playlist `x`": "Editando la lista de reproducción `x`",
|
||||||
"Show more": "Mostrar más",
|
"Show more": "Mostrar más",
|
||||||
"Show less": "Mostrar menos",
|
"Show less": "Mostrar menos",
|
||||||
"Watch on YouTube": "Ver en YouTube",
|
"Watch on YouTube": "Ver en YouTube",
|
||||||
|
@ -390,6 +390,7 @@
|
||||||
"search_filters_features_option_purchased": "Comprado",
|
"search_filters_features_option_purchased": "Comprado",
|
||||||
"search_filters_features_option_three_sixty": "360°",
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
"videoinfo_watch_on_youTube": "Ver en YouTube",
|
"videoinfo_watch_on_youTube": "Ver en YouTube",
|
||||||
|
"videoinfo_watch_on_materialious": "Ver en Materialious",
|
||||||
"preferences_save_player_pos_label": "Guardar posición de reproducción: ",
|
"preferences_save_player_pos_label": "Guardar posición de reproducción: ",
|
||||||
"generic_views_count_0": "{{count}} visualización",
|
"generic_views_count_0": "{{count}} visualización",
|
||||||
"generic_views_count_1": "{{count}} visualizaciones",
|
"generic_views_count_1": "{{count}} visualizaciones",
|
||||||
|
@ -479,7 +480,7 @@
|
||||||
"tokens_count_0": "{{count}} token",
|
"tokens_count_0": "{{count}} token",
|
||||||
"tokens_count_1": "{{count}} tokens",
|
"tokens_count_1": "{{count}} tokens",
|
||||||
"tokens_count_2": "{{count}} tokens",
|
"tokens_count_2": "{{count}} tokens",
|
||||||
"search_message_use_another_instance": " También puede <a href=\"`x`\">buscar en otra instancia</a>.",
|
"search_message_use_another_instance": "También puedes <a href=\"`x`\">buscar en otra instancia</a>.",
|
||||||
"Popular enabled: ": "¿Habilitar la sección popular? ",
|
"Popular enabled: ": "¿Habilitar la sección popular? ",
|
||||||
"error_video_not_in_playlist": "El video que solicitaste no existe en esta lista de reproducción. <a href=\"`x`\">Haz clic aquí para acceder a la página de inicio de la lista de reproducción.</a>",
|
"error_video_not_in_playlist": "El video que solicitaste no existe en esta lista de reproducción. <a href=\"`x`\">Haz clic aquí para acceder a la página de inicio de la lista de reproducción.</a>",
|
||||||
"channel_tab_streams_label": "Directos",
|
"channel_tab_streams_label": "Directos",
|
||||||
|
@ -506,5 +507,14 @@
|
||||||
"generic_channels_count_1": "{{count}} canales",
|
"generic_channels_count_1": "{{count}} canales",
|
||||||
"generic_channels_count_2": "{{count}} canales",
|
"generic_channels_count_2": "{{count}} canales",
|
||||||
"Import YouTube watch history (.json)": "Importar el historial de las visualizaciones de YouTube (.json)",
|
"Import YouTube watch history (.json)": "Importar el historial de las visualizaciones de YouTube (.json)",
|
||||||
"toggle_theme": "Alternar tema"
|
"toggle_theme": "Alternar tema",
|
||||||
|
"Add to playlist: ": "Añadir a la lista de reproducción: ",
|
||||||
|
"Add to playlist": "Añadir a la lista de reproducción",
|
||||||
|
"Answer": "Respuesta",
|
||||||
|
"Search for videos": "Buscar por vídeos",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "El feed Popular ha sido desactivado por el administrador.",
|
||||||
|
"carousel_slide": "Diapositiva {{current}} de {{total}}",
|
||||||
|
"carousel_skip": "Saltar el carrusel",
|
||||||
|
"carousel_go_to": "Ir a la diapositiva `x`",
|
||||||
|
"footer_contact_url": "Contactar al Administrador"
|
||||||
}
|
}
|
||||||
|
|
|
@ -161,13 +161,13 @@
|
||||||
"Source available here.": "Iturburua hemen eskura.",
|
"Source available here.": "Iturburua hemen eskura.",
|
||||||
"View JavaScript license information.": "JavaScriptaren lizentzi adierazpena ikusi.",
|
"View JavaScript license information.": "JavaScriptaren lizentzi adierazpena ikusi.",
|
||||||
"Blacklisted regions: ": "zerrenda beltzaren zonaldeak: ",
|
"Blacklisted regions: ": "zerrenda beltzaren zonaldeak: ",
|
||||||
"Premieres `x`": "'x' estrenaldiak",
|
"Premieres `x`": "`x` estrenaldiak",
|
||||||
"Wrong answer": "Erantzun ez zuzena",
|
"Wrong answer": "Erantzun ez zuzena",
|
||||||
"Password is a required field": "Pasahitza beharrezkoa da",
|
"Password is a required field": "Pasahitza beharrezkoa da",
|
||||||
"Wrong username or password": "Pasahitza edo ezizena gaizki",
|
"Wrong username or password": "Pasahitza edo ezizena gaizki",
|
||||||
"Password cannot be longer than 55 characters": "Pasahitza 55 karaktere baino luzeagoa ezin da izan",
|
"Password cannot be longer than 55 characters": "Pasahitza 55 karaktere baino luzeagoa ezin da izan",
|
||||||
"This channel does not exist.": "Kanal hau ez dago.",
|
"This channel does not exist.": "Kanal hau ez dago.",
|
||||||
"`x` ago": "duela 'x'",
|
"`x` ago": "duela `x`",
|
||||||
"Czech": "Txekiera",
|
"Czech": "Txekiera",
|
||||||
"preferences_region_label": "Herrialdeko edukiera: ",
|
"preferences_region_label": "Herrialdeko edukiera: ",
|
||||||
"preferences_sort_label": "Bideoak ordenatu: ",
|
"preferences_sort_label": "Bideoak ordenatu: ",
|
||||||
|
@ -207,24 +207,24 @@
|
||||||
"Public": "Orokorra",
|
"Public": "Orokorra",
|
||||||
"Unlisted": "Ez zerrendatua",
|
"Unlisted": "Ez zerrendatua",
|
||||||
"Subscription manager": "Harpidetzen kudeatzailea",
|
"Subscription manager": "Harpidetzen kudeatzailea",
|
||||||
"Updated `x` ago": "Duela 'x' eguneratua",
|
"Updated `x` ago": "Duela `x` eguneratua",
|
||||||
"Hide replies": "Erantzunak izkutatu",
|
"Hide replies": "Erantzunak izkutatu",
|
||||||
"preferences_thin_mode_label": "Urri eran: ",
|
"preferences_thin_mode_label": "Urri eran: ",
|
||||||
"Show replies": "Erantzunak erakutsi",
|
"Show replies": "Erantzunak erakutsi",
|
||||||
"Watch on YouTube": "YouTuben ikusi",
|
"Watch on YouTube": "YouTuben ikusi",
|
||||||
"Premieres in `x`": "'x'eko estrenaldiak",
|
"Premieres in `x`": "`x`eko estrenaldiak",
|
||||||
"Delete playlist `x`?": "'x' zerrenda ezabatu nahi?",
|
"Delete playlist `x`?": "`x` zerrenda ezabatu nahi?",
|
||||||
"Token is expired, please try again": "Token kadukatua, saiatu berriro",
|
"Token is expired, please try again": "Token kadukatua, saiatu berriro",
|
||||||
"CAPTCHA enabled: ": "CAPTCHA gaitu: ",
|
"CAPTCHA enabled: ": "CAPTCHA gaitu: ",
|
||||||
"Released under the AGPLv3 on Github.": "GitHubeko AGPLv3pean argitaratuta.",
|
"Released under the AGPLv3 on Github.": "GitHubeko AGPLv3pean argitaratuta.",
|
||||||
"channel:`x`": "Kanal: 'x'",
|
"channel:`x`": "Kanal: `x`",
|
||||||
"Georgian": "Georgiera",
|
"Georgian": "Georgiera",
|
||||||
"Incorrect password": "Pasahitza gaizki",
|
"Incorrect password": "Pasahitza gaizki",
|
||||||
"Playlist does not exist.": "Zerrenda ez da existitzen.",
|
"Playlist does not exist.": "Zerrenda ez da existitzen.",
|
||||||
"preferences_category_misc": "Askotariko lehentasunak",
|
"preferences_category_misc": "Askotariko lehentasunak",
|
||||||
"View `x` comments": {
|
"View `x` comments": {
|
||||||
"([^.,0-9]|^)1([^.,0-9]|$)": "'x' iruzkina ikusi",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "`x` iruzkina ikusi",
|
||||||
"": "'x' iruzkinak ikusi"
|
"": "`x` iruzkinak ikusi"
|
||||||
},
|
},
|
||||||
"Report statistics: ": "Estatistikak adierazi: ",
|
"Report statistics: ": "Estatistikak adierazi: ",
|
||||||
"preferences_max_results_label": "Jotzeko bideo zerrendaren luzera: ",
|
"preferences_max_results_label": "Jotzeko bideo zerrendaren luzera: ",
|
||||||
|
@ -237,7 +237,7 @@
|
||||||
"Hidden field \"challenge\" is a required field": "\"challenge\" eremu ezkutua beharrezkoa da",
|
"Hidden field \"challenge\" is a required field": "\"challenge\" eremu ezkutua beharrezkoa da",
|
||||||
"German": "Alemaniarra",
|
"German": "Alemaniarra",
|
||||||
"View YouTube comments": "YouTubeko iruzkinak ikusi",
|
"View YouTube comments": "YouTubeko iruzkinak ikusi",
|
||||||
"`x` is live": "'x' bizirik darrai",
|
"`x` is live": "`x` bizirik darrai",
|
||||||
"Password cannot be empty": "Pasahitza ezin da hutsik utzi",
|
"Password cannot be empty": "Pasahitza ezin da hutsik utzi",
|
||||||
"preferences_video_loop_label": "Beti begiztatu: ",
|
"preferences_video_loop_label": "Beti begiztatu: ",
|
||||||
"Only show latest unwatched video from channel: ": "kanalaren azken bideo ezikusia erakutsi soilik ",
|
"Only show latest unwatched video from channel: ": "kanalaren azken bideo ezikusia erakutsi soilik ",
|
||||||
|
@ -261,9 +261,9 @@
|
||||||
"Hide annotations": "Oharrak izkutatu",
|
"Hide annotations": "Oharrak izkutatu",
|
||||||
"Title": "Titulua",
|
"Title": "Titulua",
|
||||||
"channel name": "Kanalaren izena",
|
"channel name": "Kanalaren izena",
|
||||||
"Authorize token for `x`?": "Baimendu tokena 'x'tzako?",
|
"Authorize token for `x`?": "Baimendu tokena `x`tzako?",
|
||||||
"Private": "Pribatua",
|
"Private": "Pribatua",
|
||||||
"Editing playlist `x`": "'x' zerrenda editatu",
|
"Editing playlist `x`": "`x` zerrenda editatu",
|
||||||
"Could not pull trending pages.": "Ezin ekarri orri arrakastatsuak.",
|
"Could not pull trending pages.": "Ezin ekarri orri arrakastatsuak.",
|
||||||
"crash_page_read_the_faq": "Bide <a href=\"`x`\"> (FAQ) ohiko galderak</a>"
|
"crash_page_read_the_faq": "Bide <a href=\"`x`\"> (FAQ) ohiko galderak</a>"
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
"View playlist on YouTube": "دیدن فهرست پخش در یوتیوب",
|
"View playlist on YouTube": "دیدن فهرست پخش در یوتیوب",
|
||||||
"newest": "تازهترین",
|
"newest": "تازهترین",
|
||||||
"oldest": "کهنهترین",
|
"oldest": "کهنهترین",
|
||||||
"popular": "محبوب",
|
"popular": "پرطرفدار",
|
||||||
"last": "آخرین",
|
"last": "آخرین",
|
||||||
"Next page": "صفحه بعد",
|
"Next page": "صفحه بعد",
|
||||||
"Previous page": "صفحه قبل",
|
"Previous page": "صفحه قبل",
|
||||||
|
@ -31,7 +31,7 @@
|
||||||
"Import and Export Data": "درونبرد و برونبرد داده",
|
"Import and Export Data": "درونبرد و برونبرد داده",
|
||||||
"Import": "درونبرد",
|
"Import": "درونبرد",
|
||||||
"Import Invidious data": "وارد کردن داده JSON اینویدیوس",
|
"Import Invidious data": "وارد کردن داده JSON اینویدیوس",
|
||||||
"Import YouTube subscriptions": "وارد کردن اشتراک OPML/ یوتیوب",
|
"Import YouTube subscriptions": "وارد کردن فایل CSV یا OPML سابسکرایب های یوتیوب",
|
||||||
"Import FreeTube subscriptions (.db)": "درونبرد اشتراکهای فریتیوب (.db)",
|
"Import FreeTube subscriptions (.db)": "درونبرد اشتراکهای فریتیوب (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "درونبرد اشتراکهای نیوپایپ (.json)",
|
"Import NewPipe subscriptions (.json)": "درونبرد اشتراکهای نیوپایپ (.json)",
|
||||||
"Import NewPipe data (.zip)": "درونبرد داده نیوپایپ (.zip)",
|
"Import NewPipe data (.zip)": "درونبرد داده نیوپایپ (.zip)",
|
||||||
|
@ -328,7 +328,7 @@
|
||||||
"generic_count_seconds": "{{count}} ثانیه",
|
"generic_count_seconds": "{{count}} ثانیه",
|
||||||
"generic_count_seconds_plural": "{{count}} ثانیه",
|
"generic_count_seconds_plural": "{{count}} ثانیه",
|
||||||
"Fallback comments: ": "نظرات عقب گرد: ",
|
"Fallback comments: ": "نظرات عقب گرد: ",
|
||||||
"Popular": "محبوب",
|
"Popular": "پربیننده",
|
||||||
"Search": "جست و جو",
|
"Search": "جست و جو",
|
||||||
"Top": "بالا",
|
"Top": "بالا",
|
||||||
"About": "درباره",
|
"About": "درباره",
|
||||||
|
@ -360,7 +360,7 @@
|
||||||
"search_filters_duration_label": "مدت",
|
"search_filters_duration_label": "مدت",
|
||||||
"search_filters_features_label": "ویژگیها",
|
"search_filters_features_label": "ویژگیها",
|
||||||
"search_filters_sort_label": "به ترتیب",
|
"search_filters_sort_label": "به ترتیب",
|
||||||
"search_filters_date_option_hour": "یک ساعت گذشته",
|
"search_filters_date_option_hour": "ساعت گذشته",
|
||||||
"search_filters_date_option_today": "امروز",
|
"search_filters_date_option_today": "امروز",
|
||||||
"search_filters_date_option_week": "این هفته",
|
"search_filters_date_option_week": "این هفته",
|
||||||
"search_filters_date_option_month": "این ماه",
|
"search_filters_date_option_month": "این ماه",
|
||||||
|
@ -461,7 +461,7 @@
|
||||||
"Song: ": "آهنگ: ",
|
"Song: ": "آهنگ: ",
|
||||||
"Channel Sponsor": "اسپانسر کانال",
|
"Channel Sponsor": "اسپانسر کانال",
|
||||||
"Standard YouTube license": "پروانه استاندارد YouTube",
|
"Standard YouTube license": "پروانه استاندارد YouTube",
|
||||||
"search_message_use_another_instance": " شما همچنین میتوانید <a href=\"`x`\">در نمونه دیگر هم جستجو کنید</a>.",
|
"search_message_use_another_instance": "همچنین میتوانید <a href=\"`x`\">در نمونهای دیگر هم جستوجو کنید</a>.",
|
||||||
"Download is disabled": "دریافت غیرفعال است",
|
"Download is disabled": "دریافت غیرفعال است",
|
||||||
"crash_page_before_reporting": "پیش از گزارش ایراد، مطمئنید شوید که:",
|
"crash_page_before_reporting": "پیش از گزارش ایراد، مطمئنید شوید که:",
|
||||||
"playlist_button_add_items": "افزودن ویدیو",
|
"playlist_button_add_items": "افزودن ویدیو",
|
||||||
|
@ -484,5 +484,17 @@
|
||||||
"channel_tab_shorts_label": "Shortها",
|
"channel_tab_shorts_label": "Shortها",
|
||||||
"channel_tab_playlists_label": "فهرستهای پخش",
|
"channel_tab_playlists_label": "فهرستهای پخش",
|
||||||
"channel_tab_channels_label": "کانالها",
|
"channel_tab_channels_label": "کانالها",
|
||||||
"error_video_not_in_playlist": "ویدیوی درخواستی معلق به این فهرست پخش نیست. <a href=\"`x`\">کلیک کنید تا به صفحهٔ اصلی فهرست پخش بروید.</a>"
|
"error_video_not_in_playlist": "ویدیوی درخواستی معلق به این فهرست پخش نیست. <a href=\"`x`\">کلیک کنید تا به صفحهٔ اصلی فهرست پخش بروید.</a>",
|
||||||
|
"Add to playlist": "به لیست پخش افزوده شود",
|
||||||
|
"Answer": "پاسخ",
|
||||||
|
"Search for videos": "جست و جو برای ویدیوها",
|
||||||
|
"Add to playlist: ": "افزودن به لیست پخش ",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "بخش ویدیوهای پرطرفدار توسط مدیر غیرفعال شده است.",
|
||||||
|
"carousel_slide": "اسلاید {{current}} از {{total}}",
|
||||||
|
"carousel_skip": "رد شدن از گرداننده",
|
||||||
|
"carousel_go_to": "به اسلاید `x` برو",
|
||||||
|
"crash_page_search_issue": "دنبال <a href=\"`x`\"> گشتیم بین مشکلات در گیت هاب </a>",
|
||||||
|
"crash_page_report_issue": "اگر هیچ یک از روش های بالا کمکی نکردند لطفا <a href=\"`x`\"> (ترجیحا به انگلیسی) یک سوال جدید در گیت هاب بپرسید و </a> طوری که سوالتون شامل متن زیر باشه:",
|
||||||
|
"channel_tab_releases_label": "آثار",
|
||||||
|
"toggle_theme": "تغییر وضعیت تم"
|
||||||
}
|
}
|
||||||
|
|
120
locales/fi.json
120
locales/fi.json
|
@ -28,7 +28,7 @@
|
||||||
"Export": "Vie",
|
"Export": "Vie",
|
||||||
"Export subscriptions as OPML": "Vie tilaukset OPML-muodossa",
|
"Export subscriptions as OPML": "Vie tilaukset OPML-muodossa",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Vie tilaukset OPML-muodossa (NewPipe & FreeTube)",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Vie tilaukset OPML-muodossa (NewPipe & FreeTube)",
|
||||||
"Export data as JSON": "Vie Invidious-data JSON-muodossa",
|
"Export data as JSON": "Vie Invidiousin tiedot JSON-muodossa",
|
||||||
"Delete account?": "Poista tili?",
|
"Delete account?": "Poista tili?",
|
||||||
"History": "Historia",
|
"History": "Historia",
|
||||||
"An alternative front-end to YouTube": "Vaihtoehtoinen front-end YouTubelle",
|
"An alternative front-end to YouTube": "Vaihtoehtoinen front-end YouTubelle",
|
||||||
|
@ -46,12 +46,12 @@
|
||||||
"E-mail": "Sähköposti",
|
"E-mail": "Sähköposti",
|
||||||
"Preferences": "Asetukset",
|
"Preferences": "Asetukset",
|
||||||
"preferences_category_player": "Soittimen asetukset",
|
"preferences_category_player": "Soittimen asetukset",
|
||||||
"preferences_video_loop_label": "Toista jatkuvasti aina: ",
|
"preferences_video_loop_label": "Toista aina uudelleen: ",
|
||||||
"preferences_autoplay_label": "Automaattinen toisto: ",
|
"preferences_autoplay_label": "Automaattinen toiston aloitus: ",
|
||||||
"preferences_continue_label": "Toista seuraava oletuksena: ",
|
"preferences_continue_label": "Toista seuraava oletuksena: ",
|
||||||
"preferences_continue_autoplay_label": "Toista seuraava video automaattisesti: ",
|
"preferences_continue_autoplay_label": "Aloita seuraava video automaattisesti: ",
|
||||||
"preferences_listen_label": "Kuuntele oletuksena: ",
|
"preferences_listen_label": "Kuuntele oletuksena: ",
|
||||||
"preferences_local_label": "Proxytä videot: ",
|
"preferences_local_label": "Videot välityspalvelimen kautta: ",
|
||||||
"preferences_speed_label": "Oletusnopeus: ",
|
"preferences_speed_label": "Oletusnopeus: ",
|
||||||
"preferences_quality_label": "Ensisijainen videon laatu: ",
|
"preferences_quality_label": "Ensisijainen videon laatu: ",
|
||||||
"preferences_volume_label": "Soittimen äänenvoimakkuus: ",
|
"preferences_volume_label": "Soittimen äänenvoimakkuus: ",
|
||||||
|
@ -63,7 +63,7 @@
|
||||||
"preferences_related_videos_label": "Näytä aiheeseen liittyviä videoita: ",
|
"preferences_related_videos_label": "Näytä aiheeseen liittyviä videoita: ",
|
||||||
"preferences_annotations_label": "Näytä huomautukset oletuksena: ",
|
"preferences_annotations_label": "Näytä huomautukset oletuksena: ",
|
||||||
"preferences_extend_desc_label": "Laajenna automaattisesti videon kuvausta: ",
|
"preferences_extend_desc_label": "Laajenna automaattisesti videon kuvausta: ",
|
||||||
"preferences_vr_mode_label": "Interaktiiviset 360-asteiset videot (vaatii WebGL:n): ",
|
"preferences_vr_mode_label": "Interaktiiviset 360-videot (vaatii WebGL:n): ",
|
||||||
"preferences_category_visual": "Visuaaliset asetukset",
|
"preferences_category_visual": "Visuaaliset asetukset",
|
||||||
"preferences_player_style_label": "Soittimen tyyli: ",
|
"preferences_player_style_label": "Soittimen tyyli: ",
|
||||||
"Dark mode: ": "Tumma tila: ",
|
"Dark mode: ": "Tumma tila: ",
|
||||||
|
@ -137,9 +137,9 @@
|
||||||
"Show less": "Näytä vähemmän",
|
"Show less": "Näytä vähemmän",
|
||||||
"Watch on YouTube": "Katso YouTubessa",
|
"Watch on YouTube": "Katso YouTubessa",
|
||||||
"Switch Invidious Instance": "Vaihda Invidious-instanssia",
|
"Switch Invidious Instance": "Vaihda Invidious-instanssia",
|
||||||
"Hide annotations": "Piilota merkkaukset",
|
"Hide annotations": "Piilota huomautukset",
|
||||||
"Show annotations": "Näytä merkkaukset",
|
"Show annotations": "Näytä huomautukset",
|
||||||
"Genre: ": "Genre: ",
|
"Genre: ": "Tyylilaji: ",
|
||||||
"License: ": "Lisenssi: ",
|
"License: ": "Lisenssi: ",
|
||||||
"Family friendly? ": "Kaiken ikäisille sopiva? ",
|
"Family friendly? ": "Kaiken ikäisille sopiva? ",
|
||||||
"Wilson score: ": "Wilson-pistemäärä: ",
|
"Wilson score: ": "Wilson-pistemäärä: ",
|
||||||
|
@ -168,7 +168,7 @@
|
||||||
"Wrong username or password": "Väärä käyttäjänimi tai salasana",
|
"Wrong username or password": "Väärä käyttäjänimi tai salasana",
|
||||||
"Password cannot be empty": "Salasana ei voi olla tyhjä",
|
"Password cannot be empty": "Salasana ei voi olla tyhjä",
|
||||||
"Password cannot be longer than 55 characters": "Salasana ei voi olla yli 55 merkkiä pitkä",
|
"Password cannot be longer than 55 characters": "Salasana ei voi olla yli 55 merkkiä pitkä",
|
||||||
"Please log in": "Kirjaudu sisään, ole hyvä",
|
"Please log in": "Kirjaudu sisään",
|
||||||
"Invidious Private Feed for `x`": "Invidiousin yksityinen syöte `x`:lle",
|
"Invidious Private Feed for `x`": "Invidiousin yksityinen syöte `x`:lle",
|
||||||
"channel:`x`": "kanava:`x`",
|
"channel:`x`": "kanava:`x`",
|
||||||
"Deleted or invalid channel": "Poistettu tai virheellinen kanava",
|
"Deleted or invalid channel": "Poistettu tai virheellinen kanava",
|
||||||
|
@ -178,7 +178,7 @@
|
||||||
"`x` ago": "`x` sitten",
|
"`x` ago": "`x` sitten",
|
||||||
"Load more": "Lataa lisää",
|
"Load more": "Lataa lisää",
|
||||||
"Could not create mix.": "Sekoituksen luominen epäonnistui.",
|
"Could not create mix.": "Sekoituksen luominen epäonnistui.",
|
||||||
"Empty playlist": "Tyhjennä soittolista",
|
"Empty playlist": "Tyhjä soittolista",
|
||||||
"Not a playlist.": "Ei ole soittolista.",
|
"Not a playlist.": "Ei ole soittolista.",
|
||||||
"Playlist does not exist.": "Soittolistaa ei ole olemassa.",
|
"Playlist does not exist.": "Soittolistaa ei ole olemassa.",
|
||||||
"Could not pull trending pages.": "Nousussa olevien sivujen lataus epäonnistui.",
|
"Could not pull trending pages.": "Nousussa olevien sivujen lataus epäonnistui.",
|
||||||
|
@ -216,11 +216,11 @@
|
||||||
"Filipino": "filipino",
|
"Filipino": "filipino",
|
||||||
"Finnish": "suomi",
|
"Finnish": "suomi",
|
||||||
"French": "ranska",
|
"French": "ranska",
|
||||||
"Galician": "galego",
|
"Galician": "galicia",
|
||||||
"Georgian": "georgia",
|
"Georgian": "georgia",
|
||||||
"German": "saksa",
|
"German": "saksa",
|
||||||
"Greek": "kreikka",
|
"Greek": "kreikka",
|
||||||
"Gujarati": "gujarati",
|
"Gujarati": "gudžarati",
|
||||||
"Haitian Creole": "haitinkreoli",
|
"Haitian Creole": "haitinkreoli",
|
||||||
"Hausa": "hausa",
|
"Hausa": "hausa",
|
||||||
"Hawaiian": "havaiji",
|
"Hawaiian": "havaiji",
|
||||||
|
@ -327,11 +327,11 @@
|
||||||
"search_filters_duration_label": "Kesto",
|
"search_filters_duration_label": "Kesto",
|
||||||
"search_filters_features_label": "Ominaisuudet",
|
"search_filters_features_label": "Ominaisuudet",
|
||||||
"search_filters_sort_label": "Luokittele",
|
"search_filters_sort_label": "Luokittele",
|
||||||
"search_filters_date_option_hour": "Viimeisin tunti",
|
"search_filters_date_option_hour": "Tunnin sisään",
|
||||||
"search_filters_date_option_today": "Tänään",
|
"search_filters_date_option_today": "Tänään",
|
||||||
"search_filters_date_option_week": "Tämä viikko",
|
"search_filters_date_option_week": "Tällä viikolla",
|
||||||
"search_filters_date_option_month": "Tämä kuukausi",
|
"search_filters_date_option_month": "Tässä kuussa",
|
||||||
"search_filters_date_option_year": "Tämä vuosi",
|
"search_filters_date_option_year": "Tänä vuonna",
|
||||||
"search_filters_type_option_video": "Video",
|
"search_filters_type_option_video": "Video",
|
||||||
"search_filters_type_option_channel": "Kanava",
|
"search_filters_type_option_channel": "Kanava",
|
||||||
"search_filters_type_option_playlist": "Soittolista",
|
"search_filters_type_option_playlist": "Soittolista",
|
||||||
|
@ -346,7 +346,7 @@
|
||||||
"search_filters_features_option_location": "Sijainti",
|
"search_filters_features_option_location": "Sijainti",
|
||||||
"search_filters_features_option_hdr": "HDR",
|
"search_filters_features_option_hdr": "HDR",
|
||||||
"Current version: ": "Tämänhetkinen versio: ",
|
"Current version: ": "Tämänhetkinen versio: ",
|
||||||
"next_steps_error_message": "Sinun tulisi kokeilla seuraavia: ",
|
"next_steps_error_message": "Kokeile seuraavia: ",
|
||||||
"next_steps_error_message_refresh": "Päivitä",
|
"next_steps_error_message_refresh": "Päivitä",
|
||||||
"next_steps_error_message_go_to_youtube": "Siirry YouTubeen",
|
"next_steps_error_message_go_to_youtube": "Siirry YouTubeen",
|
||||||
"generic_count_hours": "{{count}} tunti",
|
"generic_count_hours": "{{count}} tunti",
|
||||||
|
@ -391,7 +391,7 @@
|
||||||
"subscriptions_unseen_notifs_count": "{{count}} näkemätön ilmoitus",
|
"subscriptions_unseen_notifs_count": "{{count}} näkemätön ilmoitus",
|
||||||
"subscriptions_unseen_notifs_count_plural": "{{count}} näkemätöntä ilmoitusta",
|
"subscriptions_unseen_notifs_count_plural": "{{count}} näkemätöntä ilmoitusta",
|
||||||
"crash_page_switch_instance": "yrittänyt <a href=\"`x`\">käyttää toista instassia</a>",
|
"crash_page_switch_instance": "yrittänyt <a href=\"`x`\">käyttää toista instassia</a>",
|
||||||
"videoinfo_invidious_embed_link": "Upotuslinkki",
|
"videoinfo_invidious_embed_link": "Upotettava linkki",
|
||||||
"user_saved_playlists": "`x` tallennetua soittolistaa",
|
"user_saved_playlists": "`x` tallennetua soittolistaa",
|
||||||
"crash_page_report_issue": "Jos mikään näistä ei auttanut, <a href=\"`x`\">avaathan uuden issuen GitHubissa</a> (mieluiten englanniksi) ja sisällytät seuraavan tekstin viestissäsi (ÄLÄ käännä tätä tekstiä):",
|
"crash_page_report_issue": "Jos mikään näistä ei auttanut, <a href=\"`x`\">avaathan uuden issuen GitHubissa</a> (mieluiten englanniksi) ja sisällytät seuraavan tekstin viestissäsi (ÄLÄ käännä tätä tekstiä):",
|
||||||
"preferences_quality_option_hd720": "HD720",
|
"preferences_quality_option_hd720": "HD720",
|
||||||
|
@ -410,7 +410,7 @@
|
||||||
"preferences_quality_dash_option_auto": "Auto",
|
"preferences_quality_dash_option_auto": "Auto",
|
||||||
"preferences_quality_dash_option_best": "Paras",
|
"preferences_quality_dash_option_best": "Paras",
|
||||||
"preferences_quality_option_dash": "DASH (mukautuva laatu)",
|
"preferences_quality_option_dash": "DASH (mukautuva laatu)",
|
||||||
"preferences_quality_dash_label": "Haluttava DASH-videolaatu: ",
|
"preferences_quality_dash_label": "Ensisijainen DASH-videolaatu: ",
|
||||||
"generic_count_years": "{{count}} vuosi",
|
"generic_count_years": "{{count}} vuosi",
|
||||||
"generic_count_years_plural": "{{count}} vuotta",
|
"generic_count_years_plural": "{{count}} vuotta",
|
||||||
"search_filters_features_option_purchased": "Ostettu",
|
"search_filters_features_option_purchased": "Ostettu",
|
||||||
|
@ -421,39 +421,39 @@
|
||||||
"preferences_save_player_pos_label": "Tallenna toistokohta: ",
|
"preferences_save_player_pos_label": "Tallenna toistokohta: ",
|
||||||
"footer_donate_page": "Lahjoita",
|
"footer_donate_page": "Lahjoita",
|
||||||
"footer_source_code": "Lähdekoodi",
|
"footer_source_code": "Lähdekoodi",
|
||||||
"adminprefs_modified_source_code_url_label": "URL muokattuun lähdekoodirepositoryyn",
|
"adminprefs_modified_source_code_url_label": "URL muokatun lähdekoodin repositorioon",
|
||||||
"Released under the AGPLv3 on Github.": "Julkaistu AGPLv3-lisenssin alla GitHubissa.",
|
"Released under the AGPLv3 on Github.": "Julkaistu AGPLv3-lisenssillä GitHubissa.",
|
||||||
"search_filters_duration_option_short": "Lyhyt (< 4 minuuttia)",
|
"search_filters_duration_option_short": "Lyhyt (< 4 minuuttia)",
|
||||||
"search_filters_duration_option_long": "Pitkä (> 20 minuuttia)",
|
"search_filters_duration_option_long": "Pitkä (> 20 minuuttia)",
|
||||||
"footer_documentation": "Dokumentaatio",
|
"footer_documentation": "Dokumentaatio",
|
||||||
"footer_original_source_code": "Alkuperäinen lähdekoodi",
|
"footer_original_source_code": "Alkuperäinen lähdekoodi",
|
||||||
"footer_modfied_source_code": "Muokattu lähdekoodi",
|
"footer_modfied_source_code": "Muokattu lähdekoodi",
|
||||||
"Japanese (auto-generated)": "Japani (automaattisesti luotu)",
|
"Japanese (auto-generated)": "japani (automaattisesti luotu)",
|
||||||
"German (auto-generated)": "Saksa (automaattisesti luotu)",
|
"German (auto-generated)": "saksa (automaattisesti luotu)",
|
||||||
"Portuguese (auto-generated)": "portugali (automaattisesti luotu)",
|
"Portuguese (auto-generated)": "portugali (automaattisesti luotu)",
|
||||||
"Russian (auto-generated)": "Venäjä (automaattisesti luotu)",
|
"Russian (auto-generated)": "Venäjä (automaattisesti luotu)",
|
||||||
"preferences_watch_history_label": "Ota katseluhistoria käyttöön: ",
|
"preferences_watch_history_label": "Ota katseluhistoria käyttöön: ",
|
||||||
"English (United Kingdom)": "Englanti (Iso-Britannia)",
|
"English (United Kingdom)": "englanti (Iso-Britannia)",
|
||||||
"English (United States)": "Englanti (Yhdysvallat)",
|
"English (United States)": "englanti (Yhdysvallat)",
|
||||||
"Cantonese (Hong Kong)": "Kantoninkiina (Hong Kong)",
|
"Cantonese (Hong Kong)": "kantoninkiina (Hongkong)",
|
||||||
"Chinese": "Kiina",
|
"Chinese": "kiina",
|
||||||
"Chinese (China)": "Kiina (Kiina)",
|
"Chinese (China)": "kiina (Kiina)",
|
||||||
"Chinese (Hong Kong)": "Kiina (Hong Kong)",
|
"Chinese (Hong Kong)": "kiina (Hongkong)",
|
||||||
"Chinese (Taiwan)": "Kiina (Taiwan)",
|
"Chinese (Taiwan)": "kiina (Taiwan)",
|
||||||
"Dutch (auto-generated)": "Hollanti (automaattisesti luotu)",
|
"Dutch (auto-generated)": "hollanti (automaattisesti luotu)",
|
||||||
"French (auto-generated)": "Ranska (automaattisesti luotu)",
|
"French (auto-generated)": "ranska (automaattisesti luotu)",
|
||||||
"Indonesian (auto-generated)": "Indonesia (automaattisesti luotu)",
|
"Indonesian (auto-generated)": "indonesia (automaattisesti luotu)",
|
||||||
"Interlingue": "Interlingue",
|
"Interlingue": "interlingue",
|
||||||
"Italian (auto-generated)": "Italia (automaattisesti luotu)",
|
"Italian (auto-generated)": "Italia (automaattisesti luotu)",
|
||||||
"Korean (auto-generated)": "Korea (automaattisesti luotu)",
|
"Korean (auto-generated)": "korea (automaattisesti luotu)",
|
||||||
"Portuguese (Brazil)": "portugali (Brasilia)",
|
"Portuguese (Brazil)": "portugali (Brasilia)",
|
||||||
"Spanish (auto-generated)": "Espanja (automaattisesti luotu)",
|
"Spanish (auto-generated)": "espanja (automaattisesti luotu)",
|
||||||
"Spanish (Mexico)": "Espanja (Meksiko)",
|
"Spanish (Mexico)": "espanja (Meksiko)",
|
||||||
"Spanish (Spain)": "Espanja (Espanja)",
|
"Spanish (Spain)": "espanja (Espanja)",
|
||||||
"Turkish (auto-generated)": "Turkki (automaattisesti luotu)",
|
"Turkish (auto-generated)": "turkki (automaattisesti luotu)",
|
||||||
"Vietnamese (auto-generated)": "Vietnam (automaattisesti luotu)",
|
"Vietnamese (auto-generated)": "vietnam (automaattisesti luotu)",
|
||||||
"search_filters_title": "Suodatin",
|
"search_filters_title": "Suodattimet",
|
||||||
"search_message_no_results": "Ei tuloksia löydetty.",
|
"search_message_no_results": "Tuloksia ei löytynyt.",
|
||||||
"search_message_change_filters_or_query": "Yritä hakukyselysi laajentamista ja/tai suodattimien muuttamista.",
|
"search_message_change_filters_or_query": "Yritä hakukyselysi laajentamista ja/tai suodattimien muuttamista.",
|
||||||
"search_filters_duration_option_none": "Mikä tahansa kesto",
|
"search_filters_duration_option_none": "Mikä tahansa kesto",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
|
@ -464,5 +464,37 @@
|
||||||
"search_filters_date_option_none": "Milloin tahansa",
|
"search_filters_date_option_none": "Milloin tahansa",
|
||||||
"search_filters_type_option_all": "Mikä tahansa tyyppi",
|
"search_filters_type_option_all": "Mikä tahansa tyyppi",
|
||||||
"Popular enabled: ": "Suosittu käytössä: ",
|
"Popular enabled: ": "Suosittu käytössä: ",
|
||||||
"error_video_not_in_playlist": "Pyydettyä videota ei löydy tästä soittolistasta. <a href=\"`x`\">Klikkaa tähän päästäksesi soittolistan etusivulle.</a>"
|
"error_video_not_in_playlist": "Pyydettyä videota ei ole tässä soittolistassa. <a href=\"`x`\">Klikkaa tästä päästäksesi soittolistan kotisivulle.</a>",
|
||||||
|
"Import YouTube playlist (.csv)": "Tuo YouTube-soittolista (.csv)",
|
||||||
|
"Music in this video": "Musiikki tässä videossa",
|
||||||
|
"Add to playlist": "Lisää soittolistaan",
|
||||||
|
"Add to playlist: ": "Lisää soittolistaan: ",
|
||||||
|
"Search for videos": "Etsi videoita",
|
||||||
|
"generic_button_rss": "RSS",
|
||||||
|
"Answer": "Vastaus",
|
||||||
|
"Standard YouTube license": "Vakio YouTube-lisenssi",
|
||||||
|
"Song: ": "Kappale: ",
|
||||||
|
"Album: ": "Albumi: ",
|
||||||
|
"Download is disabled": "Lataus on poistettu käytöstä",
|
||||||
|
"Channel Sponsor": "Kanavan sponsori",
|
||||||
|
"channel_tab_podcasts_label": "Podcastit",
|
||||||
|
"channel_tab_releases_label": "Julkaisut",
|
||||||
|
"channel_tab_shorts_label": "Shorts-videot",
|
||||||
|
"carousel_slide": "Dia {{current}}/{{total}}",
|
||||||
|
"carousel_skip": "Ohita karuselli",
|
||||||
|
"carousel_go_to": "Siirry diaan `x`",
|
||||||
|
"channel_tab_playlists_label": "Soittolistat",
|
||||||
|
"channel_tab_channels_label": "Kanavat",
|
||||||
|
"generic_button_delete": "Poista",
|
||||||
|
"generic_button_edit": "Muokkaa",
|
||||||
|
"generic_button_save": "Tallenna",
|
||||||
|
"generic_button_cancel": "Peru",
|
||||||
|
"playlist_button_add_items": "Lisää videoita",
|
||||||
|
"Artist: ": "Esittäjä: ",
|
||||||
|
"channel_tab_streams_label": "Suoratoistot",
|
||||||
|
"generic_channels_count": "{{count}} kanava",
|
||||||
|
"generic_channels_count_plural": "{{count}} kanavaa",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Järjestelmänvalvoja on poistanut Suositut-syötteen.",
|
||||||
|
"Import YouTube watch history (.json)": "Tuo Youtube-katseluhistoria (.json)",
|
||||||
|
"toggle_theme": "Vaihda teemaa"
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
"generic_subscriptions_count_1": "{{count}} d'abonnements",
|
"generic_subscriptions_count_1": "{{count}} d'abonnements",
|
||||||
"generic_subscriptions_count_2": "{{count}} abonnements",
|
"generic_subscriptions_count_2": "{{count}} abonnements",
|
||||||
"generic_button_delete": "Supprimer",
|
"generic_button_delete": "Supprimer",
|
||||||
"generic_button_edit": "Editer",
|
"generic_button_edit": "Modifier",
|
||||||
"generic_button_save": "Enregistrer",
|
"generic_button_save": "Enregistrer",
|
||||||
"generic_button_cancel": "Annuler",
|
"generic_button_cancel": "Annuler",
|
||||||
"generic_button_rss": "RSS",
|
"generic_button_rss": "RSS",
|
||||||
|
@ -44,7 +44,7 @@
|
||||||
"Import and Export Data": "Importer et exporter des données",
|
"Import and Export Data": "Importer et exporter des données",
|
||||||
"Import": "Importer",
|
"Import": "Importer",
|
||||||
"Import Invidious data": "Importer des données Invidious au format JSON",
|
"Import Invidious data": "Importer des données Invidious au format JSON",
|
||||||
"Import YouTube subscriptions": "Importer des abonnements YouTube/OPML",
|
"Import YouTube subscriptions": "Importer des abonnements YouTube aux formats OPML/CSV",
|
||||||
"Import FreeTube subscriptions (.db)": "Importer des abonnements FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importer des abonnements FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importer des abonnements NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importer des abonnements NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importer des données NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importer des données NewPipe (.zip)",
|
||||||
|
@ -484,7 +484,7 @@
|
||||||
"search_filters_duration_option_medium": "Moyenne (de 4 à 20 minutes)",
|
"search_filters_duration_option_medium": "Moyenne (de 4 à 20 minutes)",
|
||||||
"search_filters_apply_button": "Appliquer les filtres",
|
"search_filters_apply_button": "Appliquer les filtres",
|
||||||
"search_message_no_results": "Aucun résultat.",
|
"search_message_no_results": "Aucun résultat.",
|
||||||
"search_message_use_another_instance": " Vous pouvez également <a href=\"`x`\">effectuer votre recherche sur une autre instance</a>.",
|
"search_message_use_another_instance": "Vous pouvez également <a href=\"`x`\">effectuer votre recherche sur une autre instance</a>.",
|
||||||
"search_filters_type_option_all": "Tous les types",
|
"search_filters_type_option_all": "Tous les types",
|
||||||
"search_filters_date_label": "Date d'ajout",
|
"search_filters_date_label": "Date d'ajout",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
|
@ -504,5 +504,14 @@
|
||||||
"Import YouTube playlist (.csv)": "Importer des listes de lecture de Youtube (.csv)",
|
"Import YouTube playlist (.csv)": "Importer des listes de lecture de Youtube (.csv)",
|
||||||
"channel_tab_releases_label": "Parutions",
|
"channel_tab_releases_label": "Parutions",
|
||||||
"channel_tab_podcasts_label": "Émissions audio",
|
"channel_tab_podcasts_label": "Émissions audio",
|
||||||
"Import YouTube watch history (.json)": "Importer l'historique de visionnement YouTube (.json)"
|
"Import YouTube watch history (.json)": "Importer l'historique de visionnement YouTube (.json)",
|
||||||
|
"Add to playlist: ": "Ajouter à la playlist : ",
|
||||||
|
"Add to playlist": "Ajouter à la playlist",
|
||||||
|
"Answer": "Répondre",
|
||||||
|
"Search for videos": "Rechercher des vidéos",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Le flux populaire a été désactivé par l'administrateur.",
|
||||||
|
"carousel_skip": "Passez le carrousel",
|
||||||
|
"carousel_slide": "Diapositive {{current}} sur {{total}}",
|
||||||
|
"carousel_go_to": "Aller à la diapositive `x`",
|
||||||
|
"toggle_theme": "Changer le Thème"
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,7 +62,7 @@
|
||||||
"Import and Export Data": "डेटा को आयात और निर्यात करें",
|
"Import and Export Data": "डेटा को आयात और निर्यात करें",
|
||||||
"Import": "आयात करें",
|
"Import": "आयात करें",
|
||||||
"Import Invidious data": "Invidious JSON डेटा आयात करें",
|
"Import Invidious data": "Invidious JSON डेटा आयात करें",
|
||||||
"Import YouTube subscriptions": "YouTube/OPML सदस्यताएँ आयात करें",
|
"Import YouTube subscriptions": "YouTube CSV या OPML सदस्यताएँ आयात करें",
|
||||||
"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)",
|
||||||
|
@ -487,5 +487,14 @@
|
||||||
"Download is disabled": "डाउनलोड करना अक्षम है",
|
"Download is disabled": "डाउनलोड करना अक्षम है",
|
||||||
"generic_channels_count": "{{count}} चैनल",
|
"generic_channels_count": "{{count}} चैनल",
|
||||||
"generic_channels_count_plural": "{{count}} चैनल",
|
"generic_channels_count_plural": "{{count}} चैनल",
|
||||||
"Import YouTube watch history (.json)": "YouTube पर देखने का इतिहास आयात करें (.json)"
|
"Import YouTube watch history (.json)": "YouTube पर देखने का इतिहास आयात करें (.json)",
|
||||||
|
"Add to playlist": "प्लेलिस्ट में जोड़ें",
|
||||||
|
"Answer": "जवाब",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "लोकप्रिय फ़ीड व्यवस्थापक द्वारा अक्षम कर दिया गया है।",
|
||||||
|
"toggle_theme": "थीम टॉगल करें",
|
||||||
|
"carousel_slide": "{{total}} में से स्लाइड {{current}}",
|
||||||
|
"carousel_skip": "कैरोसेल छोड़ें",
|
||||||
|
"Add to playlist: ": "प्लेलिस्ट में जोड़ें: ",
|
||||||
|
"Search for videos": "वीडियो खोजें",
|
||||||
|
"carousel_go_to": "स्लाइड `x` पर जाएँ"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Uvezi i izvezi podatke",
|
"Import and Export Data": "Uvezi i izvezi podatke",
|
||||||
"Import": "Uvezi",
|
"Import": "Uvezi",
|
||||||
"Import Invidious data": "Uvezi Invidious JSON podatke",
|
"Import Invidious data": "Uvezi Invidious JSON podatke",
|
||||||
"Import YouTube subscriptions": "Uvezi YouTube/OPML pretplate",
|
"Import YouTube subscriptions": "Uvezi YouTube CSV ili OPML pretplate",
|
||||||
"Import FreeTube subscriptions (.db)": "Uvezi FreeTube pretplate (.db)",
|
"Import FreeTube subscriptions (.db)": "Uvezi FreeTube pretplate (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Uvezi NewPipe pretplate (.json)",
|
"Import NewPipe subscriptions (.json)": "Uvezi NewPipe pretplate (.json)",
|
||||||
"Import NewPipe data (.zip)": "Uvezi NewPipe podatke (.zip)",
|
"Import NewPipe data (.zip)": "Uvezi NewPipe podatke (.zip)",
|
||||||
|
@ -449,30 +449,30 @@
|
||||||
"Cantonese (Hong Kong)": "Kantonski (Hong Kong)",
|
"Cantonese (Hong Kong)": "Kantonski (Hong Kong)",
|
||||||
"Chinese": "Kineski",
|
"Chinese": "Kineski",
|
||||||
"Chinese (Taiwan)": "Kineski (Tajvan)",
|
"Chinese (Taiwan)": "Kineski (Tajvan)",
|
||||||
"Dutch (auto-generated)": "Nizozemski (automatski generiran)",
|
"Dutch (auto-generated)": "Nizozemski (automatski generirano)",
|
||||||
"French (auto-generated)": "Francuski (automatski generiran)",
|
"French (auto-generated)": "Francuski (automatski generirano)",
|
||||||
"Indonesian (auto-generated)": "Indonezijski (automatski generiran)",
|
"Indonesian (auto-generated)": "Indonezijski (automatski generirano)",
|
||||||
"Interlingue": "Interlingua",
|
"Interlingue": "Interlingua",
|
||||||
"Japanese (auto-generated)": "Japanski (automatski generiran)",
|
"Japanese (auto-generated)": "Japanski (automatski generirano)",
|
||||||
"Russian (auto-generated)": "Ruski (automatski generiran)",
|
"Russian (auto-generated)": "Ruski (automatski generirano)",
|
||||||
"Turkish (auto-generated)": "Turski (automatski generiran)",
|
"Turkish (auto-generated)": "Turski (automatski generirano)",
|
||||||
"Vietnamese (auto-generated)": "Vijetnamski (automatski generiran)",
|
"Vietnamese (auto-generated)": "Vijetnamski (automatski generirano)",
|
||||||
"Spanish (Spain)": "Španjolski (Španjolska)",
|
"Spanish (Spain)": "Španjolski (Španjolska)",
|
||||||
"Italian (auto-generated)": "Talijanski (automatski generiran)",
|
"Italian (auto-generated)": "Talijanski (automatski generirano)",
|
||||||
"Portuguese (Brazil)": "Portugalski (Brazil)",
|
"Portuguese (Brazil)": "Portugalski (Brazil)",
|
||||||
"Spanish (Mexico)": "Španjolski (Meksiko)",
|
"Spanish (Mexico)": "Španjolski (Meksiko)",
|
||||||
"German (auto-generated)": "Njemački (automatski generiran)",
|
"German (auto-generated)": "Njemački (automatski generirano)",
|
||||||
"Chinese (China)": "Kineski (Kina)",
|
"Chinese (China)": "Kineski (Kina)",
|
||||||
"Chinese (Hong Kong)": "Kineski (Hong Kong)",
|
"Chinese (Hong Kong)": "Kineski (Hong Kong)",
|
||||||
"Korean (auto-generated)": "Korejski (automatski generiran)",
|
"Korean (auto-generated)": "Korejski (automatski generirano)",
|
||||||
"Portuguese (auto-generated)": "Portugalski (automatski generiran)",
|
"Portuguese (auto-generated)": "Portugalski (automatski generirano)",
|
||||||
"Spanish (auto-generated)": "Španjolski (automatski generiran)",
|
"Spanish (auto-generated)": "Španjolski (automatski generirano)",
|
||||||
"preferences_watch_history_label": "Aktiviraj povijest gledanja: ",
|
"preferences_watch_history_label": "Aktiviraj povijest gledanja: ",
|
||||||
"search_filters_title": "Filtri",
|
"search_filters_title": "Filtri",
|
||||||
"search_filters_date_option_none": "Bilo koji datum",
|
"search_filters_date_option_none": "Bilo koji datum",
|
||||||
"search_filters_date_label": "Datum prijenosa",
|
"search_filters_date_label": "Datum prijenosa",
|
||||||
"search_message_no_results": "Nema rezultata.",
|
"search_message_no_results": "Nema rezultata.",
|
||||||
"search_message_use_another_instance": " Također možeš <a href=\"`x`\">tražiti na jednoj drugoj instanci</a>.",
|
"search_message_use_another_instance": "Također možeš <a href=\"`x`\">tražiti na jednoj drugoj instanci</a>.",
|
||||||
"search_message_change_filters_or_query": "Pokušaj proširiti upit za pretragu i/ili promijeni filtre.",
|
"search_message_change_filters_or_query": "Pokušaj proširiti upit za pretragu i/ili promijeni filtre.",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
"search_filters_duration_option_none": "Bilo koje duljine",
|
"search_filters_duration_option_none": "Bilo koje duljine",
|
||||||
|
@ -504,5 +504,14 @@
|
||||||
"generic_channels_count_0": "{{count}} kanal",
|
"generic_channels_count_0": "{{count}} kanal",
|
||||||
"generic_channels_count_1": "{{count}} kanala",
|
"generic_channels_count_1": "{{count}} kanala",
|
||||||
"generic_channels_count_2": "{{count}} kanala",
|
"generic_channels_count_2": "{{count}} kanala",
|
||||||
"Import YouTube watch history (.json)": "Uvezi YouTube povijest gledanja (.json)"
|
"Import YouTube watch history (.json)": "Uvezi YouTube povijest gledanja (.json)",
|
||||||
|
"Add to playlist": "Dodaj u zbirku",
|
||||||
|
"Add to playlist: ": "Dodaj u zbirku: ",
|
||||||
|
"Answer": "Odgovor",
|
||||||
|
"Search for videos": "Traži videa",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Popularni feed je administrator deaktivirao.",
|
||||||
|
"toggle_theme": "Uklj./Isklj. temu",
|
||||||
|
"carousel_slide": "Kadar {{current}} od {{total}}",
|
||||||
|
"carousel_go_to": "Idi na kadar `x`",
|
||||||
|
"carousel_skip": "Preskoči vrtuljak"
|
||||||
}
|
}
|
||||||
|
|
|
@ -464,5 +464,23 @@
|
||||||
"search_filters_features_option_vr180": "180°-os virtuális valóság",
|
"search_filters_features_option_vr180": "180°-os virtuális valóság",
|
||||||
"search_filters_apply_button": "Keresés a megadott szűrőkkel",
|
"search_filters_apply_button": "Keresés a megadott szűrőkkel",
|
||||||
"Popular enabled: ": "Népszerű engedélyezve ",
|
"Popular enabled: ": "Népszerű engedélyezve ",
|
||||||
"error_video_not_in_playlist": "A lejátszási listában keresett videó nem létezik. <a href=\"`x`\">Kattintson ide a lejátszási listához jutáshoz.</a>"
|
"error_video_not_in_playlist": "A lejátszási listában keresett videó nem létezik. <a href=\"`x`\">Kattintson ide a lejátszási listához jutáshoz.</a>",
|
||||||
|
"generic_button_delete": "Törlés",
|
||||||
|
"generic_button_rss": "RSS",
|
||||||
|
"Import YouTube playlist (.csv)": "Youtube lejátszási lista (.csv) importálása",
|
||||||
|
"Standard YouTube license": "Alap YouTube-licensz",
|
||||||
|
"Add to playlist": "Hozzáadás lejátszási listához",
|
||||||
|
"Add to playlist: ": "Hozzáadás a lejátszási listához: ",
|
||||||
|
"Answer": "Válasz",
|
||||||
|
"Search for videos": "Keresés videókhoz",
|
||||||
|
"generic_channels_count": "{{count}} csatorna",
|
||||||
|
"generic_channels_count_plural": "{{count}} csatornák",
|
||||||
|
"generic_button_edit": "Szerkesztés",
|
||||||
|
"generic_button_save": "Mentés",
|
||||||
|
"generic_button_cancel": "Mégsem",
|
||||||
|
"playlist_button_add_items": "Videók hozzáadása",
|
||||||
|
"Music in this video": "Zene ezen videóban",
|
||||||
|
"Song: ": "Dal: ",
|
||||||
|
"Album: ": "Album: ",
|
||||||
|
"Import YouTube watch history (.json)": "Youtube megtekintési előzmények (.json) importálása"
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
"invidious": "Invidious",
|
"invidious": "Invidious",
|
||||||
"Image CAPTCHA": "Imagine CAPTCHA",
|
"Image CAPTCHA": "Imagine CAPTCHA",
|
||||||
"newest": "plus nove",
|
"newest": "plus nove",
|
||||||
"generic_button_save": "Salvar",
|
"generic_button_save": "Salveguardar",
|
||||||
"Dark mode: ": "Modo obscur: ",
|
"Dark mode: ": "Modo obscur: ",
|
||||||
"preferences_dark_mode_label": "Thema: ",
|
"preferences_dark_mode_label": "Thema: ",
|
||||||
"preferences_category_subscription": "Preferentias de subscription",
|
"preferences_category_subscription": "Preferentias de subscription",
|
||||||
|
@ -23,7 +23,7 @@
|
||||||
"light": "clar",
|
"light": "clar",
|
||||||
"No": "Non",
|
"No": "Non",
|
||||||
"youtube": "YouTube",
|
"youtube": "YouTube",
|
||||||
"LIVE": "IN DIRECTE",
|
"LIVE": "IN DIRECTO",
|
||||||
"reddit": "Reddit",
|
"reddit": "Reddit",
|
||||||
"preferences_category_player": "Preferentias de reproductor",
|
"preferences_category_player": "Preferentias de reproductor",
|
||||||
"Preferences": "Preferentias",
|
"Preferences": "Preferentias",
|
||||||
|
@ -37,5 +37,9 @@
|
||||||
"E-mail": "E-mail",
|
"E-mail": "E-mail",
|
||||||
"Delete account?": "Deler conto?",
|
"Delete account?": "Deler conto?",
|
||||||
"preferences_volume_label": "Volumine del reproductor: ",
|
"preferences_volume_label": "Volumine del reproductor: ",
|
||||||
"preferences_sort_label": "Ordinar le videos per: "
|
"preferences_sort_label": "Ordinar le videos per: ",
|
||||||
|
"Next page": "Pagina sequente",
|
||||||
|
"Previous page": "Pagina previe",
|
||||||
|
"Yes": "Si",
|
||||||
|
"Import": "Importar"
|
||||||
}
|
}
|
||||||
|
|
293
locales/is.json
293
locales/is.json
|
@ -1,39 +1,39 @@
|
||||||
{
|
{
|
||||||
"LIVE": "BEINT",
|
"LIVE": "BEINT",
|
||||||
"Shared `x` ago": "Deilt `x` síðan",
|
"Shared `x` ago": "Deilt fyrir `x` síðan",
|
||||||
"Unsubscribe": "Afskrá",
|
"Unsubscribe": "Afskrá",
|
||||||
"Subscribe": "Áskrifa",
|
"Subscribe": "Áskrifa",
|
||||||
"View channel on YouTube": "Skoða rás á YouTube",
|
"View channel on YouTube": "Skoða rás á YouTube",
|
||||||
"View playlist on YouTube": "Skoða spilunarlisti á YouTube",
|
"View playlist on YouTube": "Skoða spilunarlista á YouTube",
|
||||||
"newest": "nýjasta",
|
"newest": "nýjasta",
|
||||||
"oldest": "elsta",
|
"oldest": "elsta",
|
||||||
"popular": "vinsælt",
|
"popular": "vinsælt",
|
||||||
"last": "síðast",
|
"last": "síðast",
|
||||||
"Next page": "Næsta síða",
|
"Next page": "Næsta síða",
|
||||||
"Previous page": "Fyrri síða",
|
"Previous page": "Fyrri síða",
|
||||||
"Clear watch history?": "Hreinsa áhorfssögu?",
|
"Clear watch history?": "Hreinsa áhorfsferil?",
|
||||||
"New password": "Nýtt lykilorð",
|
"New password": "Nýtt lykilorð",
|
||||||
"New passwords must match": "Nýtt lykilorð verður að passa",
|
"New passwords must match": "Nýtt lykilorð verður að passa",
|
||||||
"Authorize token?": "Leyfa tákn?",
|
"Authorize token?": "Leyfa teikn?",
|
||||||
"Authorize token for `x`?": "Leyfa tákn fyrir `x`?",
|
"Authorize token for `x`?": "Leyfa teikn fyrir `x`?",
|
||||||
"Yes": "Já",
|
"Yes": "Já",
|
||||||
"No": "Nei",
|
"No": "Nei",
|
||||||
"Import and Export Data": "Innflutningur og Útflutningur Gagna",
|
"Import and Export Data": "Inn- og útflutningur gagna",
|
||||||
"Import": "Flytja inn",
|
"Import": "Flytja inn",
|
||||||
"Import Invidious data": "Flytja inn Invidious gögn",
|
"Import Invidious data": "Flytja inn Invidious JSON-gögn",
|
||||||
"Import YouTube subscriptions": "Flytja inn YouTube áskriftir",
|
"Import YouTube subscriptions": "Flytja inn YouTube CSV eða OPML-áskriftir",
|
||||||
"Import FreeTube subscriptions (.db)": "Flytja inn FreeTube áskriftir (.db)",
|
"Import FreeTube subscriptions (.db)": "Flytja inn FreeTube áskriftir (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Flytja inn NewPipe áskriftir (.json)",
|
"Import NewPipe subscriptions (.json)": "Flytja inn NewPipe áskriftir (.json)",
|
||||||
"Import NewPipe data (.zip)": "Flytja inn NewPipe gögn (.zip)",
|
"Import NewPipe data (.zip)": "Flytja inn NewPipe gögn (.zip)",
|
||||||
"Export": "Flytja út",
|
"Export": "Flytja út",
|
||||||
"Export subscriptions as OPML": "Flytja út áskriftir sem OPML",
|
"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 subscriptions as OPML (for NewPipe & FreeTube)": "Flytja út áskriftir sem OPML (fyrir NewPipe & FreeTube)",
|
||||||
"Export data as JSON": "Flytja út gögn sem JSON",
|
"Export data as JSON": "Flytja út Invidious-gögn sem JSON",
|
||||||
"Delete account?": "Eyða reikningi?",
|
"Delete account?": "Eyða reikningi?",
|
||||||
"History": "Saga",
|
"History": "Ferill",
|
||||||
"An alternative front-end to YouTube": "Önnur framhlið fyrir YouTube",
|
"An alternative front-end to YouTube": "Annað viðmót fyrir YouTube",
|
||||||
"JavaScript license information": "JavaScript leyfi upplýsingar",
|
"JavaScript license information": "Upplýsingar um notkunarleyfi JavaScript",
|
||||||
"source": "uppspretta",
|
"source": "uppruni",
|
||||||
"Log in": "Skrá inn",
|
"Log in": "Skrá inn",
|
||||||
"Log in/register": "Innskráning/nýskráning",
|
"Log in/register": "Innskráning/nýskráning",
|
||||||
"User ID": "Notandakenni",
|
"User ID": "Notandakenni",
|
||||||
|
@ -47,33 +47,33 @@
|
||||||
"Preferences": "Kjörstillingar",
|
"Preferences": "Kjörstillingar",
|
||||||
"preferences_category_player": "Kjörstillingar spilara",
|
"preferences_category_player": "Kjörstillingar spilara",
|
||||||
"preferences_video_loop_label": "Alltaf lykkja: ",
|
"preferences_video_loop_label": "Alltaf lykkja: ",
|
||||||
"preferences_autoplay_label": "Spila sjálfkrafa: ",
|
"preferences_autoplay_label": "Sjálfvirk spilun: ",
|
||||||
"preferences_continue_label": "Spila næst sjálfgefið: ",
|
"preferences_continue_label": "Spila næst sjálfgefið: ",
|
||||||
"preferences_continue_autoplay_label": "Spila næst sjálfkrafa: ",
|
"preferences_continue_autoplay_label": "Spila næsta myndskeið sjálfkrafa: ",
|
||||||
"preferences_listen_label": "Hlusta sjálfgefið: ",
|
"preferences_listen_label": "Hlusta sjálfgefið: ",
|
||||||
"preferences_local_label": "Proxy myndbönd? ",
|
"preferences_local_label": "Milliþjónn fyrir myndskeið: ",
|
||||||
"preferences_speed_label": "Sjálfgefinn hraði: ",
|
"preferences_speed_label": "Sjálfgefinn hraði: ",
|
||||||
"preferences_quality_label": "Æskilegt myndbands gæði: ",
|
"preferences_quality_label": "Æskileg gæði myndmerkis: ",
|
||||||
"preferences_volume_label": "Spilara hljóðstyrkur: ",
|
"preferences_volume_label": "Spilara hljóðstyrkur: ",
|
||||||
"preferences_comments_label": "Sjálfgefin ummæli: ",
|
"preferences_comments_label": "Sjálfgefin ummæli: ",
|
||||||
"youtube": "YouTube",
|
"youtube": "YouTube",
|
||||||
"reddit": "reddit",
|
"reddit": "Reddit",
|
||||||
"preferences_captions_label": "Sjálfgefin texti: ",
|
"preferences_captions_label": "Sjálfgefin texti: ",
|
||||||
"Fallback captions: ": "Varatextar: ",
|
"Fallback captions: ": "Varatextar: ",
|
||||||
"preferences_related_videos_label": "Sýna tengd myndbönd? ",
|
"preferences_related_videos_label": "Sýna tengd myndskeið? ",
|
||||||
"preferences_annotations_label": "Á að sýna glósur sjálfgefið? ",
|
"preferences_annotations_label": "Á að sýna glósur sjálfgefið? ",
|
||||||
"preferences_category_visual": "Sjónrænar stillingar",
|
"preferences_category_visual": "Sjónrænar stillingar",
|
||||||
"preferences_player_style_label": "Spilara stíl: ",
|
"preferences_player_style_label": "Stíll spilara: ",
|
||||||
"Dark mode: ": "Myrkur ham: ",
|
"Dark mode: ": "Dökkur hamur: ",
|
||||||
"preferences_dark_mode_label": "Þema: ",
|
"preferences_dark_mode_label": "Þema: ",
|
||||||
"dark": "dimmt",
|
"dark": "dökkt",
|
||||||
"light": "ljóst",
|
"light": "ljóst",
|
||||||
"preferences_thin_mode_label": "Þunnt ham: ",
|
"preferences_thin_mode_label": "Grannur hamur: ",
|
||||||
"preferences_category_subscription": "Áskriftarstillingar",
|
"preferences_category_subscription": "Áskriftarstillingar",
|
||||||
"preferences_annotations_subscribed_label": "Á að sýna glósur sjálfgefið fyrir áskriftarrásir? ",
|
"preferences_annotations_subscribed_label": "Á að sýna glósur sjálfgefið fyrir áskriftarrásir? ",
|
||||||
"Redirect homepage to feed: ": "Endurbeina heimasíðu að straumi: ",
|
"Redirect homepage to feed: ": "Endurbeina heimasíðu að streymi: ",
|
||||||
"preferences_max_results_label": "Fjöldi myndbanda sem sýndir eru í straumi: ",
|
"preferences_max_results_label": "Fjöldi myndskeiða sem sýnd eru í streymi: ",
|
||||||
"preferences_sort_label": "Raða myndbönd eftir: ",
|
"preferences_sort_label": "Raða myndskeiðum eftir: ",
|
||||||
"published": "birt",
|
"published": "birt",
|
||||||
"published - reverse": "birt - afturábak",
|
"published - reverse": "birt - afturábak",
|
||||||
"alphabetically": "í stafrófsröð",
|
"alphabetically": "í stafrófsröð",
|
||||||
|
@ -88,31 +88,31 @@
|
||||||
"`x` uploaded a video": "`x` hlóð upp myndband",
|
"`x` uploaded a video": "`x` hlóð upp myndband",
|
||||||
"`x` is live": "`x` er í beinni",
|
"`x` is live": "`x` er í beinni",
|
||||||
"preferences_category_data": "Gagnastillingar",
|
"preferences_category_data": "Gagnastillingar",
|
||||||
"Clear watch history": "Hreinsa áhorfssögu",
|
"Clear watch history": "Hreinsa áhorfsferil",
|
||||||
"Import/export data": "Flytja inn/út gögn",
|
"Import/export data": "Flytja inn/út gögn",
|
||||||
"Change password": "Breyta lykilorði",
|
"Change password": "Breyta lykilorði",
|
||||||
"Manage subscriptions": "Stjórna áskriftum",
|
"Manage subscriptions": "Sýsla með áskriftir",
|
||||||
"Manage tokens": "Stjórna tákn",
|
"Manage tokens": "Sýsla með teikn",
|
||||||
"Watch history": "Áhorfssögu",
|
"Watch history": "Áhorfsferill",
|
||||||
"Delete account": "Eyða reikningi",
|
"Delete account": "Eyða reikningi",
|
||||||
"preferences_category_admin": "Kjörstillingar stjórnanda",
|
"preferences_category_admin": "Kjörstillingar stjórnanda",
|
||||||
"preferences_default_home_label": "Sjálfgefin heimasíða: ",
|
"preferences_default_home_label": "Sjálfgefin heimasíða: ",
|
||||||
"preferences_feed_menu_label": "Straum valmynd: ",
|
"preferences_feed_menu_label": "Streymisvalmynd: ",
|
||||||
"Top enabled: ": "Toppur virkur? ",
|
"Top enabled: ": "Vinsælast virkt? ",
|
||||||
"CAPTCHA enabled: ": "CAPTCHA virk? ",
|
"CAPTCHA enabled: ": "CAPTCHA virk? ",
|
||||||
"Login enabled: ": "Innskráning virk? ",
|
"Login enabled: ": "Innskráning virk? ",
|
||||||
"Registration enabled: ": "Nýskráning virkjuð? ",
|
"Registration enabled: ": "Nýskráning virkjuð? ",
|
||||||
"Report statistics: ": "Skrá talnagögn? ",
|
"Report statistics: ": "Skrá tölfræði? ",
|
||||||
"Save preferences": "Vista stillingar",
|
"Save preferences": "Vista stillingar",
|
||||||
"Subscription manager": "Áskriftarstjóri",
|
"Subscription manager": "Áskriftarstjóri",
|
||||||
"Token manager": "Táknstjóri",
|
"Token manager": "Teiknastjórnun",
|
||||||
"Token": "Tákn",
|
"Token": "Teikn",
|
||||||
"Import/export": "Flytja inn/út",
|
"Import/export": "Flytja inn/út",
|
||||||
"unsubscribe": "afskrá",
|
"unsubscribe": "afskrá",
|
||||||
"revoke": "afturkalla",
|
"revoke": "afturkalla",
|
||||||
"Subscriptions": "Áskriftir",
|
"Subscriptions": "Áskriftir",
|
||||||
"search": "leita",
|
"search": "leita",
|
||||||
"Log out": "Útskrá",
|
"Log out": "Skrá út",
|
||||||
"Source available here.": "Frumkóði aðgengilegur hér.",
|
"Source available here.": "Frumkóði aðgengilegur hér.",
|
||||||
"View JavaScript license information.": "Skoða JavaScript leyfisupplýsingar.",
|
"View JavaScript license information.": "Skoða JavaScript leyfisupplýsingar.",
|
||||||
"View privacy policy.": "Skoða meðferð persónuupplýsinga.",
|
"View privacy policy.": "Skoða meðferð persónuupplýsinga.",
|
||||||
|
@ -122,13 +122,13 @@
|
||||||
"Private": "Einka",
|
"Private": "Einka",
|
||||||
"View all playlists": "Skoða alla spilunarlista",
|
"View all playlists": "Skoða alla spilunarlista",
|
||||||
"Updated `x` ago": "Uppfært `x` síðann",
|
"Updated `x` ago": "Uppfært `x` síðann",
|
||||||
"Delete playlist `x`?": "Eiða spilunarlista `x`?",
|
"Delete playlist `x`?": "Eyða spilunarlista `x`?",
|
||||||
"Delete playlist": "Eiða spilunarlista",
|
"Delete playlist": "Eyða spilunarlista",
|
||||||
"Create playlist": "Búa til spilunarlista",
|
"Create playlist": "Búa til spilunarlista",
|
||||||
"Title": "Titill",
|
"Title": "Titill",
|
||||||
"Playlist privacy": "Spilunarlista opinberri",
|
"Playlist privacy": "Friðhelgi spilunarlista",
|
||||||
"Editing playlist `x`": "Að breyta spilunarlista `x`",
|
"Editing playlist `x`": "Breyti spilunarlista `x`",
|
||||||
"Watch on YouTube": "Horfa á YouTube",
|
"Watch on YouTube": "Skoða á YouTube",
|
||||||
"Hide annotations": "Fela glósur",
|
"Hide annotations": "Fela glósur",
|
||||||
"Show annotations": "Sýna glósur",
|
"Show annotations": "Sýna glósur",
|
||||||
"Genre: ": "Tegund: ",
|
"Genre: ": "Tegund: ",
|
||||||
|
@ -160,26 +160,26 @@
|
||||||
"Wrong username or password": "Rangt notandanafn eða lykilorð",
|
"Wrong username or password": "Rangt notandanafn eða lykilorð",
|
||||||
"Password cannot be empty": "Lykilorð má ekki vera autt",
|
"Password cannot be empty": "Lykilorð má ekki vera autt",
|
||||||
"Password cannot be longer than 55 characters": "Lykilorð má ekki vera lengra en 55 stafir",
|
"Password cannot be longer than 55 characters": "Lykilorð má ekki vera lengra en 55 stafir",
|
||||||
"Please log in": "Vinsamlegast skráðu þig inn",
|
"Please log in": "Skráðu þig inn",
|
||||||
"Invidious Private Feed for `x`": "Invidious Persónulegur Straumur fyrir `x`",
|
"Invidious Private Feed for `x`": "Persónulegt Invidious-streymi fyrir `x`",
|
||||||
"channel:`x`": "rás:`x`",
|
"channel:`x`": "rás:`x`",
|
||||||
"Deleted or invalid channel": "Eytt eða ógild rás",
|
"Deleted or invalid channel": "Eytt eða ógild rás",
|
||||||
"This channel does not exist.": "Þessi rás er ekki til.",
|
"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 get channel info.": "Ekki tókst að fá upplýsingar um rásina.",
|
||||||
"Could not fetch comments": "Ekki tókst að sækja ummæli",
|
"Could not fetch comments": "Ekki tókst að sækja ummæli",
|
||||||
"`x` ago": "`x` síðan",
|
"`x` ago": "`x` síðan",
|
||||||
"Load more": "Hlaða meira",
|
"Load more": "Hlaða meira",
|
||||||
"Could not create mix.": "Ekki tókst að búa til blöndu.",
|
"Could not create mix.": "Ekki tókst að búa til blöndu.",
|
||||||
"Empty playlist": "Tómur spilunarlisti",
|
"Empty playlist": "Tómur spilunarlisti",
|
||||||
"Not a playlist.": "Ekki spilunarlisti.",
|
"Not a playlist.": "Er ekki spilunarlisti.",
|
||||||
"Playlist does not exist.": "Spilunarlisti er ekki til.",
|
"Playlist does not exist.": "Spilunarlisti er ekki til.",
|
||||||
"Could not pull trending pages.": "Ekki tókst að draga vinsælar síður.",
|
"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 \"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",
|
"Hidden field \"token\" is a required field": "Falinn reitur \"teikn\" er nauðsynlegur reitur",
|
||||||
"Erroneous challenge": "Röng áskorun",
|
"Erroneous challenge": "Röng áskorun",
|
||||||
"Erroneous token": "Rangt tákn",
|
"Erroneous token": "Rangt teikn",
|
||||||
"No such user": "Enginn slíkur notandi",
|
"No such user": "Enginn slíkur notandi",
|
||||||
"Token is expired, please try again": "Tákn er útrunnið, vinsamlegast reyndu aftur",
|
"Token is expired, please try again": "Teiknið er útrunnið, reyndu aftur",
|
||||||
"English": "Enska",
|
"English": "Enska",
|
||||||
"English (auto-generated)": "Enska (sjálfkrafa)",
|
"English (auto-generated)": "Enska (sjálfkrafa)",
|
||||||
"Afrikaans": "Afríkanska",
|
"Afrikaans": "Afríkanska",
|
||||||
|
@ -267,14 +267,14 @@
|
||||||
"Somali": "Sómalska",
|
"Somali": "Sómalska",
|
||||||
"Southern Sotho": "Suður Sótó",
|
"Southern Sotho": "Suður Sótó",
|
||||||
"Spanish": "Spænska",
|
"Spanish": "Spænska",
|
||||||
"Spanish (Latin America)": "Spænska (Rómönsku Ameríka)",
|
"Spanish (Latin America)": "Spænska (Rómanska Ameríka)",
|
||||||
"Sundanese": "Sundaneska",
|
"Sundanese": "Sundaneska",
|
||||||
"Swahili": "Svahílí",
|
"Swahili": "Svahílí",
|
||||||
"Swedish": "Sænska",
|
"Swedish": "Sænska",
|
||||||
"Tajik": "Tadsikíska",
|
"Tajik": "Tadsikíska",
|
||||||
"Tamil": "Tamílska",
|
"Tamil": "Tamílska",
|
||||||
"Telugu": "Telúgú",
|
"Telugu": "Telúgú",
|
||||||
"Thai": "Taílenska",
|
"Thai": "Tælenska",
|
||||||
"Turkish": "Tyrkneska",
|
"Turkish": "Tyrkneska",
|
||||||
"Ukrainian": "Úkraníska",
|
"Ukrainian": "Úkraníska",
|
||||||
"Urdu": "Úrdú",
|
"Urdu": "Úrdú",
|
||||||
|
@ -286,9 +286,9 @@
|
||||||
"Yiddish": "Jiddíska",
|
"Yiddish": "Jiddíska",
|
||||||
"Yoruba": "Jórúba",
|
"Yoruba": "Jórúba",
|
||||||
"Zulu": "Zúlú",
|
"Zulu": "Zúlú",
|
||||||
"Fallback comments: ": "Vara ummæli: ",
|
"Fallback comments: ": "Ummæli til vara: ",
|
||||||
"Popular": "Vinsælt",
|
"Popular": "Vinsælt",
|
||||||
"Top": "Topp",
|
"Top": "Vinsælast",
|
||||||
"About": "Um",
|
"About": "Um",
|
||||||
"Rating: ": "Einkunn: ",
|
"Rating: ": "Einkunn: ",
|
||||||
"preferences_locale_label": "Tungumál: ",
|
"preferences_locale_label": "Tungumál: ",
|
||||||
|
@ -307,9 +307,194 @@
|
||||||
"`x` marked it with a ❤": "`x` merkti það með ❤",
|
"`x` marked it with a ❤": "`x` merkti það með ❤",
|
||||||
"Audio mode": "Hljóð ham",
|
"Audio mode": "Hljóð ham",
|
||||||
"Video mode": "Myndband ham",
|
"Video mode": "Myndband ham",
|
||||||
"channel_tab_videos_label": "Myndbönd",
|
"channel_tab_videos_label": "Myndskeið",
|
||||||
"Playlists": "Spilunarlistar",
|
"Playlists": "Spilunarlistar",
|
||||||
"channel_tab_community_label": "Samfélag",
|
"channel_tab_community_label": "Samfélag",
|
||||||
"Current version: ": "Núverandi útgáfa: ",
|
"Current version: ": "Núverandi útgáfa: ",
|
||||||
"preferences_watch_history_label": "Virkja áhorfssögu: "
|
"preferences_watch_history_label": "Virkja áhorfsferil: ",
|
||||||
|
"Chinese (China)": "Kínverska (Kína)",
|
||||||
|
"Turkish (auto-generated)": "Tyrkneska (sjálfvirkt útbúið)",
|
||||||
|
"Search": "Leita",
|
||||||
|
"preferences_save_player_pos_label": "Vista staðsetningu í afspilun: ",
|
||||||
|
"Popular enabled: ": "Vinsælt virkjað: ",
|
||||||
|
"search_filters_features_option_purchased": "Keypt",
|
||||||
|
"Standard YouTube license": "Staðlað YouTube-notkunarleyfi",
|
||||||
|
"French (auto-generated)": "Franska (sjálfvirkt útbúið)",
|
||||||
|
"Spanish (Spain)": "Spænska (Spánn)",
|
||||||
|
"search_filters_title": "Síur",
|
||||||
|
"search_filters_date_label": "Dags. innsendingar",
|
||||||
|
"search_filters_features_option_four_k": "4K",
|
||||||
|
"search_filters_features_option_hd": "HD",
|
||||||
|
"crash_page_read_the_faq": "lesið <a href=\"`x`\">Algengar spurningar (FAQ)</a>",
|
||||||
|
"Add to playlist": "Bæta á spilunarlista",
|
||||||
|
"Add to playlist: ": "Bæta á spilunarlista: ",
|
||||||
|
"Answer": "Svar",
|
||||||
|
"Search for videos": "Leita að myndskeiðum",
|
||||||
|
"generic_channels_count": "{{count}} rás",
|
||||||
|
"generic_channels_count_plural": "{{count}} rásir",
|
||||||
|
"generic_videos_count": "{{count}} myndskeið",
|
||||||
|
"generic_videos_count_plural": "{{count}} myndskeið",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Kerfisstjórinn hefur gert Vinsælt-streymið óvirkt.",
|
||||||
|
"generic_playlists_count": "{{count}} spilunarlisti",
|
||||||
|
"generic_playlists_count_plural": "{{count}} spilunarlistar",
|
||||||
|
"generic_subscribers_count": "{{count}} áskrifandi",
|
||||||
|
"generic_subscribers_count_plural": "{{count}} áskrifendur",
|
||||||
|
"generic_subscriptions_count": "{{count}} áskrift",
|
||||||
|
"generic_subscriptions_count_plural": "{{count}} áskriftir",
|
||||||
|
"generic_button_delete": "Eyða",
|
||||||
|
"Import YouTube watch history (.json)": "Flytja inn YouTube áhorfsferil (.json)",
|
||||||
|
"preferences_vr_mode_label": "Gagnvirk 360 gráðu myndskeið (krefst WebGL): ",
|
||||||
|
"preferences_quality_dash_option_auto": "Sjálfvirkt",
|
||||||
|
"preferences_quality_dash_option_best": "Best",
|
||||||
|
"preferences_quality_dash_option_worst": "Verst",
|
||||||
|
"preferences_quality_dash_label": "Æskileg DASH-gæði myndmerkis: ",
|
||||||
|
"preferences_extend_desc_label": "Sjálfvirkt útvíkka lýsingu á myndskeiði: ",
|
||||||
|
"preferences_region_label": "Land efnis: ",
|
||||||
|
"preferences_show_nick_label": "Birta gælunafn efst: ",
|
||||||
|
"tokens_count": "{{count}} teikn",
|
||||||
|
"tokens_count_plural": "{{count}} teikn",
|
||||||
|
"subscriptions_unseen_notifs_count": "{{count}} óskoðuð tilkynning",
|
||||||
|
"subscriptions_unseen_notifs_count_plural": "{{count}} óskoðaðar tilkynningar",
|
||||||
|
"Released under the AGPLv3 on Github.": "Gefið út með AGPLv3-notkunarleyfi á GitHub.",
|
||||||
|
"Music in this video": "Tónlist í þessu myndskeiði",
|
||||||
|
"Artist: ": "Flytjandi: ",
|
||||||
|
"Album: ": "Hljómplata: ",
|
||||||
|
"comments_view_x_replies": "Skoða {{count}} svar",
|
||||||
|
"comments_view_x_replies_plural": "Skoða {{count}} svör",
|
||||||
|
"comments_points_count": "{{count}} punktur",
|
||||||
|
"comments_points_count_plural": "{{count}} punktar",
|
||||||
|
"Cantonese (Hong Kong)": "Kantónska (Hong Kong)",
|
||||||
|
"Chinese": "Kínverska",
|
||||||
|
"Chinese (Hong Kong)": "Kínverska (Hong Kong)",
|
||||||
|
"Chinese (Taiwan)": "Kínverska (Taívan)",
|
||||||
|
"Japanese (auto-generated)": "Japanska (sjálfvirkt útbúið)",
|
||||||
|
"generic_count_minutes": "{{count}} mínúta",
|
||||||
|
"generic_count_minutes_plural": "{{count}} mínútur",
|
||||||
|
"generic_count_seconds": "{{count}} sekúnda",
|
||||||
|
"generic_count_seconds_plural": "{{count}} sekúndur",
|
||||||
|
"search_filters_date_option_hour": "Síðustu klukkustund",
|
||||||
|
"search_filters_apply_button": "Virkja valdar síur",
|
||||||
|
"next_steps_error_message_go_to_youtube": "Fara á YouTube",
|
||||||
|
"footer_original_source_code": "Upprunalegur grunnkóði",
|
||||||
|
"videoinfo_started_streaming_x_ago": "Byrjaði streymi fyrir `x` síðan",
|
||||||
|
"next_steps_error_message": "Á eftir þessu ættirðu að prófa: ",
|
||||||
|
"videoinfo_invidious_embed_link": "Ívefja tengil",
|
||||||
|
"download_subtitles": "Skjátextar - `x` (.vtt)",
|
||||||
|
"user_created_playlists": "`x` útbjó spilunarlista",
|
||||||
|
"user_saved_playlists": "`x` vistaði spilunarlista",
|
||||||
|
"Video unavailable": "Myndskeið ekki tiltækt",
|
||||||
|
"videoinfo_watch_on_youTube": "Skoða á YouTube",
|
||||||
|
"crash_page_you_found_a_bug": "Það lítur út eins og þú hafir fundið galla í Invidious!",
|
||||||
|
"crash_page_before_reporting": "Áður en þú tilkynnir villu, gakktu úr skugga um að þú hafir:",
|
||||||
|
"crash_page_switch_instance": "reynt að <a href=\"`x`\">nota annað tilvik</a>",
|
||||||
|
"crash_page_report_issue": "Ef ekkert af ofantöldu hjálpaði, ættirðu að <a href=\"`x`\">opna nýja verkbeiðni (issue) á GitHub</a> (helst á ensku) og láta fylgja eftirfarandi texta í skilaboðunum þínum (alls EKKI þýða þennan texta):",
|
||||||
|
"channel_tab_shorts_label": "Stuttmyndir",
|
||||||
|
"carousel_slide": "Skyggna {{current}} af {{total}}",
|
||||||
|
"carousel_go_to": "Fara á skyggnu `x`",
|
||||||
|
"channel_tab_streams_label": "Bein streymi",
|
||||||
|
"channel_tab_playlists_label": "Spilunarlistar",
|
||||||
|
"toggle_theme": "Víxla þema",
|
||||||
|
"carousel_skip": "Sleppa hringekjunni",
|
||||||
|
"preferences_quality_option_medium": "Miðlungs",
|
||||||
|
"search_message_use_another_instance": "Þú getur líka <a href=\"`x`\">leitað á öðrum netþjóni</a>.",
|
||||||
|
"footer_source_code": "Grunnkóði",
|
||||||
|
"English (United Kingdom)": "Enska (Bretland)",
|
||||||
|
"English (United States)": "Enska (Bandarísk)",
|
||||||
|
"Vietnamese (auto-generated)": "Víetnamska (sjálfvirkt útbúið)",
|
||||||
|
"generic_count_months": "{{count}} mánuður",
|
||||||
|
"generic_count_months_plural": "{{count}} mánuðir",
|
||||||
|
"search_filters_sort_option_rating": "Einkunn",
|
||||||
|
"videoinfo_youTube_embed_link": "Ívefja",
|
||||||
|
"error_video_not_in_playlist": "Umbeðið myndskeið fyrirfinnst ekki í þessum spilunarlista. <a href=\"`x`\">Smelltu hér til að fara á heimasíðu spilunarlistans.</a>",
|
||||||
|
"generic_views_count": "{{count}} áhorf",
|
||||||
|
"generic_views_count_plural": "{{count}} áhorf",
|
||||||
|
"playlist_button_add_items": "Bæta við myndskeiðum",
|
||||||
|
"Show more": "Sýna meira",
|
||||||
|
"Show less": "Sýna minna",
|
||||||
|
"Song: ": "Lag: ",
|
||||||
|
"channel_tab_podcasts_label": "Hlaðvörp (podcasts)",
|
||||||
|
"channel_tab_releases_label": "Útgáfur",
|
||||||
|
"Download is disabled": "Niðurhal er óvirkt",
|
||||||
|
"search_filters_features_option_location": "Staðsetning",
|
||||||
|
"preferences_quality_dash_option_720p": "720p",
|
||||||
|
"Switch Invidious Instance": "Skipta um Invidious-tilvik",
|
||||||
|
"search_message_no_results": "Engar niðurstöður fundust.",
|
||||||
|
"search_message_change_filters_or_query": "Reyndu að víkka leitarsviðið og/eða breyta síunum.",
|
||||||
|
"Dutch (auto-generated)": "Hollenska (sjálfvirkt útbúið)",
|
||||||
|
"German (auto-generated)": "Þýska (sjálfvirkt útbúið)",
|
||||||
|
"Indonesian (auto-generated)": "Indónesíska (sjálfvirkt útbúið)",
|
||||||
|
"Interlingue": "Interlingue",
|
||||||
|
"Italian (auto-generated)": "Ítalska (sjálfvirkt útbúið)",
|
||||||
|
"Russian (auto-generated)": "Rússneska (sjálfvirkt útbúið)",
|
||||||
|
"Spanish (auto-generated)": "Spænska (sjálfvirkt útbúið)",
|
||||||
|
"Spanish (Mexico)": "Spænska (Mexíkó)",
|
||||||
|
"generic_count_hours": "{{count}} klukkustund",
|
||||||
|
"generic_count_hours_plural": "{{count}} klukkustundir",
|
||||||
|
"generic_count_years": "{{count}} ár",
|
||||||
|
"generic_count_years_plural": "{{count}} ár",
|
||||||
|
"generic_count_weeks": "{{count}} vika",
|
||||||
|
"generic_count_weeks_plural": "{{count}} vikur",
|
||||||
|
"search_filters_date_option_none": "Hvaða dagsetning sem er",
|
||||||
|
"Channel Sponsor": "Styrktaraðili rásar",
|
||||||
|
"search_filters_date_option_week": "Í þessari viku",
|
||||||
|
"search_filters_date_option_month": "Í þessum mánuði",
|
||||||
|
"search_filters_date_option_year": "Á þessu ári",
|
||||||
|
"search_filters_type_option_playlist": "Spilunarlisti",
|
||||||
|
"search_filters_type_option_show": "Þáttur",
|
||||||
|
"search_filters_duration_label": "Tímalengd",
|
||||||
|
"search_filters_duration_option_long": "Langt (> 20 mínútur)",
|
||||||
|
"search_filters_features_option_live": "Beint",
|
||||||
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
|
"search_filters_features_option_vr180": "VR180",
|
||||||
|
"search_filters_features_option_three_d": "3D",
|
||||||
|
"search_filters_features_option_hdr": "HDR",
|
||||||
|
"search_filters_sort_label": "Raða eftir",
|
||||||
|
"search_filters_sort_option_relevance": "Samsvörun",
|
||||||
|
"footer_donate_page": "Styrkja",
|
||||||
|
"footer_modfied_source_code": "Breyttur grunnkóði",
|
||||||
|
"crash_page_refresh": "reynt að <a href=\"`x`\">endurlesa síðuna</a>",
|
||||||
|
"crash_page_search_issue": "leitað að <a href=\"`x`\">fyrirliggjandi villum á GitHub</a>",
|
||||||
|
"none": "ekkert",
|
||||||
|
"adminprefs_modified_source_code_url_label": "Slóð á gagnasafn með breyttum grunnkóða",
|
||||||
|
"preferences_quality_option_hd720": "HD720",
|
||||||
|
"preferences_quality_option_small": "Lítið",
|
||||||
|
"preferences_category_misc": "Ýmsar kjörstillingar",
|
||||||
|
"preferences_automatic_instance_redirect_label": "Sjálfvirk endurbeining tilvika (farið til vara á redirect.invidious.io): ",
|
||||||
|
"Portuguese (auto-generated)": "Portúgalska (sjálfvirkt útbúið)",
|
||||||
|
"Portuguese (Brazil)": "Portúgalska (Brasilía)",
|
||||||
|
"generic_button_edit": "Breyta",
|
||||||
|
"generic_button_save": "Vista",
|
||||||
|
"generic_button_cancel": "Hætta við",
|
||||||
|
"generic_button_rss": "RSS",
|
||||||
|
"preferences_quality_dash_option_4320p": "4320p",
|
||||||
|
"preferences_quality_dash_option_2160p": "2160p",
|
||||||
|
"preferences_quality_dash_option_1440p": "1440p",
|
||||||
|
"preferences_quality_dash_option_1080p": "1080p",
|
||||||
|
"preferences_quality_dash_option_480p": "480p",
|
||||||
|
"preferences_quality_dash_option_360p": "360p",
|
||||||
|
"preferences_quality_dash_option_240p": "240p",
|
||||||
|
"preferences_quality_dash_option_144p": "144p",
|
||||||
|
"invidious": "Invidious",
|
||||||
|
"Korean (auto-generated)": "Kóreska (sjálfvirkt útbúið)",
|
||||||
|
"generic_count_days": "{{count}} dagur",
|
||||||
|
"generic_count_days_plural": "{{count}} dagar",
|
||||||
|
"search_filters_date_option_today": "Í dag",
|
||||||
|
"search_filters_type_label": "Tegund",
|
||||||
|
"search_filters_type_option_all": "Hvaða tegund sem er",
|
||||||
|
"search_filters_type_option_video": "Myndskeið",
|
||||||
|
"search_filters_type_option_channel": "Rás",
|
||||||
|
"search_filters_type_option_movie": "Kvikmynd",
|
||||||
|
"search_filters_duration_option_none": "Hvaða lengd sem er",
|
||||||
|
"search_filters_duration_option_short": "Stutt (< 4 mínútur)",
|
||||||
|
"search_filters_duration_option_medium": "Miðlungs (4 - 20 mínútur)",
|
||||||
|
"search_filters_features_label": "Eiginleikar",
|
||||||
|
"search_filters_features_option_subtitles": "Skjátextar/CC",
|
||||||
|
"search_filters_features_option_c_commons": "Creative Commons",
|
||||||
|
"search_filters_sort_option_date": "Dags. innsendingar",
|
||||||
|
"search_filters_sort_option_views": "Fjöldi áhorfa",
|
||||||
|
"next_steps_error_message_refresh": "Endurlesa",
|
||||||
|
"footer_documentation": "Leiðbeiningar",
|
||||||
|
"channel_tab_channels_label": "Rásir",
|
||||||
|
"Import YouTube playlist (.csv)": "Flytja inn YouTube spilunarlista (.csv)",
|
||||||
|
"preferences_quality_option_dash": "DASH (aðlaganleg gæði)"
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@
|
||||||
"Import and Export Data": "Importazione ed esportazione dati",
|
"Import and Export Data": "Importazione ed esportazione dati",
|
||||||
"Import": "Importa",
|
"Import": "Importa",
|
||||||
"Import Invidious data": "Importa dati Invidious in formato JSON",
|
"Import Invidious data": "Importa dati Invidious in formato JSON",
|
||||||
"Import YouTube subscriptions": "Importa le iscrizioni da YouTube/OPML",
|
"Import YouTube subscriptions": "Importa iscrizioni in CSV o OPML di YouTube",
|
||||||
"Import FreeTube subscriptions (.db)": "Importa le iscrizioni da FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importa le iscrizioni da FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importa le iscrizioni da NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importa le iscrizioni da NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importa i dati di NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importa i dati di NewPipe (.zip)",
|
||||||
|
@ -449,7 +449,7 @@
|
||||||
"Portuguese (Brazil)": "Portoghese (Brasile)",
|
"Portuguese (Brazil)": "Portoghese (Brasile)",
|
||||||
"preferences_watch_history_label": "Attiva cronologia di riproduzione: ",
|
"preferences_watch_history_label": "Attiva cronologia di riproduzione: ",
|
||||||
"French (auto-generated)": "Francese (generati automaticamente)",
|
"French (auto-generated)": "Francese (generati automaticamente)",
|
||||||
"search_message_use_another_instance": " Puoi anche <a href=\"`x`\">cercare in un'altra istanza</a>.",
|
"search_message_use_another_instance": "Puoi anche <a href=\"`x`\">cercare in un'altra istanza</a>.",
|
||||||
"search_message_no_results": "Nessun risultato trovato.",
|
"search_message_no_results": "Nessun risultato trovato.",
|
||||||
"search_message_change_filters_or_query": "Prova ad ampliare la ricerca e/o modificare i filtri.",
|
"search_message_change_filters_or_query": "Prova ad ampliare la ricerca e/o modificare i filtri.",
|
||||||
"English (United States)": "Inglese (Stati Uniti)",
|
"English (United States)": "Inglese (Stati Uniti)",
|
||||||
|
@ -504,5 +504,14 @@
|
||||||
"generic_channels_count_0": "{{count}} canale",
|
"generic_channels_count_0": "{{count}} canale",
|
||||||
"generic_channels_count_1": "{{count}} canali",
|
"generic_channels_count_1": "{{count}} canali",
|
||||||
"generic_channels_count_2": "{{count}} canali",
|
"generic_channels_count_2": "{{count}} canali",
|
||||||
"Import YouTube watch history (.json)": "Importa la cronologia delle visualizzazioni di YouTube (.json)"
|
"Import YouTube watch history (.json)": "Importa la cronologia delle visualizzazioni di YouTube (.json)",
|
||||||
|
"Answer": "Risposta",
|
||||||
|
"toggle_theme": "Cambia Tema",
|
||||||
|
"Add to playlist": "Aggiungi alla playlist",
|
||||||
|
"Add to playlist: ": "Aggiungi alla playlist ",
|
||||||
|
"Search for videos": "Cerca dei video",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "La sezione dei contenuti popolari è stata disabilitata dall'amministratore.",
|
||||||
|
"carousel_slide": "Fotogramma {{current}} di {{total}}",
|
||||||
|
"carousel_skip": "Salta la galleria",
|
||||||
|
"carousel_go_to": "Vai al fotogramma `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -363,7 +363,7 @@
|
||||||
"search_filters_features_option_location": "場所",
|
"search_filters_features_option_location": "場所",
|
||||||
"search_filters_features_option_hdr": "HDR",
|
"search_filters_features_option_hdr": "HDR",
|
||||||
"Current version: ": "現在のバージョン: ",
|
"Current version: ": "現在のバージョン: ",
|
||||||
"next_steps_error_message": "以下をお試してください: ",
|
"next_steps_error_message": "以下をお試しください: ",
|
||||||
"next_steps_error_message_refresh": "再読み込み",
|
"next_steps_error_message_refresh": "再読み込み",
|
||||||
"next_steps_error_message_go_to_youtube": "YouTubeを開く",
|
"next_steps_error_message_go_to_youtube": "YouTubeを開く",
|
||||||
"search_filters_duration_option_short": "4分未満",
|
"search_filters_duration_option_short": "4分未満",
|
||||||
|
@ -396,7 +396,7 @@
|
||||||
"download_subtitles": "字幕 - `x` (.vtt)",
|
"download_subtitles": "字幕 - `x` (.vtt)",
|
||||||
"search_filters_features_option_purchased": "購入済み",
|
"search_filters_features_option_purchased": "購入済み",
|
||||||
"preferences_quality_option_dash": "DASH (適応的画質)",
|
"preferences_quality_option_dash": "DASH (適応的画質)",
|
||||||
"preferences_quality_dash_option_worst": "最悪",
|
"preferences_quality_dash_option_worst": "最低",
|
||||||
"preferences_quality_dash_option_best": "最高",
|
"preferences_quality_dash_option_best": "最高",
|
||||||
"videoinfo_started_streaming_x_ago": "`x`前に配信を開始",
|
"videoinfo_started_streaming_x_ago": "`x`前に配信を開始",
|
||||||
"videoinfo_watch_on_youTube": "YouTubeで視聴",
|
"videoinfo_watch_on_youTube": "YouTubeで視聴",
|
||||||
|
@ -434,7 +434,7 @@
|
||||||
"crash_page_switch_instance": "<a href=\"`x`\">別のインスタンスを使用</a>を試す",
|
"crash_page_switch_instance": "<a href=\"`x`\">別のインスタンスを使用</a>を試す",
|
||||||
"crash_page_read_the_faq": "<a href=\"`x`\">よくある質問 (FAQ)</a> を読む",
|
"crash_page_read_the_faq": "<a href=\"`x`\">よくある質問 (FAQ)</a> を読む",
|
||||||
"Popular enabled: ": "人気動画を有効化 ",
|
"Popular enabled: ": "人気動画を有効化 ",
|
||||||
"search_message_use_another_instance": " <a href=\"`x`\">別のインスタンス上での検索</a>も可能です。",
|
"search_message_use_another_instance": "<a href=\"`x`\">別のインスタンス上での検索</a>も可能です。",
|
||||||
"search_filters_apply_button": "選択したフィルターを適用",
|
"search_filters_apply_button": "選択したフィルターを適用",
|
||||||
"user_saved_playlists": "`x`個の保存済みの再生リスト",
|
"user_saved_playlists": "`x`個の保存済みの再生リスト",
|
||||||
"crash_page_you_found_a_bug": "Invidious のバグのようです!",
|
"crash_page_you_found_a_bug": "Invidious のバグのようです!",
|
||||||
|
@ -470,5 +470,14 @@
|
||||||
"generic_button_rss": "RSS",
|
"generic_button_rss": "RSS",
|
||||||
"playlist_button_add_items": "動画を追加",
|
"playlist_button_add_items": "動画を追加",
|
||||||
"generic_channels_count_0": "{{count}}個のチャンネル",
|
"generic_channels_count_0": "{{count}}個のチャンネル",
|
||||||
"Import YouTube watch history (.json)": "YouTube 視聴履歴をインポート (.json)"
|
"Import YouTube watch history (.json)": "YouTube 視聴履歴をインポート (.json)",
|
||||||
|
"Add to playlist": "再生リストに追加",
|
||||||
|
"Add to playlist: ": "再生リストに追加: ",
|
||||||
|
"Answer": "回答",
|
||||||
|
"Search for videos": "動画を検索",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "人気の動画のページは管理者によって無効にされています。",
|
||||||
|
"carousel_go_to": "スライド`x`を表示",
|
||||||
|
"carousel_slide": "スライド{{current}} / 全{{total}}個中",
|
||||||
|
"carousel_skip": "画像のスライド表示をスキップ",
|
||||||
|
"toggle_theme": "テーマの切り替え"
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,8 +12,8 @@
|
||||||
"Dark mode: ": "다크 모드: ",
|
"Dark mode: ": "다크 모드: ",
|
||||||
"preferences_player_style_label": "플레이어 스타일: ",
|
"preferences_player_style_label": "플레이어 스타일: ",
|
||||||
"preferences_category_visual": "환경 설정",
|
"preferences_category_visual": "환경 설정",
|
||||||
"preferences_vr_mode_label": "VR 영상 활성화(WebGL 필요): ",
|
"preferences_vr_mode_label": "360도 영상 활성화 (WebGL 필요): ",
|
||||||
"preferences_extend_desc_label": "자동으로 비디오 설명을 확장: ",
|
"preferences_extend_desc_label": "자동으로 비디오 설명 펼치기: ",
|
||||||
"preferences_annotations_label": "기본으로 주석 표시: ",
|
"preferences_annotations_label": "기본으로 주석 표시: ",
|
||||||
"preferences_related_videos_label": "관련 동영상 보기: ",
|
"preferences_related_videos_label": "관련 동영상 보기: ",
|
||||||
"Fallback captions: ": "대체 자막: ",
|
"Fallback captions: ": "대체 자막: ",
|
||||||
|
@ -48,7 +48,7 @@
|
||||||
"An alternative front-end to YouTube": "유튜브의 프론트엔드 대안",
|
"An alternative front-end to YouTube": "유튜브의 프론트엔드 대안",
|
||||||
"History": "시청 기록",
|
"History": "시청 기록",
|
||||||
"Delete account?": "계정을 삭제 하시겠습니까?",
|
"Delete account?": "계정을 삭제 하시겠습니까?",
|
||||||
"Export data as JSON": "JSON으로 데이터 내보내기",
|
"Export data as JSON": "인비디어스 데이터 내보내기 (.json)",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "OPML로 구독 내보내기 (뉴파이프 및 프리튜브)",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "OPML로 구독 내보내기 (뉴파이프 및 프리튜브)",
|
||||||
"Export subscriptions as OPML": "OPML로 구독 내보내기",
|
"Export subscriptions as OPML": "OPML로 구독 내보내기",
|
||||||
"Export": "내보내기",
|
"Export": "내보내기",
|
||||||
|
@ -65,13 +65,13 @@
|
||||||
"Authorize token?": "토큰을 승인하시겠습니까?",
|
"Authorize token?": "토큰을 승인하시겠습니까?",
|
||||||
"New passwords must match": "새 비밀번호는 일치해야 합니다",
|
"New passwords must match": "새 비밀번호는 일치해야 합니다",
|
||||||
"New password": "새 비밀번호",
|
"New password": "새 비밀번호",
|
||||||
"Clear watch history?": "재생 기록을 삭제 하시겠습니까?",
|
"Clear watch history?": "시청 기록을 지우시겠습니까?",
|
||||||
"Previous page": "이전 페이지",
|
"Previous page": "이전 페이지",
|
||||||
"Next page": "다음 페이지",
|
"Next page": "다음 페이지",
|
||||||
"last": "마지막",
|
"last": "마지막",
|
||||||
"Shared `x` ago": "`x` 전",
|
"Shared `x` ago": "`x` 전",
|
||||||
"popular": "인기",
|
"popular": "인기",
|
||||||
"oldest": "오래된순",
|
"oldest": "과거순",
|
||||||
"newest": "최신순",
|
"newest": "최신순",
|
||||||
"View playlist on YouTube": "유튜브에서 재생목록 보기",
|
"View playlist on YouTube": "유튜브에서 재생목록 보기",
|
||||||
"View channel on YouTube": "유튜브에서 채널 보기",
|
"View channel on YouTube": "유튜브에서 채널 보기",
|
||||||
|
@ -123,7 +123,7 @@
|
||||||
"Create playlist": "재생목록 생성",
|
"Create playlist": "재생목록 생성",
|
||||||
"Trending": "급상승",
|
"Trending": "급상승",
|
||||||
"Delete playlist": "재생목록 삭제",
|
"Delete playlist": "재생목록 삭제",
|
||||||
"Delete playlist `x`?": "재생목록 `x` 를 삭제 하시겠습니까?",
|
"Delete playlist `x`?": "재생목록 `x` 를 삭제하시겠습니까?",
|
||||||
"Updated `x` ago": "`x` 전에 업데이트됨",
|
"Updated `x` ago": "`x` 전에 업데이트됨",
|
||||||
"Released under the AGPLv3 on Github.": "깃허브에 AGPLv3 으로 배포됩니다.",
|
"Released under the AGPLv3 on Github.": "깃허브에 AGPLv3 으로 배포됩니다.",
|
||||||
"View all playlists": "모든 재생목록 보기",
|
"View all playlists": "모든 재생목록 보기",
|
||||||
|
@ -267,7 +267,7 @@
|
||||||
"Bulgarian": "불가리아어",
|
"Bulgarian": "불가리아어",
|
||||||
"Bosnian": "보스니아어",
|
"Bosnian": "보스니아어",
|
||||||
"Belarusian": "벨라루스어",
|
"Belarusian": "벨라루스어",
|
||||||
"View more comments on Reddit": "레딧에서 더 많은 댓글 보기",
|
"View more comments on Reddit": "레딧에서 댓글 더 보기",
|
||||||
"View YouTube comments": "유튜브 댓글 보기",
|
"View YouTube comments": "유튜브 댓글 보기",
|
||||||
"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.": "자바스크립트가 꺼져 있는 것 같습니다! 댓글을 보려면 여기를 클릭하세요. 댓글을 로드하는 데 시간이 조금 더 걸릴 수 있습니다.",
|
||||||
"Shared `x`": "`x` 업로드",
|
"Shared `x`": "`x` 업로드",
|
||||||
|
@ -419,7 +419,7 @@
|
||||||
"Portuguese (Brazil)": "포르투갈어 (브라질)",
|
"Portuguese (Brazil)": "포르투갈어 (브라질)",
|
||||||
"search_message_no_results": "결과가 없습니다.",
|
"search_message_no_results": "결과가 없습니다.",
|
||||||
"search_message_change_filters_or_query": "필터를 변경하시거나 검색어를 넓게 시도해보세요.",
|
"search_message_change_filters_or_query": "필터를 변경하시거나 검색어를 넓게 시도해보세요.",
|
||||||
"search_message_use_another_instance": " 당신은 <a href=\"`x`\">다른 인스턴스에서 검색</a>할 수도 있습니다.",
|
"search_message_use_another_instance": " <a href=\"`x`\">다른 인스턴스에서 검색</a>할 수도 있습니다.",
|
||||||
"English (United States)": "영어 (미국)",
|
"English (United States)": "영어 (미국)",
|
||||||
"Chinese": "중국어",
|
"Chinese": "중국어",
|
||||||
"Chinese (China)": "중국어 (중국)",
|
"Chinese (China)": "중국어 (중국)",
|
||||||
|
@ -460,7 +460,7 @@
|
||||||
"Music in this video": "동영상 속 음악",
|
"Music in this video": "동영상 속 음악",
|
||||||
"Artist: ": "아티스트: ",
|
"Artist: ": "아티스트: ",
|
||||||
"Download is disabled": "다운로드가 비활성화 되어있음",
|
"Download is disabled": "다운로드가 비활성화 되어있음",
|
||||||
"Import YouTube playlist (.csv)": "유튜브 플레이리스트 가져오기 (.csv)",
|
"Import YouTube playlist (.csv)": "유튜브 재생목록 가져오기 (.csv)",
|
||||||
"playlist_button_add_items": "동영상 추가",
|
"playlist_button_add_items": "동영상 추가",
|
||||||
"channel_tab_podcasts_label": "팟캐스트",
|
"channel_tab_podcasts_label": "팟캐스트",
|
||||||
"generic_button_delete": "삭제",
|
"generic_button_delete": "삭제",
|
||||||
|
@ -468,7 +468,16 @@
|
||||||
"generic_button_save": "저장",
|
"generic_button_save": "저장",
|
||||||
"generic_button_cancel": "취소",
|
"generic_button_cancel": "취소",
|
||||||
"generic_button_rss": "RSS",
|
"generic_button_rss": "RSS",
|
||||||
"channel_tab_releases_label": "출시",
|
"channel_tab_releases_label": "발매",
|
||||||
"generic_channels_count_0": "{{count}} 채널",
|
"generic_channels_count_0": "{{count}} 채널",
|
||||||
"Import YouTube watch history (.json)": "유튜브 시청 기록 가져오기 (.json)"
|
"Import YouTube watch history (.json)": "유튜브 시청 기록 가져오기 (.json)",
|
||||||
|
"Add to playlist": "재생목록에 추가",
|
||||||
|
"Add to playlist: ": "재생목록에 추가: ",
|
||||||
|
"Answer": "답",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "관리자가 인기 피드를 비활성화했습니다.",
|
||||||
|
"carousel_skip": "캐러셀 건너뛰기",
|
||||||
|
"carousel_go_to": "`x` 슬라이드로 이동",
|
||||||
|
"Search for videos": "비디오 검색",
|
||||||
|
"toggle_theme": "테마 전환",
|
||||||
|
"carousel_slide": "{{total}}의 슬라이드 {{current}}"
|
||||||
}
|
}
|
||||||
|
|
232
locales/lmo.json
Normal file
232
locales/lmo.json
Normal file
|
@ -0,0 +1,232 @@
|
||||||
|
{
|
||||||
|
"Add to playlist": "Giont a la playlist",
|
||||||
|
"generic_button_edit": "Modifega",
|
||||||
|
"generic_button_save": "Salva",
|
||||||
|
"LIVE": "EN DÌRETT",
|
||||||
|
"Shared `x` ago": "Compartiss `x` fa",
|
||||||
|
"View channel on YouTube": "Varda el canal sul YouTube",
|
||||||
|
"newest": "plù nöeuf",
|
||||||
|
"oldest": "plù végh",
|
||||||
|
"Search for videos": "Càuta dei video",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "la seziùn Pupular la è stada disabilidada par l'amministratòr.",
|
||||||
|
"generic_channels_count": "{{count}} canal",
|
||||||
|
"generic_channels_count_plural": "{{count}} canai",
|
||||||
|
"popular": "pupular",
|
||||||
|
"generic_views_count": "{{count}} visualisazión",
|
||||||
|
"generic_views_count_plural": "{{count}} visualisazióni",
|
||||||
|
"generic_videos_count": "{{count}} video",
|
||||||
|
"generic_videos_count_plural": "{{count}} video",
|
||||||
|
"generic_playlists_count": "{{count}} playlist",
|
||||||
|
"generic_playlists_count_plural": "{{count}} playlist",
|
||||||
|
"generic_subscriptions_count": "{{count}} inscrizion",
|
||||||
|
"generic_subscriptions_count_plural": "{{count}} inscrizioni",
|
||||||
|
"generic_button_cancel": "Cançéla",
|
||||||
|
"generic_button_delete": "Scassa via",
|
||||||
|
"Unsubscribe": "Disinscriviti",
|
||||||
|
"Next page": "Pagina siguènt",
|
||||||
|
"Previous page": "Pagina indrèe",
|
||||||
|
"Clear watch history?": "Cançélar la istoria dei video vardàa?",
|
||||||
|
"New password": "Nöeva password",
|
||||||
|
"Import and Export Data": "Importazion ed esportazion dei dat",
|
||||||
|
"Import": "Importa",
|
||||||
|
"Import Invidious data": "Importa i dat de l'Invidious en el formàt JSON",
|
||||||
|
"Import YouTube subscriptions": "Importa le inscrizioni dal YouTube/OPML",
|
||||||
|
"Import YouTube playlist (.csv)": "Importa le playlist dal YouTube (.csv)",
|
||||||
|
"Import YouTube watch history (.json)": "Importa la istoria de visualizazzion dal YouTube (.json)",
|
||||||
|
"Import FreeTube subscriptions (.db)": "Importa le inscrizioni dal FreeTube (.db)",
|
||||||
|
"Import NewPipe data (.zip)": "importa i dat del NewPipe (.zip)",
|
||||||
|
"Export": "Esporta",
|
||||||
|
"Export subscriptions as OPML": "Esporta inscrizioni com OPML",
|
||||||
|
"Export data as JSON": "Esporta i dat de l'Invidious com JSON",
|
||||||
|
"Delete account?": "Eliminà 'l profil?",
|
||||||
|
"History": "Istoria",
|
||||||
|
"An alternative front-end to YouTube": "Una interfacia alternatif al YouTube",
|
||||||
|
"JavaScript license information": "Informaziòn su la licensa JavaScript",
|
||||||
|
"source": "font",
|
||||||
|
"Log in": "Và dent",
|
||||||
|
"Text CAPTCHA": "Tèst del CAPTCHA",
|
||||||
|
"Image CAPTCHA": "Imàgen del CAPTCHA",
|
||||||
|
"Sign In": "Ven denter",
|
||||||
|
"Register": "Registres",
|
||||||
|
"E-mail": "E-mail",
|
||||||
|
"Preferences": "Priferenze",
|
||||||
|
"preferences_category_player": "Priferenze del riprodutòr",
|
||||||
|
"preferences_quality_option_dash": "DASH (qualità adatif)",
|
||||||
|
"preferences_quality_option_hd720": "HD720",
|
||||||
|
"preferences_quality_option_medium": "Media",
|
||||||
|
"preferences_quality_option_small": "Picinina",
|
||||||
|
"preferences_quality_dash_option_auto": "Auto",
|
||||||
|
"preferences_quality_dash_option_best": "Meglior",
|
||||||
|
"preferences_quality_dash_option_worst": "Peggior",
|
||||||
|
"preferences_quality_dash_option_4320p": "4320p",
|
||||||
|
"preferences_quality_dash_option_2160p": "2160p",
|
||||||
|
"preferences_quality_dash_option_1440p": "1440p",
|
||||||
|
"preferences_quality_dash_option_1080p": "1080p",
|
||||||
|
"preferences_quality_dash_option_720p": "720p",
|
||||||
|
"preferences_quality_dash_option_480p": "480p",
|
||||||
|
"preferences_quality_dash_option_360p": "360p",
|
||||||
|
"preferences_quality_dash_option_240p": "240p",
|
||||||
|
"preferences_quality_dash_option_144p": "144p",
|
||||||
|
"reddit": "Reddit",
|
||||||
|
"invidious": "Invidious",
|
||||||
|
"light": "ciar",
|
||||||
|
"dark": "scur",
|
||||||
|
"preferences_category_misc": "Priferenze varie",
|
||||||
|
"preferences_category_subscription": "Priferenze de le inscrizioni",
|
||||||
|
"published": "data de publicazion",
|
||||||
|
"published - reverse": "data de publicazion - invertì",
|
||||||
|
"alphabetically": "orden alfabetegh",
|
||||||
|
"channel name": "nòm del canal",
|
||||||
|
"channel name - reverse": "nòm del canal - invertì",
|
||||||
|
"Enable web notifications": "Empisa le notifeghe da la red",
|
||||||
|
"`x` uploaded a video": "`x` la ghàa cargà un video",
|
||||||
|
"`x` is live": "`x` l'è 'n dirétt adés",
|
||||||
|
"preferences_category_data": "Priferenze dei dat",
|
||||||
|
"Import/export data": "Importa/esporta i dat",
|
||||||
|
"Change password": "Cambia la parola ciav",
|
||||||
|
"Manage subscriptions": "Organisa le inscrizioni",
|
||||||
|
"Manage tokens": "Organisa i tokens",
|
||||||
|
"Watch history": "Istoria dei video vardà",
|
||||||
|
"Delete account": "Cançéla 'l profil",
|
||||||
|
"Save preferences": "Salva priferenze",
|
||||||
|
"Subscription manager": "Manegia le inscrizioni",
|
||||||
|
"Token": "Token",
|
||||||
|
"tokens_count": "{{count}} token",
|
||||||
|
"tokens_count_plural": "{{count}} token",
|
||||||
|
"Import/export": "Importa/esporta",
|
||||||
|
"unsubscribe": "disinscriviti",
|
||||||
|
"subscriptions_unseen_notifs_count": "{{count}} notifega mia visualisada",
|
||||||
|
"subscriptions_unseen_notifs_count_plural": "{{count}} notifeghe mia visualisade",
|
||||||
|
"Log out": "Sortiss",
|
||||||
|
"Released under the AGPLv3 on Github.": "Publicà en el GitHub suta licenza AGPLv3.",
|
||||||
|
"Source available here.": "Codegh de la font disponivel chì.",
|
||||||
|
"View privacy policy.": "Varda la pulitega de la privacy.",
|
||||||
|
"Trending": "De moda",
|
||||||
|
"Public": "Publico",
|
||||||
|
"Unlisted": "Non en lista",
|
||||||
|
"Private": "Privàt",
|
||||||
|
"View all playlists": "Varda tute le playlist",
|
||||||
|
"Updated `x` ago": "Giurnà `x` fa",
|
||||||
|
"Delete playlist `x`?": "Cançéla la playlist `x`?",
|
||||||
|
"Delete playlist": "Cançéla playlist",
|
||||||
|
"Create playlist": "Crea playlist",
|
||||||
|
"Title": "Titel",
|
||||||
|
"Playlist privacy": "Privacy de la playlist",
|
||||||
|
"Editing playlist `x`": "Modifega playlist `x`",
|
||||||
|
"playlist_button_add_items": "Gionta video",
|
||||||
|
"Show more": "Varda plù",
|
||||||
|
"Show less": "Varda mèn",
|
||||||
|
"Watch on YouTube": "Varda sul YouTube",
|
||||||
|
"Switch Invidious Instance": "Cambia la instanza del Invidious",
|
||||||
|
"search_message_no_results": "Non è stat truvà nigun resultat.",
|
||||||
|
"Cebuano": "Cebuano",
|
||||||
|
"Chinese (Traditional)": "Cines (Tradizional)",
|
||||||
|
"Corsican": "Còrso",
|
||||||
|
"Croatian": "Cruat",
|
||||||
|
"Georgian": "Georgian",
|
||||||
|
"Gujarati": "Gujarati",
|
||||||
|
"Hawaiian": "Hawaiian",
|
||||||
|
"Kurdish": "Curd",
|
||||||
|
"Latin": "Latin",
|
||||||
|
"Latvian": "Letton",
|
||||||
|
"Lithuanian": "Lituan",
|
||||||
|
"Malay": "Males",
|
||||||
|
"Maltese": "Maltes",
|
||||||
|
"Mongolian": "móngol",
|
||||||
|
"Persian": "Persian",
|
||||||
|
"Polish": "Polacch",
|
||||||
|
"Portuguese": "Portoghes",
|
||||||
|
"Romanian": "Romen",
|
||||||
|
"Scottish Gaelic": "Gaelich Scusses",
|
||||||
|
"Spanish (Latin America)": "Spagnöl (America do Sùd)",
|
||||||
|
"Thai": "Thai",
|
||||||
|
"Western Frisian": "Frisian do ponente",
|
||||||
|
"Basque": "Basco",
|
||||||
|
"Chinese (Simplified)": "Cines (Semplificà)",
|
||||||
|
"Haitian Creole": "Creolo de Haiti",
|
||||||
|
"Galician": "Galiçian",
|
||||||
|
"Hebrew": "Ebraich",
|
||||||
|
"Korean": "Corean",
|
||||||
|
"View playlist on YouTube": "Varda la playlist sul YouTube",
|
||||||
|
"Southern Sotho": "Sotho do Sùd",
|
||||||
|
"generic_button_rss": "RSS",
|
||||||
|
"Welsh": "Galés",
|
||||||
|
"Answer": "Resposta",
|
||||||
|
"New passwords must match": "Le nöeve password la deven esere uguai",
|
||||||
|
"Authorize token?": "Autorisà 'l token?",
|
||||||
|
"Authorize token for `x`?": "Autorisà 'l token par `x`?",
|
||||||
|
"Yes": "Sì",
|
||||||
|
"No": "No",
|
||||||
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Esporta inscrizioni com OPML (par 'l NewPipe e 'l FreeTube)",
|
||||||
|
"Log in/register": "Va dent/Registres",
|
||||||
|
"User ID": "ID utent",
|
||||||
|
"Password": "Parola ciav",
|
||||||
|
"Time (h:mm:ss):": "Temp (h:mm:ss):",
|
||||||
|
"Import NewPipe subscriptions (.json)": "importa le inscrizioni dal NewPipe (.json)",
|
||||||
|
"youtube": "YouTube",
|
||||||
|
"alphabetically - reverse": "orden alfabetegh - invertì",
|
||||||
|
"preferences_category_visual": "Priferenze grafeghe",
|
||||||
|
"Clear watch history": "Scompartiss la istoria dei video vardà",
|
||||||
|
"preferences_category_admin": "Priferenze de l'amministratòr",
|
||||||
|
"Token manager": "Manegia i token",
|
||||||
|
"Subscriptions": "Inscrizioni",
|
||||||
|
"search": "cerca",
|
||||||
|
"View JavaScript license information.": "Varda le informazion su la licenza JavaScript.",
|
||||||
|
"search_message_change_filters_or_query": "Ti pödi pruà a slargà la reçerca e/or a cangià i filter.",
|
||||||
|
"generic_subscribers_count": "{{count}} inscritt",
|
||||||
|
"generic_subscribers_count_plural": "{{count}} inscriti",
|
||||||
|
"Subscribe": "Inscriviti",
|
||||||
|
"last": "ùltim",
|
||||||
|
"Add to playlist: ": "Giont a la playlist: ",
|
||||||
|
"preferences_autoplay_label": "Reproduzion automatega: ",
|
||||||
|
"preferences_continue_label": "Reproduzion seguént preimpostà: ",
|
||||||
|
"preferences_continue_autoplay_label": "Fa partì en automatico el video seguént: ",
|
||||||
|
"preferences_listen_label": "Modalità de sól audio preimpostà: ",
|
||||||
|
"preferences_local_label": "Proxy par i video: ",
|
||||||
|
"preferences_watch_history_label": "Ativà la istoria de reproduzion: ",
|
||||||
|
"preferences_speed_label": "Velocità preimpostà: ",
|
||||||
|
"preferences_volume_label": "Volume del reprodutòr: ",
|
||||||
|
"preferences_region_label": "Nazion del contenut: ",
|
||||||
|
"Dark mode: ": "Tema scur ",
|
||||||
|
"preferences_dark_mode_label": "Tema: ",
|
||||||
|
"preferences_thin_mode_label": "Modalità legera: ",
|
||||||
|
"preferences_automatic_instance_redirect_label": "Reindirizazzion automatega de la instansa (rivèrt a redirect.invidious.io): ",
|
||||||
|
"Hide annotations": "Piaca le notazioni",
|
||||||
|
"Show annotations": "Mostra le notazioni",
|
||||||
|
"Family friendly? ": "Adàt a tüti? ",
|
||||||
|
"Whitelisted regions: ": "Regioni en lista bianca: ",
|
||||||
|
"Blacklisted regions: ": "Regioni en lista negher ",
|
||||||
|
"Artist: ": "Artista: ",
|
||||||
|
"Song: ": "Cansòn ",
|
||||||
|
"Album: ": "Album: ",
|
||||||
|
"View YouTube comments": "Varda i comment dal YouTube",
|
||||||
|
"Password cannot be empty": "La parola ciav la no po miga esser voeut",
|
||||||
|
"channel:`x`": "Canal:`x`",
|
||||||
|
"Bangla": "Bengales",
|
||||||
|
"Hausa": "Hausa",
|
||||||
|
"Hindi": "Hindi",
|
||||||
|
"Hmong": "Hmong",
|
||||||
|
"Igbo": "Igbo",
|
||||||
|
"Javanese": "Javanese",
|
||||||
|
"Kannada": "Kannada",
|
||||||
|
"Kazakh": "Kazach",
|
||||||
|
"Khmer": "Khmer",
|
||||||
|
"Kyrgyz": "Kirghiz",
|
||||||
|
"Lao": "Lao",
|
||||||
|
"Luxembourgish": "Lussemburghes",
|
||||||
|
"Macedonian": "Macedon",
|
||||||
|
"Malagasy": "Malagascio",
|
||||||
|
"Malayalam": "Malayalam",
|
||||||
|
"Maori": "Maori",
|
||||||
|
"Marathi": "Marati",
|
||||||
|
"Nepali": "Nepales",
|
||||||
|
"Nyanja": "Nyanja",
|
||||||
|
"Pashto": "Pashtu",
|
||||||
|
"Punjabi": "Punjabi",
|
||||||
|
"Samoan": "Samoan",
|
||||||
|
"Standard YouTube license": "licensa predefinida de Youtube",
|
||||||
|
"License: ": "Licensa: ",
|
||||||
|
"Music in this video": "Musica en sto video",
|
||||||
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Ué! Sembra che ti la g'hà desabilitàa el JavaScript. Schisa chì para vardà i comment, ma cunsidera che peul vörse 'n po plu de temp a cargà.",
|
||||||
|
"preferences_video_loop_label": "Reproduci sèmper: "
|
||||||
|
}
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Importer- og eksporter data",
|
"Import and Export Data": "Importer- og eksporter data",
|
||||||
"Import": "Importer",
|
"Import": "Importer",
|
||||||
"Import Invidious data": "Importer Invidious-JSON-data",
|
"Import Invidious data": "Importer Invidious-JSON-data",
|
||||||
"Import YouTube subscriptions": "Importer YouTube/OPML-abonnementer",
|
"Import YouTube subscriptions": "Importer YouTube CSV eller OPML-abonnementer",
|
||||||
"Import FreeTube subscriptions (.db)": "Importer FreeTube-abonnementer (.db)",
|
"Import FreeTube subscriptions (.db)": "Importer FreeTube-abonnementer (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importer NewPipe-abonnementer (.json)",
|
"Import NewPipe subscriptions (.json)": "Importer NewPipe-abonnementer (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importer NewPipe-data (.zip)",
|
"Import NewPipe data (.zip)": "Importer NewPipe-data (.zip)",
|
||||||
|
@ -322,13 +322,13 @@
|
||||||
"channel_tab_community_label": "Gemenskap",
|
"channel_tab_community_label": "Gemenskap",
|
||||||
"search_filters_sort_option_relevance": "relevans",
|
"search_filters_sort_option_relevance": "relevans",
|
||||||
"search_filters_sort_option_rating": "vurdering",
|
"search_filters_sort_option_rating": "vurdering",
|
||||||
"search_filters_sort_option_date": "dato",
|
"search_filters_sort_option_date": "Opplastingsdato",
|
||||||
"search_filters_sort_option_views": "visninger",
|
"search_filters_sort_option_views": "visninger",
|
||||||
"search_filters_type_label": "innholdstype",
|
"search_filters_type_label": "innholdstype",
|
||||||
"search_filters_duration_label": "varighet",
|
"search_filters_duration_label": "varighet",
|
||||||
"search_filters_features_label": "funksjoner",
|
"search_filters_features_label": "funksjoner",
|
||||||
"search_filters_sort_label": "sorter",
|
"search_filters_sort_label": "sorter",
|
||||||
"search_filters_date_option_hour": "time",
|
"search_filters_date_option_hour": "Siste time",
|
||||||
"search_filters_date_option_today": "i dag",
|
"search_filters_date_option_today": "i dag",
|
||||||
"search_filters_date_option_week": "uke",
|
"search_filters_date_option_week": "uke",
|
||||||
"search_filters_date_option_month": "måned",
|
"search_filters_date_option_month": "måned",
|
||||||
|
@ -459,7 +459,7 @@
|
||||||
"search_message_no_results": "Resultatløst.",
|
"search_message_no_results": "Resultatløst.",
|
||||||
"search_filters_type_option_all": "Alle typer",
|
"search_filters_type_option_all": "Alle typer",
|
||||||
"search_filters_duration_option_none": "Enhver varighet",
|
"search_filters_duration_option_none": "Enhver varighet",
|
||||||
"search_message_use_another_instance": " Du kan også <a href=\"`x`\">søke på en annen instans</a>.",
|
"search_message_use_another_instance": "Du kan også <a href=\"`x`\">søke på en annen instans</a>.",
|
||||||
"search_filters_date_label": "Opplastningsdato",
|
"search_filters_date_label": "Opplastningsdato",
|
||||||
"search_filters_apply_button": "Bruk valgte filtre",
|
"search_filters_apply_button": "Bruk valgte filtre",
|
||||||
"search_filters_date_option_none": "Siden begynnelsen",
|
"search_filters_date_option_none": "Siden begynnelsen",
|
||||||
|
@ -487,5 +487,14 @@
|
||||||
"playlist_button_add_items": "Legg til videoer",
|
"playlist_button_add_items": "Legg til videoer",
|
||||||
"generic_channels_count": "{{count}} kanal",
|
"generic_channels_count": "{{count}} kanal",
|
||||||
"generic_channels_count_plural": "{{count}} kanaler",
|
"generic_channels_count_plural": "{{count}} kanaler",
|
||||||
"Import YouTube watch history (.json)": "Importere YouTube visningshistorikk (.json)"
|
"Import YouTube watch history (.json)": "Importere YouTube visningshistorikk (.json)",
|
||||||
|
"carousel_go_to": "Gå til lysark `x`",
|
||||||
|
"Search for videos": "Søk i videoer",
|
||||||
|
"Answer": "Svar",
|
||||||
|
"carousel_slide": "Lysark {{current}} av {{total}}",
|
||||||
|
"carousel_skip": "Hopp over karusellen",
|
||||||
|
"Add to playlist": "Legg til i spilleliste",
|
||||||
|
"Add to playlist: ": "Legg til i spilleliste: ",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Populært-kilden er koblet ut av administratoren.",
|
||||||
|
"toggle_theme": "Endre utseende"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Gegevens im- en exporteren",
|
"Import and Export Data": "Gegevens im- en exporteren",
|
||||||
"Import": "Importeren",
|
"Import": "Importeren",
|
||||||
"Import Invidious data": "JSON-gegevens Invidious importeren",
|
"Import Invidious data": "JSON-gegevens Invidious importeren",
|
||||||
"Import YouTube subscriptions": "YouTube-/OPML-abonnementen importeren",
|
"Import YouTube subscriptions": "YouTube CVS of OPML-abonnementen importeren",
|
||||||
"Import FreeTube subscriptions (.db)": "FreeTube-abonnementen importeren (.db)",
|
"Import FreeTube subscriptions (.db)": "FreeTube-abonnementen importeren (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "NewPipe-abonnementen importeren (.json)",
|
"Import NewPipe subscriptions (.json)": "NewPipe-abonnementen importeren (.json)",
|
||||||
"Import NewPipe data (.zip)": "NewPipe-gegevens importeren (.zip)",
|
"Import NewPipe data (.zip)": "NewPipe-gegevens importeren (.zip)",
|
||||||
|
@ -86,7 +86,7 @@
|
||||||
"Only show latest unwatched video from channel: ": "Alleen nieuwste niet-bekeken video van kanaal tonen: ",
|
"Only show latest unwatched video from channel: ": "Alleen nieuwste niet-bekeken video van kanaal tonen: ",
|
||||||
"preferences_unseen_only_label": "Alleen niet-bekeken videos tonen: ",
|
"preferences_unseen_only_label": "Alleen niet-bekeken videos tonen: ",
|
||||||
"preferences_notifications_only_label": "Alleen meldingen tonen (als die er zijn): ",
|
"preferences_notifications_only_label": "Alleen meldingen tonen (als die er zijn): ",
|
||||||
"Enable web notifications": "Systemmeldingen inschakelen",
|
"Enable web notifications": "Systeemmeldingen inschakelen",
|
||||||
"`x` uploaded a video": "`x` heeft een video geüpload",
|
"`x` uploaded a video": "`x` heeft een video geüpload",
|
||||||
"`x` is live": "`x` zendt nu live uit",
|
"`x` is live": "`x` zendt nu live uit",
|
||||||
"preferences_category_data": "Gegevensinstellingen",
|
"preferences_category_data": "Gegevensinstellingen",
|
||||||
|
@ -192,15 +192,15 @@
|
||||||
"Arabic": "Arabisch",
|
"Arabic": "Arabisch",
|
||||||
"Armenian": "Armeens",
|
"Armenian": "Armeens",
|
||||||
"Azerbaijani": "Azerbeidzjaans",
|
"Azerbaijani": "Azerbeidzjaans",
|
||||||
"Bangla": "Bangla",
|
"Bangla": "Bengaals",
|
||||||
"Basque": "Baskisch",
|
"Basque": "Baskisch",
|
||||||
"Belarusian": "Wit-Rrussisch",
|
"Belarusian": "Wit-Russisch",
|
||||||
"Bosnian": "Bosnisch",
|
"Bosnian": "Bosnisch",
|
||||||
"Bulgarian": "Bulgaars",
|
"Bulgarian": "Bulgaars",
|
||||||
"Burmese": "Birmaans",
|
"Burmese": "Birmaans",
|
||||||
"Catalan": "Catalaans",
|
"Catalan": "Catalaans",
|
||||||
"Cebuano": "Cebuano",
|
"Cebuano": "Cebuaans",
|
||||||
"Chinese (Simplified)": "Chinees (Veereenvoudigd)",
|
"Chinese (Simplified)": "Chinees (Vereenvoudigd)",
|
||||||
"Chinese (Traditional)": "Chinees (Traditioneel)",
|
"Chinese (Traditional)": "Chinees (Traditioneel)",
|
||||||
"Corsican": "Corsicaans",
|
"Corsican": "Corsicaans",
|
||||||
"Croatian": "Kroatisch",
|
"Croatian": "Kroatisch",
|
||||||
|
@ -217,23 +217,23 @@
|
||||||
"German": "Duits",
|
"German": "Duits",
|
||||||
"Greek": "Grieks",
|
"Greek": "Grieks",
|
||||||
"Gujarati": "Gujarati",
|
"Gujarati": "Gujarati",
|
||||||
"Haitian Creole": "Creools",
|
"Haitian Creole": "Haïtiaans Creools",
|
||||||
"Hausa": "Hausa",
|
"Hausa": "Hausa",
|
||||||
"Hawaiian": "Hawaïaans",
|
"Hawaiian": "Hawaïaans",
|
||||||
"Hebrew": "Heebreeuws",
|
"Hebrew": "Hebreeuws",
|
||||||
"Hindi": "Hindi",
|
"Hindi": "Hindi",
|
||||||
"Hmong": "Hmong",
|
"Hmong": "Hmong",
|
||||||
"Hungarian": "Hongaars",
|
"Hungarian": "Hongaars",
|
||||||
"Icelandic": "IJslands",
|
"Icelandic": "IJslands",
|
||||||
"Igbo": "Igbo",
|
"Igbo": "Ikbo",
|
||||||
"Indonesian": "Indonesisch",
|
"Indonesian": "Indonesisch",
|
||||||
"Irish": "Iers",
|
"Irish": "Iers",
|
||||||
"Italian": "Italiaans",
|
"Italian": "Italiaans",
|
||||||
"Japanese": "Japans",
|
"Japanese": "Japans",
|
||||||
"Javanese": "Javaans",
|
"Javanese": "Javaans",
|
||||||
"Kannada": "Kannada",
|
"Kannada": "Kannada-taal",
|
||||||
"Kazakh": "Kazachs",
|
"Kazakh": "Kazachs",
|
||||||
"Khmer": "Khmer",
|
"Khmer": "Khmer-taal",
|
||||||
"Korean": "Koreaans",
|
"Korean": "Koreaans",
|
||||||
"Kurdish": "Koerdisch",
|
"Kurdish": "Koerdisch",
|
||||||
"Kyrgyz": "Kirgizisch",
|
"Kyrgyz": "Kirgizisch",
|
||||||
|
@ -245,10 +245,10 @@
|
||||||
"Macedonian": "Macedonisch",
|
"Macedonian": "Macedonisch",
|
||||||
"Malagasy": "Malagassisch",
|
"Malagasy": "Malagassisch",
|
||||||
"Malay": "Maleisisch",
|
"Malay": "Maleisisch",
|
||||||
"Malayalam": "Malayalam",
|
"Malayalam": "Malayalam-taal",
|
||||||
"Maltese": "Maltees",
|
"Maltese": "Maltees",
|
||||||
"Maori": "Maorisch",
|
"Maori": "Maorisch",
|
||||||
"Marathi": "Marathi",
|
"Marathi": "Marathi-taal",
|
||||||
"Mongolian": "Mongools",
|
"Mongolian": "Mongools",
|
||||||
"Nepali": "Nepalees",
|
"Nepali": "Nepalees",
|
||||||
"Norwegian Bokmål": "Noors (Bokmål)",
|
"Norwegian Bokmål": "Noors (Bokmål)",
|
||||||
|
@ -309,7 +309,7 @@
|
||||||
"(edited)": "(bewerkt)",
|
"(edited)": "(bewerkt)",
|
||||||
"YouTube comment permalink": "Link naar YouTube-reactie",
|
"YouTube comment permalink": "Link naar YouTube-reactie",
|
||||||
"permalink": "permalink",
|
"permalink": "permalink",
|
||||||
"`x` marked it with a ❤": "`x` heeft dit gemarkeerd met ❤",
|
"`x` marked it with a ❤": "`x` heeft dit gemarkeerd met een ❤",
|
||||||
"Audio mode": "Audiomodus",
|
"Audio mode": "Audiomodus",
|
||||||
"Video mode": "Videomodus",
|
"Video mode": "Videomodus",
|
||||||
"channel_tab_videos_label": "Video's",
|
"channel_tab_videos_label": "Video's",
|
||||||
|
@ -317,13 +317,13 @@
|
||||||
"channel_tab_community_label": "Gemeenschap",
|
"channel_tab_community_label": "Gemeenschap",
|
||||||
"search_filters_sort_option_relevance": "relevantie",
|
"search_filters_sort_option_relevance": "relevantie",
|
||||||
"search_filters_sort_option_rating": "beoordeling",
|
"search_filters_sort_option_rating": "beoordeling",
|
||||||
"search_filters_sort_option_date": "datum",
|
"search_filters_sort_option_date": "Upload datum",
|
||||||
"search_filters_sort_option_views": "keren bekeken",
|
"search_filters_sort_option_views": "keren bekeken",
|
||||||
"search_filters_type_label": "Type inhoud",
|
"search_filters_type_label": "Type inhoud",
|
||||||
"search_filters_duration_label": "duur",
|
"search_filters_duration_label": "duur",
|
||||||
"search_filters_features_label": "eigenschappen",
|
"search_filters_features_label": "eigenschappen",
|
||||||
"search_filters_sort_label": "sorteren",
|
"search_filters_sort_label": "sorteren",
|
||||||
"search_filters_date_option_hour": "uur",
|
"search_filters_date_option_hour": "Laatste uur",
|
||||||
"search_filters_date_option_today": "vandaag",
|
"search_filters_date_option_today": "vandaag",
|
||||||
"search_filters_date_option_week": "week",
|
"search_filters_date_option_week": "week",
|
||||||
"search_filters_date_option_month": "maand",
|
"search_filters_date_option_month": "maand",
|
||||||
|
@ -357,7 +357,7 @@
|
||||||
"footer_original_source_code": "Originele bron-code",
|
"footer_original_source_code": "Originele bron-code",
|
||||||
"footer_modfied_source_code": "Gewijzigde bron-code",
|
"footer_modfied_source_code": "Gewijzigde bron-code",
|
||||||
"adminprefs_modified_source_code_url_label": "URL naar gewijzigde bron-code-opslagplaats",
|
"adminprefs_modified_source_code_url_label": "URL naar gewijzigde bron-code-opslagplaats",
|
||||||
"next_steps_error_message": "Daarna moet u proberen om: ",
|
"next_steps_error_message": "Waarna u zou kunnen proberen om: ",
|
||||||
"footer_source_code": "Bron-code",
|
"footer_source_code": "Bron-code",
|
||||||
"search_filters_duration_option_long": "Lang (> 20 minuten)",
|
"search_filters_duration_option_long": "Lang (> 20 minuten)",
|
||||||
"preferences_quality_option_dash": "DASH (adaptieve kwaliteit)",
|
"preferences_quality_option_dash": "DASH (adaptieve kwaliteit)",
|
||||||
|
@ -396,7 +396,7 @@
|
||||||
"Dutch (auto-generated)": "Nederlands (automatisch gegenereerd)",
|
"Dutch (auto-generated)": "Nederlands (automatisch gegenereerd)",
|
||||||
"tokens_count": "{{count}} token",
|
"tokens_count": "{{count}} token",
|
||||||
"tokens_count_plural": "{{count}} tokens",
|
"tokens_count_plural": "{{count}} tokens",
|
||||||
"generic_count_seconds": "{{count}} second",
|
"generic_count_seconds": "{{count}} seconde",
|
||||||
"generic_count_seconds_plural": "{{count}} seconden",
|
"generic_count_seconds_plural": "{{count}} seconden",
|
||||||
"generic_count_weeks": "{{count}} week",
|
"generic_count_weeks": "{{count}} week",
|
||||||
"generic_count_weeks_plural": "{{count}} weken",
|
"generic_count_weeks_plural": "{{count}} weken",
|
||||||
|
@ -449,8 +449,8 @@
|
||||||
"generic_playlists_count_plural": "{{count}} afspeellijsten",
|
"generic_playlists_count_plural": "{{count}} afspeellijsten",
|
||||||
"Chinese (Hong Kong)": "Chinees (Hongkong)",
|
"Chinese (Hong Kong)": "Chinees (Hongkong)",
|
||||||
"Korean (auto-generated)": "Koreaans (automatisch gegenereerd)",
|
"Korean (auto-generated)": "Koreaans (automatisch gegenereerd)",
|
||||||
"search_filters_apply_button": "Geselecteerd filters toepassen",
|
"search_filters_apply_button": "Geselecteerde filters toepassen",
|
||||||
"search_message_use_another_instance": " Je kan ook <a href=\"`x`\">zoeken op een andere instantie</a>.",
|
"search_message_use_another_instance": "Je kan ook <a href=\"`x`\">zoeken op een andere instantie</a>.",
|
||||||
"Cantonese (Hong Kong)": "Kantonees (Hongkong)",
|
"Cantonese (Hong Kong)": "Kantonees (Hongkong)",
|
||||||
"Chinese (China)": "Chinees (China)",
|
"Chinese (China)": "Chinees (China)",
|
||||||
"crash_page_read_the_faq": "de <a href=\"`x`\">veelgestelde vragen (FAQ)</a> gelezen hebt",
|
"crash_page_read_the_faq": "de <a href=\"`x`\">veelgestelde vragen (FAQ)</a> gelezen hebt",
|
||||||
|
@ -477,7 +477,7 @@
|
||||||
"Song: ": "Lied: ",
|
"Song: ": "Lied: ",
|
||||||
"generic_channels_count": "{{count}} kanaal",
|
"generic_channels_count": "{{count}} kanaal",
|
||||||
"generic_channels_count_plural": "{{count}} kanalen",
|
"generic_channels_count_plural": "{{count}} kanalen",
|
||||||
"Popular enabled: ": "Populair geactiveerd: ",
|
"Popular enabled: ": "Populair ingeschakeld: ",
|
||||||
"channel_tab_playlists_label": "Afspeellijsten",
|
"channel_tab_playlists_label": "Afspeellijsten",
|
||||||
"generic_button_edit": "Bewerken",
|
"generic_button_edit": "Bewerken",
|
||||||
"Music in this video": "Muziek in deze video",
|
"Music in this video": "Muziek in deze video",
|
||||||
|
@ -487,5 +487,14 @@
|
||||||
"generic_button_delete": "Verwijderen",
|
"generic_button_delete": "Verwijderen",
|
||||||
"Import YouTube playlist (.csv)": "YouTube-afspeellijst importeren (.csv)",
|
"Import YouTube playlist (.csv)": "YouTube-afspeellijst importeren (.csv)",
|
||||||
"Standard YouTube license": "Standaard YouTube-licentie",
|
"Standard YouTube license": "Standaard YouTube-licentie",
|
||||||
"Import YouTube watch history (.json)": "YouTube-kijkgeschiedenis importeren (.json)"
|
"Import YouTube watch history (.json)": "YouTube-kijkgeschiedenis importeren (.json)",
|
||||||
|
"Add to playlist": "Aan afspeellijst toevoegen",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "De Populaire feed werd uitgeschakeld door een beheerder.",
|
||||||
|
"carousel_slide": "Dia {{current}} van {{total}}",
|
||||||
|
"carousel_go_to": "Naar dia `x` gaan",
|
||||||
|
"Add to playlist: ": "Aan afspeellijst toevoegen: ",
|
||||||
|
"Answer": "Antwoorden",
|
||||||
|
"Search for videos": "Naar video's zoeken",
|
||||||
|
"carousel_skip": "Carousel overslaan",
|
||||||
|
"toggle_theme": "Thema omschakelen"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,13 +21,13 @@
|
||||||
"Import and Export Data": "Import i eksport danych",
|
"Import and Export Data": "Import i eksport danych",
|
||||||
"Import": "Import",
|
"Import": "Import",
|
||||||
"Import Invidious data": "Importuj dane JSON Invidious",
|
"Import Invidious data": "Importuj dane JSON Invidious",
|
||||||
"Import YouTube subscriptions": "Importuj subskrybcje z YouTube/OPML",
|
"Import YouTube subscriptions": "Importuj subskrypcje YouTube w formacie CSV lub OPML",
|
||||||
"Import FreeTube subscriptions (.db)": "Importuj subskrybcje z FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importuj subskrypcje FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importuj subskrybcje z NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importuj subskrypcje NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importuj dane NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importuj dane NewPipe (.zip)",
|
||||||
"Export": "Eksport",
|
"Export": "Eksport",
|
||||||
"Export subscriptions as OPML": "Eksportuj subskrybcje jako OPML",
|
"Export subscriptions as OPML": "Eksportuj subskrypcje jako OPML",
|
||||||
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksportuj subskrybcje jako OPML (dla NewPipe i FreeTube)",
|
"Export subscriptions as OPML (for NewPipe & FreeTube)": "Eksportuj subskrypcje jako OPML (dla NewPipe i FreeTube)",
|
||||||
"Export data as JSON": "Eksportuj dane Invidious jako JSON",
|
"Export data as JSON": "Eksportuj dane Invidious jako JSON",
|
||||||
"Delete account?": "Usunąć konto?",
|
"Delete account?": "Usunąć konto?",
|
||||||
"History": "Historia",
|
"History": "Historia",
|
||||||
|
@ -73,7 +73,7 @@
|
||||||
"preferences_thin_mode_label": "Tryb minimalny: ",
|
"preferences_thin_mode_label": "Tryb minimalny: ",
|
||||||
"preferences_category_misc": "Różne preferencje",
|
"preferences_category_misc": "Różne preferencje",
|
||||||
"preferences_automatic_instance_redirect_label": "Automatycznie przekierowanie instancji (powrót do redirect.invidious.io): ",
|
"preferences_automatic_instance_redirect_label": "Automatycznie przekierowanie instancji (powrót do redirect.invidious.io): ",
|
||||||
"preferences_category_subscription": "Preferencje subskrybcji",
|
"preferences_category_subscription": "Preferencje subskrypcji",
|
||||||
"preferences_annotations_subscribed_label": "Domyślnie wyświetlaj adnotacje dla subskrybowanych kanałów: ",
|
"preferences_annotations_subscribed_label": "Domyślnie wyświetlaj adnotacje dla subskrybowanych kanałów: ",
|
||||||
"Redirect homepage to feed: ": "Przekieruj stronę główną do subskrybcji: ",
|
"Redirect homepage to feed: ": "Przekieruj stronę główną do subskrybcji: ",
|
||||||
"preferences_max_results_label": "Liczba filmów widoczna na stronie subskrybcji: ",
|
"preferences_max_results_label": "Liczba filmów widoczna na stronie subskrybcji: ",
|
||||||
|
@ -95,7 +95,7 @@
|
||||||
"Clear watch history": "Wyczyść historię",
|
"Clear watch history": "Wyczyść historię",
|
||||||
"Import/export data": "Import/Eksport danych",
|
"Import/export data": "Import/Eksport danych",
|
||||||
"Change password": "Zmień hasło",
|
"Change password": "Zmień hasło",
|
||||||
"Manage subscriptions": "Organizuj subskrybcje",
|
"Manage subscriptions": "Organizuj subskrypcje",
|
||||||
"Manage tokens": "Zarządzaj tokenami",
|
"Manage tokens": "Zarządzaj tokenami",
|
||||||
"Watch history": "Historia",
|
"Watch history": "Historia",
|
||||||
"Delete account": "Usuń konto",
|
"Delete account": "Usuń konto",
|
||||||
|
@ -115,7 +115,7 @@
|
||||||
"Import/export": "Import/Eksport",
|
"Import/export": "Import/Eksport",
|
||||||
"unsubscribe": "odsubskrybuj",
|
"unsubscribe": "odsubskrybuj",
|
||||||
"revoke": "cofnij",
|
"revoke": "cofnij",
|
||||||
"Subscriptions": "Subskrybcje",
|
"Subscriptions": "Subskrypcje",
|
||||||
"search": "szukaj",
|
"search": "szukaj",
|
||||||
"Log out": "Wyloguj",
|
"Log out": "Wyloguj",
|
||||||
"Source available here.": "Kod źródłowy dostępny tutaj.",
|
"Source available here.": "Kod źródłowy dostępny tutaj.",
|
||||||
|
@ -478,7 +478,7 @@
|
||||||
"search_filters_date_label": "Data przesłania",
|
"search_filters_date_label": "Data przesłania",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
"search_filters_date_option_none": "Dowolna data",
|
"search_filters_date_option_none": "Dowolna data",
|
||||||
"search_message_use_another_instance": " Możesz także <a href=\"`x`\">wyszukać w innej instancji</a>.",
|
"search_message_use_another_instance": "Możesz także <a href=\"`x`\">wyszukać w innej instancji</a>.",
|
||||||
"search_filters_type_option_all": "Dowolny typ",
|
"search_filters_type_option_all": "Dowolny typ",
|
||||||
"search_filters_duration_option_none": "Dowolna długość",
|
"search_filters_duration_option_none": "Dowolna długość",
|
||||||
"search_filters_duration_option_medium": "Średnia (4-20 minut)",
|
"search_filters_duration_option_medium": "Średnia (4-20 minut)",
|
||||||
|
@ -505,5 +505,13 @@
|
||||||
"generic_channels_count_1": "{{count}} kanały",
|
"generic_channels_count_1": "{{count}} kanały",
|
||||||
"generic_channels_count_2": "{{count}} kanałów",
|
"generic_channels_count_2": "{{count}} kanałów",
|
||||||
"Import YouTube watch history (.json)": "Importuj historię oglądania z YouTube (.json)",
|
"Import YouTube watch history (.json)": "Importuj historię oglądania z YouTube (.json)",
|
||||||
"toggle_theme": "Przełącz motyw"
|
"toggle_theme": "Przełącz motyw",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Kanał Popularne został wyłączony przez administratora.",
|
||||||
|
"Answer": "Odpowiedź",
|
||||||
|
"Search for videos": "Wyszukaj filmy",
|
||||||
|
"Add to playlist": "Dodaj do playlisty",
|
||||||
|
"Add to playlist: ": "Dodaj do playlisty: ",
|
||||||
|
"carousel_slide": "Slajd {{current}} z {{total}}",
|
||||||
|
"carousel_skip": "Pomiń karuzelę",
|
||||||
|
"carousel_go_to": "Przejdź do slajdu `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,27 +1,27 @@
|
||||||
{
|
{
|
||||||
"LIVE": "AO VIVO",
|
"LIVE": "AO VIVO",
|
||||||
"Shared `x` ago": "Compartilhado `x` atrás",
|
"Shared `x` ago": "Publicado há `x`",
|
||||||
"Unsubscribe": "Cancelar inscrição",
|
"Unsubscribe": "Cancelar inscrição",
|
||||||
"Subscribe": "Inscrever-se",
|
"Subscribe": "Inscrever-se",
|
||||||
"View channel on YouTube": "Ver canal no YouTube",
|
"View channel on YouTube": "Ver canal no YouTube",
|
||||||
"View playlist on YouTube": "Ver lista de reprodução no YouTube",
|
"View playlist on YouTube": "Ver playlist no YouTube",
|
||||||
"newest": "mais recentes",
|
"newest": "mais recentes",
|
||||||
"oldest": "mais antigos",
|
"oldest": "mais antigos",
|
||||||
"popular": "populares",
|
"popular": "populares",
|
||||||
"last": "último",
|
"last": "últimos",
|
||||||
"Next page": "Próxima página",
|
"Next page": "Próxima página",
|
||||||
"Previous page": "Página anterior",
|
"Previous page": "Página anterior",
|
||||||
"Clear watch history?": "Limpar histórico de reprodução?",
|
"Clear watch history?": "Limpar histórico de exibição?",
|
||||||
"New password": "Nova senha",
|
"New password": "Nova senha",
|
||||||
"New passwords must match": "Nova senha deve ser igual",
|
"New passwords must match": "As senhas devem ser iguais",
|
||||||
"Authorize token?": "Autorizar o token?",
|
"Authorize token?": "Autorizar token?",
|
||||||
"Authorize token for `x`?": "Autorizar o token para `x`?",
|
"Authorize token for `x`?": "Autorizar token para `x`?",
|
||||||
"Yes": "Sim",
|
"Yes": "Sim",
|
||||||
"No": "Não",
|
"No": "Não",
|
||||||
"Import and Export Data": "Importar e Exportar Dados",
|
"Import and Export Data": "Importar/exportar dados",
|
||||||
"Import": "Importar",
|
"Import": "Importar",
|
||||||
"Import Invidious data": "Importar dados em JSON do Invidious",
|
"Import Invidious data": "Importar dados JSON do Invidious",
|
||||||
"Import YouTube subscriptions": "Importar inscrições do YouTube/OPML",
|
"Import YouTube subscriptions": "Importar inscrições no formato CSV ou OPML do YouTube",
|
||||||
"Import FreeTube subscriptions (.db)": "Importar inscrições do FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importar inscrições do FreeTube (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importar inscrições do NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importar inscrições do NewPipe (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importar dados do NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importar dados do NewPipe (.zip)",
|
||||||
|
@ -32,49 +32,49 @@
|
||||||
"Delete account?": "Excluir conta?",
|
"Delete account?": "Excluir conta?",
|
||||||
"History": "Histórico",
|
"History": "Histórico",
|
||||||
"An alternative front-end to YouTube": "Uma interface alternativa para o YouTube",
|
"An alternative front-end to YouTube": "Uma interface alternativa para o YouTube",
|
||||||
"JavaScript license information": "Informação de licença do JavaScript",
|
"JavaScript license information": "Informações sobre a licença do JavaScript",
|
||||||
"source": "código-fonte",
|
"source": "fonte",
|
||||||
"Log in": "Entrar",
|
"Log in": "Fazer login",
|
||||||
"Log in/register": "Entrar/Registrar",
|
"Log in/register": "Fazer login/criar conta",
|
||||||
"User ID": "Usuário",
|
"User ID": "Usuário",
|
||||||
"Password": "Senha",
|
"Password": "Senha",
|
||||||
"Time (h:mm:ss):": "Hora (h:mm:ss):",
|
"Time (h:mm:ss):": "Hora (h:mm:ss):",
|
||||||
"Text CAPTCHA": "CAPTCHA em texto",
|
"Text CAPTCHA": "Mudar para um desafio de texto",
|
||||||
"Image CAPTCHA": "CAPTCHA em imagem",
|
"Image CAPTCHA": "Mudar para um desafio visual",
|
||||||
"Sign In": "Entrar",
|
"Sign In": "Fazer login",
|
||||||
"Register": "Registrar",
|
"Register": "Criar conta",
|
||||||
"E-mail": "E-mail",
|
"E-mail": "E-mail",
|
||||||
"Preferences": "Preferências",
|
"Preferences": "Preferências",
|
||||||
"preferences_category_player": "Preferências do reprodutor",
|
"preferences_category_player": "Preferências de reprodução",
|
||||||
"preferences_video_loop_label": "Repetir sempre: ",
|
"preferences_video_loop_label": "Repetir sempre: ",
|
||||||
"preferences_autoplay_label": "Reprodução automática: ",
|
"preferences_autoplay_label": "Reprodução automática: ",
|
||||||
"preferences_continue_label": "Sempre reproduzir próximo: ",
|
"preferences_continue_label": "Reproduzir a seguir, por padrão: ",
|
||||||
"preferences_continue_autoplay_label": "Reproduzir próximo vídeo automaticamente: ",
|
"preferences_continue_autoplay_label": "Reproduzir próximo vídeo automaticamente: ",
|
||||||
"preferences_listen_label": "Apenas áudio por padrão: ",
|
"preferences_listen_label": "Apenas áudio por padrão: ",
|
||||||
"preferences_local_label": "Usar proxy nos vídeos: ",
|
"preferences_local_label": "Usar proxy nos vídeos: ",
|
||||||
"preferences_speed_label": "Velocidade padrão: ",
|
"preferences_speed_label": "Velocidade padrão: ",
|
||||||
"preferences_quality_label": "Qualidade de vídeo preferida: ",
|
"preferences_quality_label": "Qualidade de vídeo preferida: ",
|
||||||
"preferences_volume_label": "Volume de reprodução: ",
|
"preferences_volume_label": "Volume de reprodução: ",
|
||||||
"preferences_comments_label": "Preferência de comentários: ",
|
"preferences_comments_label": "Comentários padrão: ",
|
||||||
"youtube": "YouTube",
|
"youtube": "YouTube",
|
||||||
"reddit": "Reddit",
|
"reddit": "Reddit",
|
||||||
"preferences_captions_label": "Preferência de legendas: ",
|
"preferences_captions_label": "Legendas padrão: ",
|
||||||
"Fallback captions: ": "Legendas alternativas: ",
|
"Fallback captions: ": "Legendas alternativas: ",
|
||||||
"preferences_related_videos_label": "Mostrar vídeos relacionados: ",
|
"preferences_related_videos_label": "Mostrar vídeos relacionados: ",
|
||||||
"preferences_annotations_label": "Sempre mostrar anotações: ",
|
"preferences_annotations_label": "Sempre mostrar anotações: ",
|
||||||
"preferences_extend_desc_label": "Estenda automaticamente a descrição do vídeo: ",
|
"preferences_extend_desc_label": "Expandir automaticamente a descrição do vídeo: ",
|
||||||
"preferences_vr_mode_label": "Vídeos interativos de 360 graus (requer WebGL): ",
|
"preferences_vr_mode_label": "Vídeos interativos de 360 graus (requer WebGL): ",
|
||||||
"preferences_category_visual": "Preferências visuais",
|
"preferences_category_visual": "Preferências visuais",
|
||||||
"preferences_player_style_label": "Estilo do tocador: ",
|
"preferences_player_style_label": "Estilo de reprodução: ",
|
||||||
"Dark mode: ": "Modo escuro: ",
|
"Dark mode: ": "Modo escuro: ",
|
||||||
"preferences_dark_mode_label": "Tema: ",
|
"preferences_dark_mode_label": "Tema: ",
|
||||||
"dark": "escuro",
|
"dark": "escuro",
|
||||||
"light": "claro",
|
"light": "claro",
|
||||||
"preferences_thin_mode_label": "Modo compacto: ",
|
"preferences_thin_mode_label": "Modo compacto: ",
|
||||||
"preferences_category_misc": "Preferências diversas",
|
"preferences_category_misc": "Preferências diversas",
|
||||||
"preferences_automatic_instance_redirect_label": "Redirecionamento de instância automática (fallback para redirect.invidious.io): ",
|
"preferences_automatic_instance_redirect_label": "Redirecionamento automático de instâncias (alternativa para redirect.invidious.io): ",
|
||||||
"preferences_category_subscription": "Preferências de inscrições",
|
"preferences_category_subscription": "Preferências de inscrições",
|
||||||
"preferences_annotations_subscribed_label": "Sempre mostrar anotações dos vídeos de canais inscritos: ",
|
"preferences_annotations_subscribed_label": "Mostrar anotações por padrão para canais inscritos? ",
|
||||||
"Redirect homepage to feed: ": "Redirecionar página inicial para o feed: ",
|
"Redirect homepage to feed: ": "Redirecionar página inicial para o feed: ",
|
||||||
"preferences_max_results_label": "Número de vídeos no feed: ",
|
"preferences_max_results_label": "Número de vídeos no feed: ",
|
||||||
"preferences_sort_label": "Ordenar vídeos por: ",
|
"preferences_sort_label": "Ordenar vídeos por: ",
|
||||||
|
@ -84,30 +84,30 @@
|
||||||
"alphabetically - reverse": "alfabética - ordem inversa",
|
"alphabetically - reverse": "alfabética - ordem inversa",
|
||||||
"channel name": "nome do canal",
|
"channel name": "nome do canal",
|
||||||
"channel name - reverse": "nome do canal - ordem inversa",
|
"channel name - reverse": "nome do canal - ordem inversa",
|
||||||
"Only show latest video from channel: ": "Mostrar apenas o vídeo mais recente do canal: ",
|
"Only show latest video from channel: ": "Mostrar apenas vídeos mais recentes do canal: ",
|
||||||
"Only show latest unwatched video from channel: ": "Mostrar apenas o vídeo mais recente não visualizado do canal: ",
|
"Only show latest unwatched video from channel: ": "Mostrar apenas vídeos mais recentes não assistido do canal: ",
|
||||||
"preferences_unseen_only_label": "Mostrar apenas vídeos não visualizados: ",
|
"preferences_unseen_only_label": "Mostrar apenas vídeos não assistido: ",
|
||||||
"preferences_notifications_only_label": "Mostrar apenas notificações (se existentes): ",
|
"preferences_notifications_only_label": "Mostrar apenas notificações (se houver): ",
|
||||||
"Enable web notifications": "Ativar notificações pela web",
|
"Enable web notifications": "Ativar notificações da Web",
|
||||||
"`x` uploaded a video": "`x` publicou um novo vídeo",
|
"`x` uploaded a video": "`x` publicou um vídeo",
|
||||||
"`x` is live": "`x` está ao vivo",
|
"`x` is live": "`x` está ao vivo",
|
||||||
"preferences_category_data": "Preferências de dados",
|
"preferences_category_data": "Preferências de dados",
|
||||||
"Clear watch history": "Limpar histórico de reprodução",
|
"Clear watch history": "Limpar histórico de exibição",
|
||||||
"Import/export data": "Importar/Exportar dados",
|
"Import/export data": "Importar/exportar dados",
|
||||||
"Change password": "Alterar senha",
|
"Change password": "Alterar senha",
|
||||||
"Manage subscriptions": "Gerenciar inscrições",
|
"Manage subscriptions": "Gerenciar inscrições",
|
||||||
"Manage tokens": "Gerenciar tokens",
|
"Manage tokens": "Gerenciar tokens",
|
||||||
"Watch history": "Histórico de reprodução",
|
"Watch history": "Histórico de exibição",
|
||||||
"Delete account": "Apagar sua conta",
|
"Delete account": "Excluir conta",
|
||||||
"preferences_category_admin": "Preferências de administrador",
|
"preferences_category_admin": "Preferências de administrador",
|
||||||
"preferences_default_home_label": "Página de início padrão: ",
|
"preferences_default_home_label": "Página inicial padrão: ",
|
||||||
"preferences_feed_menu_label": "Menu do feed: ",
|
"preferences_feed_menu_label": "Guias de feed preferidos: ",
|
||||||
"preferences_show_nick_label": "Mostrar o nickname no topo: ",
|
"preferences_show_nick_label": "Mostrar nome de usuário na parte superior: ",
|
||||||
"Top enabled: ": "Habilitar destaques: ",
|
"Top enabled: ": "Destaques ativados: ",
|
||||||
"CAPTCHA enabled: ": "Habilitar CAPTCHA: ",
|
"CAPTCHA enabled: ": "CAPTCHA ativado: ",
|
||||||
"Login enabled: ": "Habilitar login: ",
|
"Login enabled: ": "Fazer login ativado: ",
|
||||||
"Registration enabled: ": "Habilitar registro: ",
|
"Registration enabled: ": "Criar conta ativado: ",
|
||||||
"Report statistics: ": "Habilitar estatísticas: ",
|
"Report statistics: ": "Relatório de estatísticas: ",
|
||||||
"Save preferences": "Salvar preferências",
|
"Save preferences": "Salvar preferências",
|
||||||
"Subscription manager": "Gerenciador de inscrições",
|
"Subscription manager": "Gerenciador de inscrições",
|
||||||
"Token manager": "Gerenciador de tokens",
|
"Token manager": "Gerenciador de tokens",
|
||||||
|
@ -115,24 +115,24 @@
|
||||||
"tokens_count_0": "{{count}} token",
|
"tokens_count_0": "{{count}} token",
|
||||||
"tokens_count_1": "{{count}} tokens",
|
"tokens_count_1": "{{count}} tokens",
|
||||||
"tokens_count_2": "{{count}} tokens",
|
"tokens_count_2": "{{count}} tokens",
|
||||||
"Import/export": "Importar/Exportar",
|
"Import/export": "Importar/exportar",
|
||||||
"unsubscribe": "cancelar inscrição",
|
"unsubscribe": "cancelar inscrição",
|
||||||
"revoke": "revogar",
|
"revoke": "revogar",
|
||||||
"Subscriptions": "Inscrições",
|
"Subscriptions": "Inscrições",
|
||||||
"search": "Pesquisar",
|
"search": "pesquisar",
|
||||||
"Log out": "Sair",
|
"Log out": "Sair",
|
||||||
"Released under the AGPLv3 on Github.": "Lançado sob a AGPLv3 no GitHub.",
|
"Released under the AGPLv3 on Github.": "Lançado sob a AGPLv3 no GitHub.",
|
||||||
"Source available here.": "Código-fonte disponível aqui.",
|
"Source available here.": "Código-fonte disponível aqui.",
|
||||||
"View JavaScript license information.": "Ver informações da licença do JavaScript.",
|
"View JavaScript license information.": "Informações de licença JavaScript.",
|
||||||
"View privacy policy.": "Ver a política de privacidade.",
|
"View privacy policy.": "Política de privacidade.",
|
||||||
"Trending": "Tendências",
|
"Trending": "Em alta",
|
||||||
"Public": "Público",
|
"Public": "Público",
|
||||||
"Unlisted": "Não listado",
|
"Unlisted": "Não listado",
|
||||||
"Private": "Privado",
|
"Private": "Privado",
|
||||||
"View all playlists": "Mostrar todas listas de reprodução",
|
"View all playlists": "Ver todas as playlists",
|
||||||
"Updated `x` ago": "Atualizado `x` atrás",
|
"Updated `x` ago": "Atualizado `x` atrás",
|
||||||
"Delete playlist `x`?": "Apagar a playlist `x`?",
|
"Delete playlist `x`?": "Excluir playlist `x`?",
|
||||||
"Delete playlist": "Apagar playlist",
|
"Delete playlist": "Excluir playlist",
|
||||||
"Create playlist": "Criar playlist",
|
"Create playlist": "Criar playlist",
|
||||||
"Title": "Título",
|
"Title": "Título",
|
||||||
"Playlist privacy": "Privacidade da playlist",
|
"Playlist privacy": "Privacidade da playlist",
|
||||||
|
@ -140,24 +140,24 @@
|
||||||
"Show more": "Mostrar mais",
|
"Show more": "Mostrar mais",
|
||||||
"Show less": "Mostrar menos",
|
"Show less": "Mostrar menos",
|
||||||
"Watch on YouTube": "Assistir no YouTube",
|
"Watch on YouTube": "Assistir no YouTube",
|
||||||
"Switch Invidious Instance": "Mudar a instância do Invidious",
|
"Switch Invidious Instance": "Alterar instância Invidious",
|
||||||
"Hide annotations": "Ocultar anotações",
|
"Hide annotations": "Ocultar anotações",
|
||||||
"Show annotations": "Mostrar anotações",
|
"Show annotations": "Mostrar anotações",
|
||||||
"Genre: ": "Gênero: ",
|
"Genre: ": "Gênero: ",
|
||||||
"License: ": "Licença: ",
|
"License: ": "Licença: ",
|
||||||
"Family friendly? ": "Filtrar conteúdo impróprio: ",
|
"Family friendly? ": "Filtrar conteúdo impróprio: ",
|
||||||
"Wilson score: ": "Pontuação de Wilson: ",
|
"Wilson score: ": "Pontuação de Wilson: ",
|
||||||
"Engagement: ": "Empenho: ",
|
"Engagement: ": "Engajamento: ",
|
||||||
"Whitelisted regions: ": "Regiões permitidas: ",
|
"Whitelisted regions: ": "Regiões permitidas: ",
|
||||||
"Blacklisted regions: ": "Regiões bloqueadas: ",
|
"Blacklisted regions: ": "Regiões bloqueadas: ",
|
||||||
"Shared `x`": "Compartilhado `x`",
|
"Shared `x`": "Publicado em `x`",
|
||||||
"Premieres in `x`": "Estreia em `x`",
|
"Premieres in `x`": "Estreia em `x`",
|
||||||
"Premieres `x`": "Estreia `x`",
|
"Premieres `x`": "Estreia `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.": "Oi! Parece que seu JavaScript está desativado. Clique aqui para ver os comentários, entretanto eles podem levar um pouco mais de tempo para carregar.",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Olá! Parece que você está com o JavaScript desativado. Clique aqui para ver os comentários, mas lembre-se de que eles podem demorar um pouco mais para carregar.",
|
||||||
"View YouTube comments": "Ver comentários no YouTube",
|
"View YouTube comments": "Ver comentários no YouTube",
|
||||||
"View more comments on Reddit": "Ver mais comentários no Reddit",
|
"View more comments on Reddit": "Ver mais comentários no Reddit",
|
||||||
"View `x` comments": {
|
"View `x` comments": {
|
||||||
"([^.,0-9]|^)1([^.,0-9]|$)": "Ver `x` comentários",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Ver `x` comentário",
|
||||||
"": "Ver `x` comentários"
|
"": "Ver `x` comentários"
|
||||||
},
|
},
|
||||||
"View Reddit comments": "Ver comentários no Reddit",
|
"View Reddit comments": "Ver comentários no Reddit",
|
||||||
|
@ -166,7 +166,7 @@
|
||||||
"Incorrect password": "Senha incorreta",
|
"Incorrect password": "Senha incorreta",
|
||||||
"Wrong answer": "Resposta incorreta",
|
"Wrong answer": "Resposta incorreta",
|
||||||
"Erroneous CAPTCHA": "CAPTCHA inválido",
|
"Erroneous CAPTCHA": "CAPTCHA inválido",
|
||||||
"CAPTCHA is a required field": "O CAPTCHA é um campo obrigatório",
|
"CAPTCHA is a required field": "CAPTCHA é um campo obrigatório",
|
||||||
"User ID is a required field": "O nome de usuário é um campo obrigatório",
|
"User ID is a required field": "O nome de usuário é um campo obrigatório",
|
||||||
"Password is a required field": "A senha é um campo obrigatório",
|
"Password is a required field": "A senha é um campo obrigatório",
|
||||||
"Wrong username or password": "Nome de usuário ou senha inválidos",
|
"Wrong username or password": "Nome de usuário ou senha inválidos",
|
||||||
|
@ -175,17 +175,17 @@
|
||||||
"Please log in": "Por favor, inicie sua sessão",
|
"Please log in": "Por favor, inicie sua sessão",
|
||||||
"Invidious Private Feed for `x`": "Feed Privado do Invidious para `x`",
|
"Invidious Private Feed for `x`": "Feed Privado do Invidious para `x`",
|
||||||
"channel:`x`": "canal: `x`",
|
"channel:`x`": "canal: `x`",
|
||||||
"Deleted or invalid channel": "Este canal foi apagado ou é inválido",
|
"Deleted or invalid channel": "Canal excluído ou inválido",
|
||||||
"This channel does not exist.": "Este canal não existe.",
|
"This channel does not exist.": "Este canal não existe.",
|
||||||
"Could not get channel info.": "Não foi possível obter as informações do canal.",
|
"Could not get channel info.": "Não foi possível obter as informações do canal.",
|
||||||
"Could not fetch comments": "Não foi possível obter os comentários",
|
"Could not fetch comments": "Não foi possível obter os comentários",
|
||||||
"`x` ago": "`x` atrás",
|
"`x` ago": "`x` atrás",
|
||||||
"Load more": "Carregar mais",
|
"Load more": "Carregar mais",
|
||||||
"Could not create mix.": "Não foi possível criar o mix.",
|
"Could not create mix.": "Não foi possível criar o mix.",
|
||||||
"Empty playlist": "Lista de reprodução vazia",
|
"Empty playlist": "Playlist vazia",
|
||||||
"Not a playlist.": "Não é uma lista de reprodução.",
|
"Not a playlist.": "Não é uma playlist.",
|
||||||
"Playlist does not exist.": "A lista de reprodução não existe.",
|
"Playlist does not exist.": "A playlist não existe.",
|
||||||
"Could not pull trending pages.": "Não foi possível obter as páginas dos vídeos em alta.",
|
"Could not pull trending pages.": "Não foi possível obter as páginas de vídeos em alta.",
|
||||||
"Hidden field \"challenge\" is a required field": "O campo oculto \"desafio\" é obrigatório",
|
"Hidden field \"challenge\" is a required field": "O campo oculto \"desafio\" é obrigatório",
|
||||||
"Hidden field \"token\" is a required field": "O campo oculto \"token\" é obrigatório",
|
"Hidden field \"token\" is a required field": "O campo oculto \"token\" é obrigatório",
|
||||||
"Erroneous challenge": "Desafio inválido",
|
"Erroneous challenge": "Desafio inválido",
|
||||||
|
@ -319,87 +319,87 @@
|
||||||
"generic_count_seconds_0": "{{count}} segundo",
|
"generic_count_seconds_0": "{{count}} segundo",
|
||||||
"generic_count_seconds_1": "{{count}} segundos",
|
"generic_count_seconds_1": "{{count}} segundos",
|
||||||
"generic_count_seconds_2": "{{count}} segundos",
|
"generic_count_seconds_2": "{{count}} segundos",
|
||||||
"Fallback comments: ": "Comentários alternativos: ",
|
"Fallback comments: ": "Alternativa para comentários: ",
|
||||||
"Popular": "Populares",
|
"Popular": "Populares",
|
||||||
"Search": "Procurar",
|
"Search": "Pesquisar",
|
||||||
"Top": "No topo",
|
"Top": "Destaques",
|
||||||
"About": "Sobre",
|
"About": "Sobre",
|
||||||
"Rating: ": "Avaliação: ",
|
"Rating: ": "Avaliação: ",
|
||||||
"preferences_locale_label": "Idioma: ",
|
"preferences_locale_label": "Idioma: ",
|
||||||
"View as playlist": "Ver como lista de reprodução",
|
"View as playlist": "Ver como playlist",
|
||||||
"Default": "Padrão",
|
"Default": "Padrão",
|
||||||
"Music": "Músicas",
|
"Music": "Músicas",
|
||||||
"Gaming": "Jogos",
|
"Gaming": "Jogos",
|
||||||
"News": "Notícias",
|
"News": "Notícias",
|
||||||
"Movies": "Filmes",
|
"Movies": "Filmes",
|
||||||
"Download": "Baixar",
|
"Download": "Download",
|
||||||
"Download as: ": "Baixar como: ",
|
"Download as: ": "Baixar como: ",
|
||||||
"%A %B %-d, %Y": "%A %-d %B %Y",
|
"%A %B %-d, %Y": "%A %-d %B %Y",
|
||||||
"(edited)": "(editado)",
|
"(edited)": "(editado)",
|
||||||
"YouTube comment permalink": "Link permanente do comentário no YouTube",
|
"YouTube comment permalink": "Link permanente do comentário no YouTube",
|
||||||
"permalink": "Link permanente",
|
"permalink": "Link permanente",
|
||||||
"`x` marked it with a ❤": "`x` foi marcado como ❤",
|
"`x` marked it with a ❤": "`x` foi marcado com um ❤",
|
||||||
"Audio mode": "Modo de áudio",
|
"Audio mode": "Modo de áudio",
|
||||||
"Video mode": "Modo de vídeo",
|
"Video mode": "Modo de vídeo",
|
||||||
"channel_tab_videos_label": "Vídeos",
|
"channel_tab_videos_label": "Vídeos",
|
||||||
"Playlists": "Listas de reprodução",
|
"Playlists": "Playlists",
|
||||||
"channel_tab_community_label": "Comunidade",
|
"channel_tab_community_label": "Comunidade",
|
||||||
"search_filters_sort_option_relevance": "relevância",
|
"search_filters_sort_option_relevance": "Relevância",
|
||||||
"search_filters_sort_option_rating": "avaliação",
|
"search_filters_sort_option_rating": "Avaliação",
|
||||||
"search_filters_sort_option_date": "data",
|
"search_filters_sort_option_date": "Data de publicação",
|
||||||
"search_filters_sort_option_views": "visualizações",
|
"search_filters_sort_option_views": "Visualizações",
|
||||||
"search_filters_type_label": "content_type",
|
"search_filters_type_label": "Tipo",
|
||||||
"search_filters_duration_label": "duração",
|
"search_filters_duration_label": "Duração",
|
||||||
"search_filters_features_label": "recursos",
|
"search_filters_features_label": "Características",
|
||||||
"search_filters_sort_label": "ordenar",
|
"search_filters_sort_label": "Ordenar por",
|
||||||
"search_filters_date_option_hour": "hora",
|
"search_filters_date_option_hour": "Últimas horas",
|
||||||
"search_filters_date_option_today": "hoje",
|
"search_filters_date_option_today": "Hoje",
|
||||||
"search_filters_date_option_week": "semana",
|
"search_filters_date_option_week": "Esta semana",
|
||||||
"search_filters_date_option_month": "mês",
|
"search_filters_date_option_month": "Este mês",
|
||||||
"search_filters_date_option_year": "ano",
|
"search_filters_date_option_year": "Este ano",
|
||||||
"search_filters_type_option_video": "vídeo",
|
"search_filters_type_option_video": "Vídeo",
|
||||||
"search_filters_type_option_channel": "Canal",
|
"search_filters_type_option_channel": "Canal",
|
||||||
"search_filters_type_option_playlist": "playlist",
|
"search_filters_type_option_playlist": "Playlist",
|
||||||
"search_filters_type_option_movie": "filme",
|
"search_filters_type_option_movie": "Filme",
|
||||||
"search_filters_type_option_show": "show",
|
"search_filters_type_option_show": "Séries",
|
||||||
"search_filters_features_option_hd": "hd",
|
"search_filters_features_option_hd": "HD",
|
||||||
"search_filters_features_option_subtitles": "legendas",
|
"search_filters_features_option_subtitles": "Legendas",
|
||||||
"search_filters_features_option_c_commons": "creative_commons",
|
"search_filters_features_option_c_commons": "Creative Commons",
|
||||||
"search_filters_features_option_three_d": "3d",
|
"search_filters_features_option_three_d": "3D",
|
||||||
"search_filters_features_option_live": "ao vivo",
|
"search_filters_features_option_live": "AO VIVO",
|
||||||
"search_filters_features_option_four_k": "4k",
|
"search_filters_features_option_four_k": "4K",
|
||||||
"search_filters_features_option_location": "localização",
|
"search_filters_features_option_location": "Localização",
|
||||||
"search_filters_features_option_hdr": "hdr",
|
"search_filters_features_option_hdr": "HDR",
|
||||||
"Current version: ": "Versão atual: ",
|
"Current version: ": "Versão atual: ",
|
||||||
"next_steps_error_message": "Depois disso, você deve tentar: ",
|
"next_steps_error_message": "Depois disso, você deve tentar: ",
|
||||||
"next_steps_error_message_refresh": "Atualizar",
|
"next_steps_error_message_refresh": "Recarregar",
|
||||||
"next_steps_error_message_go_to_youtube": "Ir para o YouTube",
|
"next_steps_error_message_go_to_youtube": "Ir para o YouTube",
|
||||||
"footer_donate_page": "Doe",
|
"footer_donate_page": "Doar",
|
||||||
"adminprefs_modified_source_code_url_label": "URL para repositório de código fonte modificado",
|
"adminprefs_modified_source_code_url_label": "URL para o repositório do código-fonte modificado",
|
||||||
"search_filters_duration_option_long": "Longo (> 20 minutos)",
|
"search_filters_duration_option_long": "Longo (> 20 minutos)",
|
||||||
"search_filters_duration_option_short": "Curto (< 4 minutos)",
|
"search_filters_duration_option_short": "Curto (< 4 minutos)",
|
||||||
"footer_documentation": "Documentação",
|
"footer_documentation": "Documentação",
|
||||||
"footer_source_code": "Código fonte",
|
"footer_source_code": "Código-fonte",
|
||||||
"footer_original_source_code": "Código fonte original",
|
"footer_original_source_code": "Código-fonte original",
|
||||||
"footer_modfied_source_code": "Código-fonte modificado",
|
"footer_modfied_source_code": "Código-fonte modificado",
|
||||||
"preferences_quality_dash_label": "Qualidade de vídeo do painel preferida: ",
|
"preferences_quality_dash_label": "Qualidade de vídeo DASH preferida: ",
|
||||||
"preferences_region_label": "País do conteúdo: ",
|
"preferences_region_label": "País do conteúdo: ",
|
||||||
"preferences_quality_dash_option_4320p": "4320p",
|
"preferences_quality_dash_option_4320p": "4320p",
|
||||||
"generic_videos_count_0": "{{count}} vídeo",
|
"generic_videos_count_0": "{{count}} vídeo",
|
||||||
"generic_videos_count_1": "{{count}} vídeos",
|
"generic_videos_count_1": "{{count}} vídeos",
|
||||||
"generic_videos_count_2": "{{count}} vídeos",
|
"generic_videos_count_2": "{{count}} vídeos",
|
||||||
"generic_playlists_count_0": "{{count}} lista de reprodução",
|
"generic_playlists_count_0": "{{count}} playlist",
|
||||||
"generic_playlists_count_1": "{{count}} listas de reprodução",
|
"generic_playlists_count_1": "{{count}} playlists",
|
||||||
"generic_playlists_count_2": "{{count}} listas de reprodução",
|
"generic_playlists_count_2": "{{count}} playlists",
|
||||||
"generic_subscribers_count_0": "{{count}} inscrito",
|
"generic_subscribers_count_0": "{{count}} inscrito",
|
||||||
"generic_subscribers_count_1": "{{count}} inscritos",
|
"generic_subscribers_count_1": "{{count}} inscritos",
|
||||||
"generic_subscribers_count_2": "{{count}} inscritos",
|
"generic_subscribers_count_2": "{{count}} inscritos",
|
||||||
"generic_subscriptions_count_0": "{{count}} inscrição",
|
"generic_subscriptions_count_0": "{{count}} inscrição",
|
||||||
"generic_subscriptions_count_1": "{{count}} inscrições",
|
"generic_subscriptions_count_1": "{{count}} inscrições",
|
||||||
"generic_subscriptions_count_2": "{{count}} inscrições",
|
"generic_subscriptions_count_2": "{{count}} inscrições",
|
||||||
"subscriptions_unseen_notifs_count_0": "{{count}} notificação não vista",
|
"subscriptions_unseen_notifs_count_0": "{{count}} notificação não visualizada",
|
||||||
"subscriptions_unseen_notifs_count_1": "{{count}} notificações não vistas",
|
"subscriptions_unseen_notifs_count_1": "{{count}} notificações não visualizadas",
|
||||||
"subscriptions_unseen_notifs_count_2": "{{count}} notificações não vistas",
|
"subscriptions_unseen_notifs_count_2": "{{count}} notificações não visualizadas",
|
||||||
"comments_view_x_replies_0": "Ver {{count}} resposta",
|
"comments_view_x_replies_0": "Ver {{count}} resposta",
|
||||||
"comments_view_x_replies_1": "Ver {{count}} respostas",
|
"comments_view_x_replies_1": "Ver {{count}} respostas",
|
||||||
"comments_view_x_replies_2": "Ver {{count}} respostas",
|
"comments_view_x_replies_2": "Ver {{count}} respostas",
|
||||||
|
@ -407,14 +407,14 @@
|
||||||
"comments_points_count_1": "{{count}} pontos",
|
"comments_points_count_1": "{{count}} pontos",
|
||||||
"comments_points_count_2": "{{count}} pontos",
|
"comments_points_count_2": "{{count}} pontos",
|
||||||
"crash_page_you_found_a_bug": "Parece que você encontrou um erro no Invidious!",
|
"crash_page_you_found_a_bug": "Parece que você encontrou um erro no Invidious!",
|
||||||
"crash_page_before_reporting": "Antes de reportar um erro, verifique se você:",
|
"crash_page_before_reporting": "Antes de informar um erro, verifique se você:",
|
||||||
"preferences_save_player_pos_label": "Salvar a posição de reprodução: ",
|
"preferences_save_player_pos_label": "Salvar posição de reprodução: ",
|
||||||
"search_filters_features_option_purchased": "Comprado",
|
"search_filters_features_option_purchased": "Comprado",
|
||||||
"crash_page_refresh": "tentou <a href=\"`x`\">recarregar a página</a>",
|
"crash_page_refresh": "tentou <a href=\"`x`\">recarregar a página</a>",
|
||||||
"crash_page_switch_instance": "tentou <a href=\"`x`\">usar outra instância</a>",
|
"crash_page_switch_instance": "tentou <a href=\"`x`\">usar outra instância</a>",
|
||||||
"crash_page_search_issue": "procurou por um <a href=\"`x`\">erro existente no GitHub</a>",
|
"crash_page_search_issue": "procurou por um <a href=\"`x`\">erro existente no GitHub</a>",
|
||||||
"crash_page_report_issue": "Se nenhuma opção acima ajudou, por favor <a href=\"`x`\">abra um novo problema no Github</a> (preferencialmente em inglês) e inclua o seguinte texto (NÃO traduza):",
|
"crash_page_report_issue": "Se nenhuma opção acima ajudou, por favor <a href=\"`x`\">abra um novo problema no Github</a> (preferencialmente em inglês) e inclua o seguinte texto (NÃO traduza):",
|
||||||
"crash_page_read_the_faq": "leia as <a href=\"`x`\">Perguntas frequentes (FAQ)</a>",
|
"crash_page_read_the_faq": "leu as <a href=\"`x`\">Perguntas frequentes (FAQ)</a>",
|
||||||
"generic_views_count_0": "{{count}} visualização",
|
"generic_views_count_0": "{{count}} visualização",
|
||||||
"generic_views_count_1": "{{count}} visualizações",
|
"generic_views_count_1": "{{count}} visualizações",
|
||||||
"generic_views_count_2": "{{count}} visualizações",
|
"generic_views_count_2": "{{count}} visualizações",
|
||||||
|
@ -422,8 +422,8 @@
|
||||||
"preferences_quality_option_hd720": "HD720",
|
"preferences_quality_option_hd720": "HD720",
|
||||||
"preferences_quality_option_small": "Pequeno",
|
"preferences_quality_option_small": "Pequeno",
|
||||||
"preferences_quality_dash_option_auto": "Auto",
|
"preferences_quality_dash_option_auto": "Auto",
|
||||||
"preferences_quality_dash_option_best": "Melhor",
|
"preferences_quality_dash_option_best": "Melhor qualidade",
|
||||||
"preferences_quality_dash_option_worst": "Pior",
|
"preferences_quality_dash_option_worst": "Pior qualidade",
|
||||||
"preferences_quality_dash_option_2160p": "2160p",
|
"preferences_quality_dash_option_2160p": "2160p",
|
||||||
"preferences_quality_dash_option_1440p": "1440p",
|
"preferences_quality_dash_option_1440p": "1440p",
|
||||||
"preferences_quality_dash_option_1080p": "1080p",
|
"preferences_quality_dash_option_1080p": "1080p",
|
||||||
|
@ -435,17 +435,17 @@
|
||||||
"invidious": "Invidious",
|
"invidious": "Invidious",
|
||||||
"preferences_quality_option_medium": "Médio",
|
"preferences_quality_option_medium": "Médio",
|
||||||
"search_filters_features_option_three_sixty": "360°",
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
"none": "none",
|
"none": "nenhum",
|
||||||
"videoinfo_watch_on_youTube": "Assistir no YouTube",
|
"videoinfo_watch_on_youTube": "Assistir no YouTube",
|
||||||
"videoinfo_youTube_embed_link": "Embutir",
|
"videoinfo_youTube_embed_link": "Embed",
|
||||||
"videoinfo_invidious_embed_link": "Link Embutido",
|
"videoinfo_invidious_embed_link": "Embed link",
|
||||||
"download_subtitles": "Legendas - `x` (.vtt)",
|
"download_subtitles": "Legendas - `x` (.vtt)",
|
||||||
"user_created_playlists": "`x` listas de reprodução criadas",
|
"user_created_playlists": "`x` playlists criadas",
|
||||||
"user_saved_playlists": "`x` listas de reprodução salvas",
|
"user_saved_playlists": "`x` playlists salvas",
|
||||||
"Video unavailable": "Vídeo indisponível",
|
"Video unavailable": "Vídeo indisponível",
|
||||||
"videoinfo_started_streaming_x_ago": "Iniciou a transmissão a `x`",
|
"videoinfo_started_streaming_x_ago": "Iniciou a transmissão a `x`",
|
||||||
"search_filters_title": "Filtro",
|
"search_filters_title": "Filtro",
|
||||||
"preferences_watch_history_label": "Ative o histórico de exibição: ",
|
"preferences_watch_history_label": "Ativar histórico de exibição: ",
|
||||||
"search_message_no_results": "Nenhum resultado encontrado.",
|
"search_message_no_results": "Nenhum resultado encontrado.",
|
||||||
"search_message_change_filters_or_query": "Tente ampliar sua consulta de pesquisa e/ou alterar os filtros.",
|
"search_message_change_filters_or_query": "Tente ampliar sua consulta de pesquisa e/ou alterar os filtros.",
|
||||||
"English (United Kingdom)": "Inglês (Reino Unido)",
|
"English (United Kingdom)": "Inglês (Reino Unido)",
|
||||||
|
@ -465,7 +465,7 @@
|
||||||
"Portuguese (Brazil)": "Português (Brasil)",
|
"Portuguese (Brazil)": "Português (Brasil)",
|
||||||
"Russian (auto-generated)": "Russo (gerado automaticamente)",
|
"Russian (auto-generated)": "Russo (gerado automaticamente)",
|
||||||
"Vietnamese (auto-generated)": "Vietnamita (gerado automaticamente)",
|
"Vietnamese (auto-generated)": "Vietnamita (gerado automaticamente)",
|
||||||
"search_filters_date_label": "Data de upload",
|
"search_filters_date_label": "Data de publicação",
|
||||||
"search_filters_date_option_none": "Qualquer data",
|
"search_filters_date_option_none": "Qualquer data",
|
||||||
"Dutch (auto-generated)": "Holandês (gerado automaticamente)",
|
"Dutch (auto-generated)": "Holandês (gerado automaticamente)",
|
||||||
"French (auto-generated)": "Francês (gerado automaticamente)",
|
"French (auto-generated)": "Francês (gerado automaticamente)",
|
||||||
|
@ -474,26 +474,26 @@
|
||||||
"Spanish (auto-generated)": "Espanhol (gerado automaticamente)",
|
"Spanish (auto-generated)": "Espanhol (gerado automaticamente)",
|
||||||
"Spanish (Mexico)": "Espanhol (México)",
|
"Spanish (Mexico)": "Espanhol (México)",
|
||||||
"search_filters_duration_option_none": "Qualquer duração",
|
"search_filters_duration_option_none": "Qualquer duração",
|
||||||
"search_message_use_another_instance": " Você também pode <a href=\"`x`\">pesquisar em outra instância</a>.",
|
"search_message_use_another_instance": "Você também pode <a href=\"`x`\">pesquisar em outra instância</a>.",
|
||||||
"Spanish (Spain)": "Espanhol (Espanha)",
|
"Spanish (Spain)": "Espanhol (Espanha)",
|
||||||
"Turkish (auto-generated)": "Turco (gerado automaticamente)",
|
"Turkish (auto-generated)": "Turco (gerado automaticamente)",
|
||||||
"search_filters_duration_option_medium": "Médio (4 - 20 minutos)",
|
"search_filters_duration_option_medium": "Médio (4 - 20 minutos)",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
"Popular enabled: ": "Popular habilitado: ",
|
"Popular enabled: ": "Página \"Populares\" ativada: ",
|
||||||
"error_video_not_in_playlist": "O vídeo solicitado não existe nesta playlist. <a href=\"`x`\">Clique aqui para acessar a página inicial da playlist.</a>",
|
"error_video_not_in_playlist": "O vídeo solicitado não existe nesta playlist. <a href=\"`x`\">Clique aqui para acessar a página inicial da playlist.</a>",
|
||||||
"channel_tab_channels_label": "Canais",
|
"channel_tab_channels_label": "Canais",
|
||||||
"channel_tab_playlists_label": "Listas de reprodução",
|
"channel_tab_playlists_label": "Playlists",
|
||||||
"channel_tab_shorts_label": "Curtos",
|
"channel_tab_shorts_label": "Shorts",
|
||||||
"channel_tab_streams_label": "Ao Vivo",
|
"channel_tab_streams_label": "Transmissão ao vivo",
|
||||||
"Music in this video": "Música neste vídeo",
|
"Music in this video": "Música neste vídeo",
|
||||||
"Artist: ": "Artista: ",
|
"Artist: ": "Artista: ",
|
||||||
"Album: ": "Álbum: ",
|
"Album: ": "Álbum: ",
|
||||||
"Standard YouTube license": "Licença padrão do YouTube",
|
"Standard YouTube license": "Licença padrão do YouTube",
|
||||||
"Song: ": "Música: ",
|
"Song: ": "Música: ",
|
||||||
"Channel Sponsor": "Patrocinador do Canal",
|
"Channel Sponsor": "Patrocinador do canal",
|
||||||
"Download is disabled": "Download está desabilitado",
|
"Download is disabled": "Download indisponível",
|
||||||
"Import YouTube playlist (.csv)": "Importar lista de reprodução do YouTube (.csv)",
|
"Import YouTube playlist (.csv)": "Importar playlist do YouTube (.csv)",
|
||||||
"generic_button_delete": "Apagar",
|
"generic_button_delete": "Excluir",
|
||||||
"generic_button_save": "Salvar",
|
"generic_button_save": "Salvar",
|
||||||
"generic_button_edit": "Editar",
|
"generic_button_edit": "Editar",
|
||||||
"playlist_button_add_items": "Adicionar vídeos",
|
"playlist_button_add_items": "Adicionar vídeos",
|
||||||
|
@ -504,6 +504,14 @@
|
||||||
"generic_channels_count_0": "{{count}} canal",
|
"generic_channels_count_0": "{{count}} canal",
|
||||||
"generic_channels_count_1": "{{count}} canais",
|
"generic_channels_count_1": "{{count}} canais",
|
||||||
"generic_channels_count_2": "{{count}} canais",
|
"generic_channels_count_2": "{{count}} canais",
|
||||||
"Import YouTube watch history (.json)": "Importar histórico de reprodução do YouTube (.json)",
|
"Import YouTube watch history (.json)": "Importar histórico de exibição do YouTube (.json)",
|
||||||
"toggle_theme": "Alternar Tema"
|
"toggle_theme": "Alternar tema",
|
||||||
|
"Add to playlist": "Adicionar à playlist",
|
||||||
|
"Add to playlist: ": "Adicionar à playlist: ",
|
||||||
|
"Search for videos": "Pesquisar vídeos",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "O feed \"Populares\" foi desativado pelo administrador.",
|
||||||
|
"Answer": "Resposta",
|
||||||
|
"carousel_slide": "Slide {{current}} de {{total}}",
|
||||||
|
"carousel_skip": "Ignorar carrossel",
|
||||||
|
"carousel_go_to": "Ir ao slide `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -130,12 +130,12 @@
|
||||||
"Private": "Privado",
|
"Private": "Privado",
|
||||||
"View all playlists": "Ver todas as listas de reprodução",
|
"View all playlists": "Ver todas as listas de reprodução",
|
||||||
"Updated `x` ago": "Atualizado `x` atrás",
|
"Updated `x` ago": "Atualizado `x` atrás",
|
||||||
"Delete playlist `x`?": "Eliminar a lista de reprodução 'x'?",
|
"Delete playlist `x`?": "Eliminar a lista de reprodução `x`?",
|
||||||
"Delete playlist": "Eliminar lista de reprodução",
|
"Delete playlist": "Eliminar lista de reprodução",
|
||||||
"Create playlist": "Criar lista de reprodução",
|
"Create playlist": "Criar lista de reprodução",
|
||||||
"Title": "Título",
|
"Title": "Título",
|
||||||
"Playlist privacy": "Privacidade da lista de reprodução",
|
"Playlist privacy": "Privacidade da lista de reprodução",
|
||||||
"Editing playlist `x`": "A editar lista de reprodução 'x'",
|
"Editing playlist `x`": "A editar lista de reprodução `x`",
|
||||||
"Show more": "Mostrar mais",
|
"Show more": "Mostrar mais",
|
||||||
"Show less": "Mostrar menos",
|
"Show less": "Mostrar menos",
|
||||||
"Watch on YouTube": "Ver no YouTube",
|
"Watch on YouTube": "Ver no YouTube",
|
||||||
|
@ -150,8 +150,8 @@
|
||||||
"Whitelisted regions: ": "Regiões permitidas: ",
|
"Whitelisted regions: ": "Regiões permitidas: ",
|
||||||
"Blacklisted regions: ": "Regiões bloqueadas: ",
|
"Blacklisted regions: ": "Regiões bloqueadas: ",
|
||||||
"Shared `x`": "Partilhado `x`",
|
"Shared `x`": "Partilhado `x`",
|
||||||
"Premieres in `x`": "Estreias em 'x'",
|
"Premieres in `x`": "Estreias em `x`",
|
||||||
"Premieres `x`": "Estreias 'x'",
|
"Premieres `x`": "Estreias `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.": "Olá! Parece que o JavaScript está desativado. Clique aqui para ver os comentários, entretanto eles podem levar mais tempo para carregar.",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Olá! Parece que o JavaScript está desativado. Clique aqui para ver os comentários, entretanto eles podem levar mais tempo para carregar.",
|
||||||
"View YouTube comments": "Ver comentários do YouTube",
|
"View YouTube comments": "Ver comentários do YouTube",
|
||||||
"View more comments on Reddit": "Ver mais comentários no Reddit",
|
"View more comments on Reddit": "Ver mais comentários no Reddit",
|
||||||
|
@ -173,7 +173,7 @@
|
||||||
"Password cannot be longer than 55 characters": "A palavra-chave não pode ser superior a 55 caracteres",
|
"Password cannot be longer than 55 characters": "A palavra-chave não pode ser superior a 55 caracteres",
|
||||||
"Please log in": "Por favor, inicie sessão",
|
"Please log in": "Por favor, inicie sessão",
|
||||||
"Invidious Private Feed for `x`": "Feed Privado do Invidious para `x`",
|
"Invidious Private Feed for `x`": "Feed Privado do Invidious para `x`",
|
||||||
"channel:`x`": "canal:'x'",
|
"channel:`x`": "canal:`x`",
|
||||||
"Deleted or invalid channel": "Canal eliminado ou inválido",
|
"Deleted or invalid channel": "Canal eliminado ou inválido",
|
||||||
"This channel does not exist.": "Este canal não existe.",
|
"This channel does not exist.": "Este canal não existe.",
|
||||||
"Could not get channel info.": "Não foi possível obter as informações do canal.",
|
"Could not get channel info.": "Não foi possível obter as informações do canal.",
|
||||||
|
|
170
locales/pt.json
170
locales/pt.json
|
@ -1,25 +1,25 @@
|
||||||
{
|
{
|
||||||
"search_filters_type_option_show": "Série",
|
"search_filters_type_option_show": "Séries",
|
||||||
"search_filters_sort_option_views": "Visualizações",
|
"search_filters_sort_option_views": "Visualizações",
|
||||||
"search_filters_sort_option_date": "Data de carregamento",
|
"search_filters_sort_option_date": "Data de carregamento",
|
||||||
"search_filters_sort_option_rating": "Avaliação",
|
"search_filters_sort_option_rating": "Avaliação",
|
||||||
"search_filters_sort_option_relevance": "Relevância",
|
"search_filters_sort_option_relevance": "Relevância",
|
||||||
"Switch Invidious Instance": "Mudar a instância do Invidious",
|
"Switch Invidious Instance": "Alterar instância Invidious",
|
||||||
"Show less": "Mostrar menos",
|
"Show less": "Mostrar menos",
|
||||||
"Show more": "Mostrar mais",
|
"Show more": "Mostrar mais",
|
||||||
"Released under the AGPLv3 on Github.": "Lançado sob a AGPLv3 no GitHub.",
|
"Released under the AGPLv3 on Github.": "Disponibilizada sob a AGPLv3 no GitHub.",
|
||||||
"preferences_show_nick_label": "Mostrar nome de utilizador em cima: ",
|
"preferences_show_nick_label": "Mostrar nome de utilizador em cima: ",
|
||||||
"preferences_automatic_instance_redirect_label": "Redirecionamento de instância automática (solução de último recurso para redirect.invidious.io): ",
|
"preferences_automatic_instance_redirect_label": "Redirecionamento de instância automática (solução de último recurso para redirect.invidious.io): ",
|
||||||
"preferences_category_misc": "Preferências diversas",
|
"preferences_category_misc": "Preferências diversas",
|
||||||
"preferences_vr_mode_label": "Vídeos interativos de 360 graus (necessita de WebGL): ",
|
"preferences_vr_mode_label": "Vídeos interativos de 360 graus (requer WebGL): ",
|
||||||
"preferences_extend_desc_label": "Estender automaticamente a descrição do vídeo: ",
|
"preferences_extend_desc_label": "Expandir automaticamente a descrição do vídeo: ",
|
||||||
"next_steps_error_message_go_to_youtube": "Ir para o YouTube",
|
"next_steps_error_message_go_to_youtube": "Ir para o YouTube",
|
||||||
"next_steps_error_message": "Pode tentar as seguintes opções: ",
|
"next_steps_error_message": "Pode tentar as seguintes opções: ",
|
||||||
"next_steps_error_message_refresh": "Atualizar",
|
"next_steps_error_message_refresh": "Recarregar",
|
||||||
"search_filters_features_option_hdr": "HDR",
|
"search_filters_features_option_hdr": "HDR",
|
||||||
"search_filters_features_option_location": "Localização",
|
"search_filters_features_option_location": "Localização",
|
||||||
"search_filters_features_option_four_k": "4K",
|
"search_filters_features_option_four_k": "4K",
|
||||||
"search_filters_features_option_live": "Ao Vivo",
|
"search_filters_features_option_live": "Direto",
|
||||||
"search_filters_features_option_three_d": "3D",
|
"search_filters_features_option_three_d": "3D",
|
||||||
"search_filters_features_option_c_commons": "Creative Commons",
|
"search_filters_features_option_c_commons": "Creative Commons",
|
||||||
"search_filters_features_option_subtitles": "Legendas",
|
"search_filters_features_option_subtitles": "Legendas",
|
||||||
|
@ -37,11 +37,11 @@
|
||||||
"search_filters_features_label": "Funcionalidades",
|
"search_filters_features_label": "Funcionalidades",
|
||||||
"search_filters_duration_label": "Duração",
|
"search_filters_duration_label": "Duração",
|
||||||
"search_filters_type_label": "Tipo",
|
"search_filters_type_label": "Tipo",
|
||||||
"permalink": "hiperligação permanente",
|
"permalink": "ligação permanente",
|
||||||
"YouTube comment permalink": "Hiperligação permanente do comentário no YouTube",
|
"YouTube comment permalink": "Ligação permanente do comentário no YouTube",
|
||||||
"Download as: ": "Descarregar como: ",
|
"Download as: ": "Descarregar como: ",
|
||||||
"Download": "Descarregar",
|
"Download": "Descarregar",
|
||||||
"Default": "Predefinido",
|
"Default": "Padrão",
|
||||||
"Top": "Destaques",
|
"Top": "Destaques",
|
||||||
"Search": "Pesquisar",
|
"Search": "Pesquisar",
|
||||||
"generic_count_years_0": "{{count}} ano",
|
"generic_count_years_0": "{{count}} ano",
|
||||||
|
@ -67,21 +67,21 @@
|
||||||
"generic_count_seconds_2": "{{count}} segundos",
|
"generic_count_seconds_2": "{{count}} segundos",
|
||||||
"Chinese (Traditional)": "Chinês (tradicional)",
|
"Chinese (Traditional)": "Chinês (tradicional)",
|
||||||
"Chinese (Simplified)": "Chinês (simplificado)",
|
"Chinese (Simplified)": "Chinês (simplificado)",
|
||||||
"Could not pull trending pages.": "Não foi possível obter as páginas de tendências.",
|
"Could not pull trending pages.": "Não foi possível obter a página de tendências.",
|
||||||
"Could not create mix.": "Não foi possível criar a mistura.",
|
"Could not create mix.": "Não foi possível criar o mix.",
|
||||||
"Deleted or invalid channel": "Canal eliminado ou inválido",
|
"Deleted or invalid channel": "Canal eliminado ou inválido",
|
||||||
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Olá! Parece que o JavaScript está desativado. Clique aqui para ver os comentários, entretanto eles podem levar mais tempo para carregar.",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Olá! Parece que o JavaScript está desativado. Clique aqui para ver os comentários, mas tenha e conta que podem levar mais tempo para carregar.",
|
||||||
"Delete playlist": "Eliminar lista de reprodução",
|
"Delete playlist": "Eliminar lista de reprodução",
|
||||||
"Delete playlist `x`?": "Eliminar a lista de reprodução 'x'?",
|
"Delete playlist `x`?": "Eliminar lista de reprodução `x`?",
|
||||||
"search": "pesquisar",
|
"search": "pesquisar",
|
||||||
"unsubscribe": "anular subscrição",
|
"unsubscribe": "anular subscrição",
|
||||||
"Import/export": "Importar / exportar",
|
"Import/export": "Importar/exportar",
|
||||||
"Save preferences": "Guardar preferências",
|
"Save preferences": "Guardar preferências",
|
||||||
"Top enabled: ": "Destaques ativados: ",
|
"Top enabled: ": "Destaques ativados: ",
|
||||||
"Delete account": "Eliminar conta",
|
"Delete account": "Eliminar conta",
|
||||||
"Import/export data": "Importar / exportar dados",
|
"Import/export data": "Importar/exportar dados",
|
||||||
"preferences_annotations_label": "Mostrar anotações sempre: ",
|
"preferences_annotations_label": "Mostrar anotações sempre: ",
|
||||||
"preferences_continue_label": "Reproduzir sempre o próximo: ",
|
"preferences_continue_label": "Reproduzir sempre o seguinte: ",
|
||||||
"Sign In": "Entrar",
|
"Sign In": "Entrar",
|
||||||
"Log in/register": "Iniciar sessão/registar",
|
"Log in/register": "Iniciar sessão/registar",
|
||||||
"Delete account?": "Eliminar conta?",
|
"Delete account?": "Eliminar conta?",
|
||||||
|
@ -93,7 +93,7 @@
|
||||||
"Danish": "Dinamarquês",
|
"Danish": "Dinamarquês",
|
||||||
"Czech": "Checo",
|
"Czech": "Checo",
|
||||||
"Croatian": "Croata",
|
"Croatian": "Croata",
|
||||||
"Corsican": "Corso",
|
"Corsican": "Córsego",
|
||||||
"Cebuano": "Cebuano",
|
"Cebuano": "Cebuano",
|
||||||
"Catalan": "Catalão",
|
"Catalan": "Catalão",
|
||||||
"Burmese": "Birmanês",
|
"Burmese": "Birmanês",
|
||||||
|
@ -107,10 +107,10 @@
|
||||||
"Arabic": "Árabe",
|
"Arabic": "Árabe",
|
||||||
"Amharic": "Amárico",
|
"Amharic": "Amárico",
|
||||||
"Albanian": "Albanês",
|
"Albanian": "Albanês",
|
||||||
"Afrikaans": "Africano",
|
"Afrikaans": "Africânder",
|
||||||
"English (auto-generated)": "Inglês (auto-gerado)",
|
"English (auto-generated)": "Inglês (auto-gerado)",
|
||||||
"English": "Inglês",
|
"English": "Inglês",
|
||||||
"Token is expired, please try again": "Token expirou, tente novamente",
|
"Token is expired, please try again": "Token caducado, tente novamente",
|
||||||
"No such user": "Utilizador inválido",
|
"No such user": "Utilizador inválido",
|
||||||
"Erroneous token": "Token inválido",
|
"Erroneous token": "Token inválido",
|
||||||
"Erroneous challenge": "Desafio inválido",
|
"Erroneous challenge": "Desafio inválido",
|
||||||
|
@ -124,29 +124,29 @@
|
||||||
"Could not fetch comments": "Não foi possível obter os comentários",
|
"Could not fetch comments": "Não foi possível obter os comentários",
|
||||||
"Could not get channel info.": "Não foi possível obter as informações do canal.",
|
"Could not get channel info.": "Não foi possível obter as informações do canal.",
|
||||||
"This channel does not exist.": "Este canal não existe.",
|
"This channel does not exist.": "Este canal não existe.",
|
||||||
"channel:`x`": "canal:'x'",
|
"channel:`x`": "canal:`x`",
|
||||||
"Invidious Private Feed for `x`": "Feed Privado do Invidious para `x`",
|
"Invidious Private Feed for `x`": "Feed Privado do Invidious para `x`",
|
||||||
"Please log in": "Por favor, inicie sessão",
|
"Please log in": "Por favor, inicie sessão",
|
||||||
"Password cannot be longer than 55 characters": "A palavra-chave não pode ser superior a 55 caracteres",
|
"Password cannot be longer than 55 characters": "A palavra-passe não pode ter mais do que 55 caracteres",
|
||||||
"Password cannot be empty": "A palavra-chave não pode estar vazia",
|
"Password cannot be empty": "A palavra-passe não pode estar vazia",
|
||||||
"Wrong username or password": "Nome de utilizador ou palavra-chave incorreto",
|
"Wrong username or password": "Nome de utilizador ou palavra-passe incorreta",
|
||||||
"Password is a required field": "Palavra-chave é um campo obrigatório",
|
"Password is a required field": "Palavra-passe é um campo obrigatório",
|
||||||
"User ID is a required field": "O nome de utilizador é um campo obrigatório",
|
"User ID is a required field": "O nome de utilizador é um campo obrigatório",
|
||||||
"CAPTCHA is a required field": "CAPTCHA é um campo obrigatório",
|
"CAPTCHA is a required field": "CAPTCHA é um campo obrigatório",
|
||||||
"Erroneous CAPTCHA": "CAPTCHA inválido",
|
"Erroneous CAPTCHA": "CAPTCHA inválido",
|
||||||
"Wrong answer": "Resposta errada",
|
"Wrong answer": "Resposta errada",
|
||||||
"Incorrect password": "Palavra-chave incorreta",
|
"Incorrect password": "Palavra-passe incorreta",
|
||||||
"Show replies": "Mostrar respostas",
|
"Show replies": "Mostrar respostas",
|
||||||
"Hide replies": "Ocultar respostas",
|
"Hide replies": "Ocultar respostas",
|
||||||
"View Reddit comments": "Ver comentários do Reddit",
|
"View Reddit comments": "Ver comentários do Reddit",
|
||||||
"View `x` comments": {
|
"View `x` comments": {
|
||||||
"": "Ver `x` comentários",
|
"": "Ver `x` comentários",
|
||||||
"([^.,0-9]|^)1([^.,0-9]|$)": "Ver `x` comentários"
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Ver `x` comentário"
|
||||||
},
|
},
|
||||||
"View more comments on Reddit": "Ver mais comentários no Reddit",
|
"View more comments on Reddit": "Ver mais comentários no Reddit",
|
||||||
"View YouTube comments": "Ver comentários do YouTube",
|
"View YouTube comments": "Ver comentários do YouTube",
|
||||||
"Premieres `x`": "Estreias 'x'",
|
"Premieres `x`": "Estreia `x`",
|
||||||
"Premieres in `x`": "Estreias em 'x'",
|
"Premieres in `x`": "Estreia a `x`",
|
||||||
"Shared `x`": "Partilhado `x`",
|
"Shared `x`": "Partilhado `x`",
|
||||||
"Blacklisted regions: ": "Regiões bloqueadas: ",
|
"Blacklisted regions: ": "Regiões bloqueadas: ",
|
||||||
"Whitelisted regions: ": "Regiões permitidas: ",
|
"Whitelisted regions: ": "Regiões permitidas: ",
|
||||||
|
@ -158,44 +158,44 @@
|
||||||
"Show annotations": "Mostrar anotações",
|
"Show annotations": "Mostrar anotações",
|
||||||
"Hide annotations": "Ocultar anotações",
|
"Hide annotations": "Ocultar anotações",
|
||||||
"Watch on YouTube": "Ver no YouTube",
|
"Watch on YouTube": "Ver no YouTube",
|
||||||
"Editing playlist `x`": "A editar lista de reprodução 'x'",
|
"Editing playlist `x`": "A editar lista de reprodução `x`",
|
||||||
"Playlist privacy": "Privacidade da lista de reprodução",
|
"Playlist privacy": "Privacidade da lista de reprodução",
|
||||||
"Title": "Título",
|
"Title": "Título",
|
||||||
"Create playlist": "Criar lista de reprodução",
|
"Create playlist": "Criar lista de reprodução",
|
||||||
"Updated `x` ago": "Atualizado `x` atrás",
|
"Updated `x` ago": "Atualizado há `x`",
|
||||||
"View all playlists": "Ver todas as listas de reprodução",
|
"View all playlists": "Ver todas as listas de reprodução",
|
||||||
"Private": "Privado",
|
"Private": "Privado",
|
||||||
"Unlisted": "Não listado",
|
"Unlisted": "Não listado",
|
||||||
"Public": "Público",
|
"Public": "Público",
|
||||||
"Trending": "Tendências",
|
"Trending": "Tendências",
|
||||||
"View privacy policy.": "Ver a política de privacidade.",
|
"View privacy policy.": "Ver política de privacidade.",
|
||||||
"View JavaScript license information.": "Ver informações da licença do JavaScript.",
|
"View JavaScript license information.": "Ver informações da licença JavaScript.",
|
||||||
"Source available here.": "Código-fonte disponível aqui.",
|
"Source available here.": "Código-fonte disponível aqui.",
|
||||||
"Log out": "Terminar sessão",
|
"Log out": "Terminar sessão",
|
||||||
"Subscriptions": "Subscrições",
|
"Subscriptions": "Subscrições",
|
||||||
"revoke": "revogar",
|
"revoke": "revogar",
|
||||||
"tokens_count_0": "{{count}} Token",
|
"tokens_count_0": "{{count}} token",
|
||||||
"tokens_count_1": "{{count}} Tokens",
|
"tokens_count_1": "{{count}} tokens",
|
||||||
"tokens_count_2": "{{count}} Tokens",
|
"tokens_count_2": "{{count}} tokens",
|
||||||
"Token": "Token",
|
"Token": "Token",
|
||||||
"Token manager": "Gerir tokens",
|
"Token manager": "Gestor de tokens",
|
||||||
"Subscription manager": "Gerir subscrições",
|
"Subscription manager": "Gestor de subscrições",
|
||||||
"Report statistics: ": "Relatório de estatísticas: ",
|
"Report statistics: ": "Relatório de estatísticas: ",
|
||||||
"Registration enabled: ": "Registar ativado: ",
|
"Registration enabled: ": "Registar ativado: ",
|
||||||
"Login enabled: ": "Iniciar sessão ativado: ",
|
"Login enabled: ": "Iniciar sessão ativado: ",
|
||||||
"CAPTCHA enabled: ": "CAPTCHA ativado: ",
|
"CAPTCHA enabled: ": "CAPTCHA ativado: ",
|
||||||
"preferences_feed_menu_label": "Menu de subscrições: ",
|
"preferences_feed_menu_label": "Menu de subscrições: ",
|
||||||
"preferences_default_home_label": "Página inicial predefinida: ",
|
"preferences_default_home_label": "Página inicial padrão: ",
|
||||||
"preferences_category_admin": "Preferências de administrador",
|
"preferences_category_admin": "Preferências de administrador",
|
||||||
"Watch history": "Histórico de reprodução",
|
"Watch history": "Histórico de reprodução",
|
||||||
"Manage tokens": "Gerir tokens",
|
"Manage tokens": "Gerir tokens",
|
||||||
"Manage subscriptions": "Gerir as subscrições",
|
"Manage subscriptions": "Gerir subscrições",
|
||||||
"Change password": "Alterar palavra-chave",
|
"Change password": "Alterar palavra-passe",
|
||||||
"Clear watch history": "Limpar histórico de reprodução",
|
"Clear watch history": "Limpar histórico de reprodução",
|
||||||
"preferences_category_data": "Preferências de dados",
|
"preferences_category_data": "Preferências de dados",
|
||||||
"`x` is live": "`x` está em direto",
|
"`x` is live": "`x` está em direto",
|
||||||
"`x` uploaded a video": "`x` publicou um novo vídeo",
|
"`x` uploaded a video": "`x` publicou um vídeo",
|
||||||
"Enable web notifications": "Ativar notificações pela web",
|
"Enable web notifications": "Ativar notificações web",
|
||||||
"preferences_notifications_only_label": "Mostrar apenas notificações (se existirem): ",
|
"preferences_notifications_only_label": "Mostrar apenas notificações (se existirem): ",
|
||||||
"preferences_unseen_only_label": "Mostrar apenas vídeos não visualizados: ",
|
"preferences_unseen_only_label": "Mostrar apenas vídeos não visualizados: ",
|
||||||
"Only show latest unwatched video from channel: ": "Mostrar apenas vídeos mais recentes não visualizados do canal: ",
|
"Only show latest unwatched video from channel: ": "Mostrar apenas vídeos mais recentes não visualizados do canal: ",
|
||||||
|
@ -207,9 +207,9 @@
|
||||||
"published - reverse": "publicado - inverso",
|
"published - reverse": "publicado - inverso",
|
||||||
"published": "publicado",
|
"published": "publicado",
|
||||||
"preferences_sort_label": "Ordenar vídeos por: ",
|
"preferences_sort_label": "Ordenar vídeos por: ",
|
||||||
"preferences_max_results_label": "Quantidade de vídeos nas subscrições: ",
|
"preferences_max_results_label": "Número de vídeos nas subscrições: ",
|
||||||
"Redirect homepage to feed: ": "Redirecionar página inicial para subscrições: ",
|
"Redirect homepage to feed: ": "Redirecionar página inicial para subscrições: ",
|
||||||
"preferences_annotations_subscribed_label": "Mostrar sempre anotações aos canais subscritos: ",
|
"preferences_annotations_subscribed_label": "Mostrar sempre anotações nos canais subscritos: ",
|
||||||
"preferences_category_subscription": "Preferências de subscrições",
|
"preferences_category_subscription": "Preferências de subscrições",
|
||||||
"preferences_thin_mode_label": "Modo compacto: ",
|
"preferences_thin_mode_label": "Modo compacto: ",
|
||||||
"light": "claro",
|
"light": "claro",
|
||||||
|
@ -220,11 +220,11 @@
|
||||||
"preferences_category_visual": "Preferências visuais",
|
"preferences_category_visual": "Preferências visuais",
|
||||||
"preferences_related_videos_label": "Mostrar vídeos relacionados: ",
|
"preferences_related_videos_label": "Mostrar vídeos relacionados: ",
|
||||||
"Fallback captions: ": "Legendas alternativas: ",
|
"Fallback captions: ": "Legendas alternativas: ",
|
||||||
"preferences_captions_label": "Legendas predefinidas: ",
|
"preferences_captions_label": "Legendas padrão: ",
|
||||||
"reddit": "Reddit",
|
"reddit": "Reddit",
|
||||||
"youtube": "YouTube",
|
"youtube": "YouTube",
|
||||||
"preferences_comments_label": "Preferência dos comentários: ",
|
"preferences_comments_label": "Comentários padrão: ",
|
||||||
"preferences_volume_label": "Volume da reprodução: ",
|
"preferences_volume_label": "Volume de reprodução: ",
|
||||||
"preferences_quality_label": "Qualidade de vídeo preferida: ",
|
"preferences_quality_label": "Qualidade de vídeo preferida: ",
|
||||||
"preferences_speed_label": "Velocidade preferida: ",
|
"preferences_speed_label": "Velocidade preferida: ",
|
||||||
"preferences_local_label": "Usar proxy nos vídeos: ",
|
"preferences_local_label": "Usar proxy nos vídeos: ",
|
||||||
|
@ -239,11 +239,11 @@
|
||||||
"Image CAPTCHA": "Imagem CAPTCHA",
|
"Image CAPTCHA": "Imagem CAPTCHA",
|
||||||
"Text CAPTCHA": "Texto CAPTCHA",
|
"Text CAPTCHA": "Texto CAPTCHA",
|
||||||
"Time (h:mm:ss):": "Tempo (h:mm:ss):",
|
"Time (h:mm:ss):": "Tempo (h:mm:ss):",
|
||||||
"Password": "Palavra-chave",
|
"Password": "Palavra-passe",
|
||||||
"User ID": "Utilizador",
|
"User ID": "Utilizador",
|
||||||
"Log in": "Iniciar sessão",
|
"Log in": "Iniciar sessão",
|
||||||
"source": "código-fonte",
|
"source": "fonte",
|
||||||
"JavaScript license information": "Informação de licença do JavaScript",
|
"JavaScript license information": "Informação da licença JavaScript",
|
||||||
"An alternative front-end to YouTube": "Uma interface alternativa ao YouTube",
|
"An alternative front-end to YouTube": "Uma interface alternativa ao YouTube",
|
||||||
"History": "Histórico",
|
"History": "Histórico",
|
||||||
"Export data as JSON": "Exportar dados Invidious como JSON",
|
"Export data as JSON": "Exportar dados Invidious como JSON",
|
||||||
|
@ -253,18 +253,18 @@
|
||||||
"Import NewPipe data (.zip)": "Importar dados do NewPipe (.zip)",
|
"Import NewPipe data (.zip)": "Importar dados do NewPipe (.zip)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importar subscrições do NewPipe (.json)",
|
"Import NewPipe subscriptions (.json)": "Importar subscrições do NewPipe (.json)",
|
||||||
"Import FreeTube subscriptions (.db)": "Importar subscrições do FreeTube (.db)",
|
"Import FreeTube subscriptions (.db)": "Importar subscrições do FreeTube (.db)",
|
||||||
"Import YouTube subscriptions": "Importar subscrições do YouTube/OPML",
|
"Import YouTube subscriptions": "Importar via YouTube csv ou subscrição OPML",
|
||||||
"Import Invidious data": "Importar dados JSON do Invidious",
|
"Import Invidious data": "Importar dados JSON do Invidious",
|
||||||
"Import": "Importar",
|
"Import": "Importar",
|
||||||
"No": "Não",
|
"No": "Não",
|
||||||
"Yes": "Sim",
|
"Yes": "Sim",
|
||||||
"Authorize token for `x`?": "Autorizar token para `x`?",
|
"Authorize token for `x`?": "Autorizar 'token' para `x`?",
|
||||||
"Authorize token?": "Autorizar token?",
|
"Authorize token?": "Autorizar 'token'?",
|
||||||
"New passwords must match": "As novas palavra-chaves devem corresponder",
|
"New passwords must match": "As novas palavras-passe devem ser iguais",
|
||||||
"New password": "Nova palavra-chave",
|
"New password": "Nova palavra-passe",
|
||||||
"Clear watch history?": "Limpar histórico de reprodução?",
|
"Clear watch history?": "Limpar histórico de reprodução?",
|
||||||
"Previous page": "Página anterior",
|
"Previous page": "Página anterior",
|
||||||
"Next page": "Próxima página",
|
"Next page": "Página seguinte",
|
||||||
"last": "últimos",
|
"last": "últimos",
|
||||||
"Current version: ": "Versão atual: ",
|
"Current version: ": "Versão atual: ",
|
||||||
"channel_tab_community_label": "Comunidade",
|
"channel_tab_community_label": "Comunidade",
|
||||||
|
@ -272,19 +272,19 @@
|
||||||
"channel_tab_videos_label": "Vídeos",
|
"channel_tab_videos_label": "Vídeos",
|
||||||
"Video mode": "Modo de vídeo",
|
"Video mode": "Modo de vídeo",
|
||||||
"Audio mode": "Modo de áudio",
|
"Audio mode": "Modo de áudio",
|
||||||
"`x` marked it with a ❤": "`x` foi marcado como ❤",
|
"`x` marked it with a ❤": "`x` foi marcado com um ❤",
|
||||||
"(edited)": "(editado)",
|
"(edited)": "(editado)",
|
||||||
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
"%A %B %-d, %Y": "%A %B %-d, %Y",
|
||||||
"Movies": "Filmes",
|
"Movies": "Filmes",
|
||||||
"News": "Notícias",
|
"News": "Notícias",
|
||||||
"Gaming": "Jogos",
|
"Gaming": "Jogos",
|
||||||
"Music": "Música",
|
"Music": "Músicas",
|
||||||
"View as playlist": "Ver como lista de reprodução",
|
"View as playlist": "Ver como lista de reprodução",
|
||||||
"preferences_locale_label": "Idioma: ",
|
"preferences_locale_label": "Idioma: ",
|
||||||
"Rating: ": "Avaliação: ",
|
"Rating: ": "Avaliação: ",
|
||||||
"About": "Sobre",
|
"About": "Acerca",
|
||||||
"Popular": "Popular",
|
"Popular": "Popular",
|
||||||
"Fallback comments: ": "Comentários alternativos: ",
|
"Fallback comments: ": "Alternativa para comentários: ",
|
||||||
"Zulu": "Zulu",
|
"Zulu": "Zulu",
|
||||||
"Yoruba": "Ioruba",
|
"Yoruba": "Ioruba",
|
||||||
"Yiddish": "Iídiche",
|
"Yiddish": "Iídiche",
|
||||||
|
@ -329,7 +329,7 @@
|
||||||
"Marathi": "Marathi",
|
"Marathi": "Marathi",
|
||||||
"Maori": "Maori",
|
"Maori": "Maori",
|
||||||
"Maltese": "Maltês",
|
"Maltese": "Maltês",
|
||||||
"Malayalam": "Malaiala",
|
"Malayalam": "Malaialaio",
|
||||||
"Malay": "Malaio",
|
"Malay": "Malaio",
|
||||||
"Malagasy": "Malgaxe",
|
"Malagasy": "Malgaxe",
|
||||||
"Macedonian": "Macedónio",
|
"Macedonian": "Macedónio",
|
||||||
|
@ -365,15 +365,15 @@
|
||||||
"Galician": "Galego",
|
"Galician": "Galego",
|
||||||
"French": "Francês",
|
"French": "Francês",
|
||||||
"Finnish": "Finlandês",
|
"Finnish": "Finlandês",
|
||||||
"popular": "popular",
|
"popular": "populares",
|
||||||
"oldest": "mais antigos",
|
"oldest": "antigos",
|
||||||
"newest": "mais recentes",
|
"newest": "recentes",
|
||||||
"View playlist on YouTube": "Ver lista de reprodução no YouTube",
|
"View playlist on YouTube": "Ver lista de reprodução no YouTube",
|
||||||
"View channel on YouTube": "Ver canal no YouTube",
|
"View channel on YouTube": "Ver canal no YouTube",
|
||||||
"Subscribe": "Subscrever",
|
"Subscribe": "Subscrever",
|
||||||
"Unsubscribe": "Anular subscrição",
|
"Unsubscribe": "Anular subscrição",
|
||||||
"Shared `x` ago": "Partilhado `x` atrás",
|
"Shared `x` ago": "Partilhado `x` atrás",
|
||||||
"LIVE": "AO VIVO",
|
"LIVE": "Direto",
|
||||||
"search_filters_duration_option_short": "Curto (< 4 minutos)",
|
"search_filters_duration_option_short": "Curto (< 4 minutos)",
|
||||||
"search_filters_duration_option_long": "Longo (> 20 minutos)",
|
"search_filters_duration_option_long": "Longo (> 20 minutos)",
|
||||||
"footer_source_code": "Código-fonte",
|
"footer_source_code": "Código-fonte",
|
||||||
|
@ -386,7 +386,7 @@
|
||||||
"preferences_quality_dash_label": "Qualidade de vídeo DASH preferida: ",
|
"preferences_quality_dash_label": "Qualidade de vídeo DASH preferida: ",
|
||||||
"preferences_quality_option_small": "Baixa",
|
"preferences_quality_option_small": "Baixa",
|
||||||
"preferences_quality_option_hd720": "HD720",
|
"preferences_quality_option_hd720": "HD720",
|
||||||
"preferences_quality_dash_option_auto": "Automático",
|
"preferences_quality_dash_option_auto": "Automática",
|
||||||
"preferences_quality_dash_option_best": "Melhor",
|
"preferences_quality_dash_option_best": "Melhor",
|
||||||
"preferences_quality_dash_option_4320p": "4320p",
|
"preferences_quality_dash_option_4320p": "4320p",
|
||||||
"preferences_quality_dash_option_2160p": "2160p",
|
"preferences_quality_dash_option_2160p": "2160p",
|
||||||
|
@ -397,7 +397,7 @@
|
||||||
"preferences_quality_dash_option_144p": "144p",
|
"preferences_quality_dash_option_144p": "144p",
|
||||||
"search_filters_features_option_purchased": "Comprado",
|
"search_filters_features_option_purchased": "Comprado",
|
||||||
"search_filters_features_option_three_sixty": "360°",
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
"videoinfo_invidious_embed_link": "Incorporar hiperligação",
|
"videoinfo_invidious_embed_link": "Incorporar ligação",
|
||||||
"Video unavailable": "Vídeo não disponível",
|
"Video unavailable": "Vídeo não disponível",
|
||||||
"invidious": "Invidious",
|
"invidious": "Invidious",
|
||||||
"preferences_quality_option_medium": "Média",
|
"preferences_quality_option_medium": "Média",
|
||||||
|
@ -408,7 +408,7 @@
|
||||||
"preferences_quality_dash_option_worst": "Pior",
|
"preferences_quality_dash_option_worst": "Pior",
|
||||||
"none": "nenhum",
|
"none": "nenhum",
|
||||||
"videoinfo_youTube_embed_link": "Incorporar",
|
"videoinfo_youTube_embed_link": "Incorporar",
|
||||||
"preferences_save_player_pos_label": "Guardar a posição de reprodução atual do vídeo: ",
|
"preferences_save_player_pos_label": "Guardar posição de reprodução: ",
|
||||||
"download_subtitles": "Legendas - `x` (.vtt)",
|
"download_subtitles": "Legendas - `x` (.vtt)",
|
||||||
"generic_views_count_0": "{{count}} visualização",
|
"generic_views_count_0": "{{count}} visualização",
|
||||||
"generic_views_count_1": "{{count}} visualizações",
|
"generic_views_count_1": "{{count}} visualizações",
|
||||||
|
@ -427,12 +427,12 @@
|
||||||
"comments_view_x_replies_0": "Ver {{count}} resposta",
|
"comments_view_x_replies_0": "Ver {{count}} resposta",
|
||||||
"comments_view_x_replies_1": "Ver {{count}} respostas",
|
"comments_view_x_replies_1": "Ver {{count}} respostas",
|
||||||
"comments_view_x_replies_2": "Ver {{count}} respostas",
|
"comments_view_x_replies_2": "Ver {{count}} respostas",
|
||||||
"generic_subscribers_count_0": "{{count}} inscrito",
|
"generic_subscribers_count_0": "{{count}} subscritor",
|
||||||
"generic_subscribers_count_1": "{{count}} inscritos",
|
"generic_subscribers_count_1": "{{count}} subscritores",
|
||||||
"generic_subscribers_count_2": "{{count}} inscritos",
|
"generic_subscribers_count_2": "{{count}} subscritores",
|
||||||
"generic_subscriptions_count_0": "{{count}} inscrição",
|
"generic_subscriptions_count_0": "{{count}} subscrição",
|
||||||
"generic_subscriptions_count_1": "{{count}} inscrições",
|
"generic_subscriptions_count_1": "{{count}} subscrições",
|
||||||
"generic_subscriptions_count_2": "{{count}} inscrições",
|
"generic_subscriptions_count_2": "{{count}} subscrições",
|
||||||
"comments_points_count_0": "{{count}} ponto",
|
"comments_points_count_0": "{{count}} ponto",
|
||||||
"comments_points_count_1": "{{count}} pontos",
|
"comments_points_count_1": "{{count}} pontos",
|
||||||
"comments_points_count_2": "{{count}} pontos",
|
"comments_points_count_2": "{{count}} pontos",
|
||||||
|
@ -440,7 +440,7 @@
|
||||||
"crash_page_before_reporting": "Antes de reportar um erro, verifique se:",
|
"crash_page_before_reporting": "Antes de reportar um erro, verifique se:",
|
||||||
"crash_page_refresh": "tentou <a href=\"`x`\">recarregar a página</a>",
|
"crash_page_refresh": "tentou <a href=\"`x`\">recarregar a página</a>",
|
||||||
"crash_page_switch_instance": "tentou <a href=\"`x`\">usar outra instância</a>",
|
"crash_page_switch_instance": "tentou <a href=\"`x`\">usar outra instância</a>",
|
||||||
"crash_page_read_the_faq": "leia as <a href=\"`x`\">Perguntas frequentes (FAQ)</a>",
|
"crash_page_read_the_faq": "leu as <a href=\"`x`\">Perguntas frequentes (FAQ)</a>",
|
||||||
"crash_page_search_issue": "procurou se <a href=\"`x`\">o erro já foi reportado no GitHub</a>",
|
"crash_page_search_issue": "procurou se <a href=\"`x`\">o erro já foi reportado no GitHub</a>",
|
||||||
"crash_page_report_issue": "Se nenhuma opção acima ajudou, por favor <a href=\"`x`\">abra um novo problema no Github</a> (preferencialmente em inglês) e inclua o seguinte texto (NÃO o traduza):",
|
"crash_page_report_issue": "Se nenhuma opção acima ajudou, por favor <a href=\"`x`\">abra um novo problema no Github</a> (preferencialmente em inglês) e inclua o seguinte texto (NÃO o traduza):",
|
||||||
"user_created_playlists": "`x` listas de reprodução criadas",
|
"user_created_playlists": "`x` listas de reprodução criadas",
|
||||||
|
@ -448,7 +448,7 @@
|
||||||
"Chinese (Taiwan)": "Chinês (Taiwan)",
|
"Chinese (Taiwan)": "Chinês (Taiwan)",
|
||||||
"search_message_no_results": "Nenhum resultado encontrado.",
|
"search_message_no_results": "Nenhum resultado encontrado.",
|
||||||
"search_message_change_filters_or_query": "Tente alargar os termos genéricos da pesquisa e/ou alterar os filtros.",
|
"search_message_change_filters_or_query": "Tente alargar os termos genéricos da pesquisa e/ou alterar os filtros.",
|
||||||
"search_message_use_another_instance": " Também pode <a href=\"`x`\">pesquisar noutra instância</a>.",
|
"search_message_use_another_instance": "Também pode <a href=\"`x`\">pesquisar noutra instância</a>.",
|
||||||
"English (United Kingdom)": "Inglês (Reino Unido)",
|
"English (United Kingdom)": "Inglês (Reino Unido)",
|
||||||
"English (United States)": "Inglês (Estados Unidos)",
|
"English (United States)": "Inglês (Estados Unidos)",
|
||||||
"Cantonese (Hong Kong)": "Cantonês (Hong Kong)",
|
"Cantonese (Hong Kong)": "Cantonês (Hong Kong)",
|
||||||
|
@ -484,7 +484,7 @@
|
||||||
"channel_tab_playlists_label": "Listas de reprodução",
|
"channel_tab_playlists_label": "Listas de reprodução",
|
||||||
"channel_tab_channels_label": "Canais",
|
"channel_tab_channels_label": "Canais",
|
||||||
"channel_tab_shorts_label": "Curtos",
|
"channel_tab_shorts_label": "Curtos",
|
||||||
"channel_tab_streams_label": "Diretos",
|
"channel_tab_streams_label": "Emissões em direto",
|
||||||
"Music in this video": "Música neste vídeo",
|
"Music in this video": "Música neste vídeo",
|
||||||
"Artist: ": "Artista: ",
|
"Artist: ": "Artista: ",
|
||||||
"Album: ": "Álbum: ",
|
"Album: ": "Álbum: ",
|
||||||
|
@ -493,17 +493,25 @@
|
||||||
"Standard YouTube license": "Licença padrão do YouTube",
|
"Standard YouTube license": "Licença padrão do YouTube",
|
||||||
"Download is disabled": "A descarga está desativada",
|
"Download is disabled": "A descarga está desativada",
|
||||||
"Import YouTube playlist (.csv)": "Importar lista de reprodução do YouTube (.csv)",
|
"Import YouTube playlist (.csv)": "Importar lista de reprodução do YouTube (.csv)",
|
||||||
"generic_button_delete": "Deletar",
|
"generic_button_delete": "Eliminar",
|
||||||
"generic_button_edit": "Editar",
|
"generic_button_edit": "Editar",
|
||||||
"generic_button_rss": "RSS",
|
"generic_button_rss": "RSS",
|
||||||
"channel_tab_podcasts_label": "Podcasts",
|
"channel_tab_podcasts_label": "Podcasts",
|
||||||
"channel_tab_releases_label": "Lançamentos",
|
"channel_tab_releases_label": "Lançamentos",
|
||||||
"generic_button_save": "Salvar",
|
"generic_button_save": "Guardar",
|
||||||
"generic_button_cancel": "Cancelar",
|
"generic_button_cancel": "Cancelar",
|
||||||
"playlist_button_add_items": "Adicionar vídeos",
|
"playlist_button_add_items": "Adicionar vídeos",
|
||||||
"generic_channels_count_0": "{{count}} canal",
|
"generic_channels_count_0": "{{count}} canal",
|
||||||
"generic_channels_count_1": "{{count}} canais",
|
"generic_channels_count_1": "{{count}} canais",
|
||||||
"generic_channels_count_2": "{{count}} canais",
|
"generic_channels_count_2": "{{count}} canais",
|
||||||
"Import YouTube watch history (.json)": "Importar histórico de reprodução do YouTube (.json)",
|
"Import YouTube watch history (.json)": "Importar histórico de reprodução do YouTube (.json)",
|
||||||
"toggle_theme": "Trocar tema"
|
"toggle_theme": "Trocar tema",
|
||||||
|
"Add to playlist": "Adicionar à lista de reprodução",
|
||||||
|
"Add to playlist: ": "Adicionar à lista de reprodução: ",
|
||||||
|
"Answer": "Responder",
|
||||||
|
"Search for videos": "Procurar vídeos",
|
||||||
|
"carousel_slide": "Diapositivo {{current}} de{{total}}",
|
||||||
|
"carousel_skip": "Ignorar carrossel",
|
||||||
|
"carousel_go_to": "Ir para o diapositivo`x`",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "O feed Popular foi desativado por um administrador."
|
||||||
}
|
}
|
||||||
|
|
|
@ -478,5 +478,6 @@
|
||||||
"search_filters_type_option_all": "orice tip",
|
"search_filters_type_option_all": "orice tip",
|
||||||
"preferences_quality_dash_option_240p": "240p",
|
"preferences_quality_dash_option_240p": "240p",
|
||||||
"preferences_quality_dash_option_144p": "144p",
|
"preferences_quality_dash_option_144p": "144p",
|
||||||
"Show less": "Afișați mai puțin"
|
"Show less": "Afișați mai puțin",
|
||||||
|
"Add to playlist": "Adaugă la playlist"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Импорт и экспорт данных",
|
"Import and Export Data": "Импорт и экспорт данных",
|
||||||
"Import": "Импорт",
|
"Import": "Импорт",
|
||||||
"Import Invidious data": "Импортировать JSON с данными Invidious",
|
"Import Invidious data": "Импортировать JSON с данными Invidious",
|
||||||
"Import YouTube subscriptions": "Импортировать подписки из YouTube/OPML",
|
"Import YouTube subscriptions": "Импортировать подписки из CSV или OPML",
|
||||||
"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)",
|
||||||
|
@ -504,5 +504,14 @@
|
||||||
"generic_channels_count_0": "{{count}} канал",
|
"generic_channels_count_0": "{{count}} канал",
|
||||||
"generic_channels_count_1": "{{count}} канала",
|
"generic_channels_count_1": "{{count}} канала",
|
||||||
"generic_channels_count_2": "{{count}} каналов",
|
"generic_channels_count_2": "{{count}} каналов",
|
||||||
"Import YouTube watch history (.json)": "Импортировать историю просмотра из YouTube (.json)"
|
"Import YouTube watch history (.json)": "Импортировать историю просмотра из YouTube (.json)",
|
||||||
|
"Add to playlist": "Добавить в плейлист",
|
||||||
|
"Add to playlist: ": "Добавить в плейлист: ",
|
||||||
|
"Answer": "Ответить",
|
||||||
|
"Search for videos": "Поиск видео",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Лента популярного была отключена администратором.",
|
||||||
|
"toggle_theme": "Переключатель тем",
|
||||||
|
"carousel_slide": "Пролистано {{current}} из {{total}}",
|
||||||
|
"carousel_skip": "Пропустить всё",
|
||||||
|
"carousel_go_to": "Перейти к странице `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -257,13 +257,13 @@
|
||||||
"Video mode": "Mënyrë video",
|
"Video mode": "Mënyrë video",
|
||||||
"channel_tab_videos_label": "Video",
|
"channel_tab_videos_label": "Video",
|
||||||
"search_filters_sort_option_rating": "Vlerësim",
|
"search_filters_sort_option_rating": "Vlerësim",
|
||||||
"search_filters_sort_option_date": "Datë Ngarkimi",
|
"search_filters_sort_option_date": "Datë ngarkimi",
|
||||||
"search_filters_sort_option_views": "Numër parjesh",
|
"search_filters_sort_option_views": "Numër parjesh",
|
||||||
"search_filters_type_label": "Lloj",
|
"search_filters_type_label": "Lloj",
|
||||||
"search_filters_duration_label": "Kohëzgjatje",
|
"search_filters_duration_label": "Kohëzgjatje",
|
||||||
"search_filters_features_label": "Veçori",
|
"search_filters_features_label": "Veçori",
|
||||||
"search_filters_sort_label": "Renditi Sipas",
|
"search_filters_sort_label": "Renditi Sipas",
|
||||||
"search_filters_date_option_hour": "Orën e Fundit",
|
"search_filters_date_option_hour": "Orën e fundit",
|
||||||
"search_filters_date_option_today": "Sot",
|
"search_filters_date_option_today": "Sot",
|
||||||
"search_filters_duration_option_long": "E gjatë (> 20 minuta)",
|
"search_filters_duration_option_long": "E gjatë (> 20 minuta)",
|
||||||
"search_filters_features_option_hd": "HD",
|
"search_filters_features_option_hd": "HD",
|
||||||
|
@ -435,14 +435,14 @@
|
||||||
"tokens_count_plural": "{{count}} tokenë",
|
"tokens_count_plural": "{{count}} tokenë",
|
||||||
"preferences_save_player_pos_label": "Mba mend pozicionin e luajtjes: ",
|
"preferences_save_player_pos_label": "Mba mend pozicionin e luajtjes: ",
|
||||||
"Import Invidious data": "Importoni të dhëna JSON Invidious",
|
"Import Invidious data": "Importoni të dhëna JSON Invidious",
|
||||||
"Import YouTube subscriptions": "Importoni pajtime YouTube/OPML",
|
"Import YouTube subscriptions": "Importoni pajtime YouTube CSV ose OPML",
|
||||||
"Export data as JSON": "Eksportoji të dhënat Invidious si JSON",
|
"Export data as JSON": "Eksportoji të dhënat Invidious si JSON",
|
||||||
"preferences_vr_mode_label": "Video me ndërveprim 360 gradë (lyp WebGL): ",
|
"preferences_vr_mode_label": "Video me ndërveprim 360 gradë (lyp WebGL): ",
|
||||||
"Shared `x`": "Ndarë me të tjerë më `x`",
|
"Shared `x`": "Ndarë me të tjerë më `x`",
|
||||||
"search_filters_title": "Filtra",
|
"search_filters_title": "Filtra",
|
||||||
"Popular enabled: ": "Me populloret të aktivizuara: ",
|
"Popular enabled: ": "Me populloret të aktivizuara: ",
|
||||||
"error_video_not_in_playlist": "Videoja e kërkuar s’ekziston në këtë luajlistë. <a href=\"`x`\">Klikoni këtu për faqen hyrëse të luajlistës.</a>",
|
"error_video_not_in_playlist": "Videoja e kërkuar s’ekziston në këtë luajlistë. <a href=\"`x`\">Klikoni këtu për faqen hyrëse të luajlistës.</a>",
|
||||||
"search_message_use_another_instance": " Mundeni edhe të <a href=\"`x`\">kërkoni në një instancë tjetër</a>.",
|
"search_message_use_another_instance": "Mundeni edhe të <a href=\"`x`\">kërkoni në një instancë tjetër</a>.",
|
||||||
"search_filters_date_label": "Datë ngarkimi",
|
"search_filters_date_label": "Datë ngarkimi",
|
||||||
"preferences_watch_history_label": "Aktivizo historik parjesh: ",
|
"preferences_watch_history_label": "Aktivizo historik parjesh: ",
|
||||||
"Top enabled: ": "Me kryesueset të aktivizuara: ",
|
"Top enabled: ": "Me kryesueset të aktivizuara: ",
|
||||||
|
@ -484,5 +484,13 @@
|
||||||
"Import YouTube watch history (.json)": "Importo historik parjesh YouTube (.json)",
|
"Import YouTube watch history (.json)": "Importo historik parjesh YouTube (.json)",
|
||||||
"preferences_local_label": "Video përmes ndërmjetësi: ",
|
"preferences_local_label": "Video përmes ndërmjetësi: ",
|
||||||
"Fallback captions: ": "Titra nga halli: ",
|
"Fallback captions: ": "Titra nga halli: ",
|
||||||
"Erroneous challenge": "Zgjidhje e gabuar"
|
"Erroneous challenge": "Zgjidhje e gabuar",
|
||||||
|
"Add to playlist: ": "Shtoje te luajlistë: ",
|
||||||
|
"Add to playlist": "Shtoje te luajlistë",
|
||||||
|
"Answer": "Përgjigje",
|
||||||
|
"Search for videos": "Kërko për video",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Prurja Popullore është çaktivizuar nga përgjegjësi.",
|
||||||
|
"carousel_skip": "Anashkaloje Rrotullamen",
|
||||||
|
"carousel_slide": "Diapozitiv {{current}} nga {{total}}",
|
||||||
|
"carousel_go_to": "Kalo te diapozitivi `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Uvoz i izvoz podataka",
|
"Import and Export Data": "Uvoz i izvoz podataka",
|
||||||
"Import": "Uvezi",
|
"Import": "Uvezi",
|
||||||
"Import Invidious data": "Uvezi Invidious JSON podatke",
|
"Import Invidious data": "Uvezi Invidious JSON podatke",
|
||||||
"Import YouTube subscriptions": "Uvezi YouTube/OPML praćenja",
|
"Import YouTube subscriptions": "Uvezi YouTube CSV ili OPML praćenja",
|
||||||
"Import FreeTube subscriptions (.db)": "Uvezi FreeTube praćenja (.db)",
|
"Import FreeTube subscriptions (.db)": "Uvezi FreeTube praćenja (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Uvezi NewPipe praćenja (.json)",
|
"Import NewPipe subscriptions (.json)": "Uvezi NewPipe praćenja (.json)",
|
||||||
"Import NewPipe data (.zip)": "Uvezi NewPipe podatke (.zip)",
|
"Import NewPipe data (.zip)": "Uvezi NewPipe podatke (.zip)",
|
||||||
|
@ -174,7 +174,7 @@
|
||||||
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hej! Izgleda da ste isključili JavaScript. Kliknite ovde da biste videli komentare, imajte na umu da će možda potrajati malo duže da se učitaju.",
|
"Hi! Looks like you have JavaScript turned off. Click here to view comments, keep in mind they may take a bit longer to load.": "Hej! Izgleda da ste isključili JavaScript. Kliknite ovde da biste videli komentare, imajte na umu da će možda potrajati malo duže da se učitaju.",
|
||||||
"View `x` comments": {
|
"View `x` comments": {
|
||||||
"([^.,0-9]|^)1([^.,0-9]|$)": "Pogledaj `x` komentar",
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Pogledaj `x` komentar",
|
||||||
"": "Pogledaj`x` komentare"
|
"": "Pogledaj`x` komentara"
|
||||||
},
|
},
|
||||||
"View Reddit comments": "Pogledaj Reddit komentare",
|
"View Reddit comments": "Pogledaj Reddit komentare",
|
||||||
"CAPTCHA is a required field": "CAPTCHA je obavezno polje",
|
"CAPTCHA is a required field": "CAPTCHA je obavezno polje",
|
||||||
|
@ -211,7 +211,7 @@
|
||||||
"About": "O sajtu",
|
"About": "O sajtu",
|
||||||
"footer_source_code": "Izvorni kôd",
|
"footer_source_code": "Izvorni kôd",
|
||||||
"footer_original_source_code": "Originalni izvorni kôd",
|
"footer_original_source_code": "Originalni izvorni kôd",
|
||||||
"preferences_related_videos_label": "Prikaži povezane video snimke: ",
|
"preferences_related_videos_label": "Prikaži srodne video snimke: ",
|
||||||
"preferences_annotations_label": "Podrazumevano prikaži napomene: ",
|
"preferences_annotations_label": "Podrazumevano prikaži napomene: ",
|
||||||
"preferences_extend_desc_label": "Automatski proširi opis video snimka: ",
|
"preferences_extend_desc_label": "Automatski proširi opis video snimka: ",
|
||||||
"preferences_vr_mode_label": "Interaktivni video snimci od 360 stepeni (zahteva WebGl): ",
|
"preferences_vr_mode_label": "Interaktivni video snimci od 360 stepeni (zahteva WebGl): ",
|
||||||
|
@ -404,7 +404,7 @@
|
||||||
"generic_count_months_0": "{{count}} mesec",
|
"generic_count_months_0": "{{count}} mesec",
|
||||||
"generic_count_months_1": "{{count}} meseca",
|
"generic_count_months_1": "{{count}} meseca",
|
||||||
"generic_count_months_2": "{{count}} meseci",
|
"generic_count_months_2": "{{count}} meseci",
|
||||||
"search_message_use_another_instance": " Takođe, možete <a href=\"`x`\">pretraživati na drugoj instanci</a>.",
|
"search_message_use_another_instance": "Takođe, možete <a href=\"`x`\">pretraživati na drugoj instanci</a>.",
|
||||||
"generic_subscribers_count_0": "{{count}} pratilac",
|
"generic_subscribers_count_0": "{{count}} pratilac",
|
||||||
"generic_subscribers_count_1": "{{count}} pratioca",
|
"generic_subscribers_count_1": "{{count}} pratioca",
|
||||||
"generic_subscribers_count_2": "{{count}} pratilaca",
|
"generic_subscribers_count_2": "{{count}} pratilaca",
|
||||||
|
@ -504,5 +504,14 @@
|
||||||
"generic_views_count_0": "{{count}} pregled",
|
"generic_views_count_0": "{{count}} pregled",
|
||||||
"generic_views_count_1": "{{count}} pregleda",
|
"generic_views_count_1": "{{count}} pregleda",
|
||||||
"generic_views_count_2": "{{count}} pregleda",
|
"generic_views_count_2": "{{count}} pregleda",
|
||||||
"Import YouTube watch history (.json)": "Uvezi YouTube istoriju gledanja (.json)"
|
"Import YouTube watch history (.json)": "Uvezi YouTube istoriju gledanja (.json)",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Administrator je onemogućio fid „Popularno“.",
|
||||||
|
"Add to playlist: ": "Dodajte na plejlistu: ",
|
||||||
|
"Add to playlist": "Dodaj na plejlistu",
|
||||||
|
"carousel_slide": "Slajd {{current}} od {{total}}",
|
||||||
|
"carousel_go_to": "Idi na slajd `x`",
|
||||||
|
"Answer": "Odgovor",
|
||||||
|
"Search for videos": "Pretražite video snimke",
|
||||||
|
"carousel_skip": "Preskoči karusel",
|
||||||
|
"toggle_theme": "Подеси тему"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Увоз и извоз података",
|
"Import and Export Data": "Увоз и извоз података",
|
||||||
"Import": "Увези",
|
"Import": "Увези",
|
||||||
"Import Invidious data": "Увези Invidious JSON податке",
|
"Import Invidious data": "Увези Invidious JSON податке",
|
||||||
"Import YouTube subscriptions": "Увези YouTube/OPML праћења",
|
"Import YouTube subscriptions": "Увези YouTube CSV или OPML праћења",
|
||||||
"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)",
|
||||||
|
@ -60,7 +60,7 @@
|
||||||
"reddit": "Reddit",
|
"reddit": "Reddit",
|
||||||
"preferences_captions_label": "Подразумевани титлови: ",
|
"preferences_captions_label": "Подразумевани титлови: ",
|
||||||
"Fallback captions: ": "Резервни титлови: ",
|
"Fallback captions: ": "Резервни титлови: ",
|
||||||
"preferences_related_videos_label": "Прикажи повезане видео снимке: ",
|
"preferences_related_videos_label": "Прикажи сродне видео снимке: ",
|
||||||
"preferences_annotations_label": "Подразумевано прикажи напомене: ",
|
"preferences_annotations_label": "Подразумевано прикажи напомене: ",
|
||||||
"preferences_category_visual": "Визуелна подешавања",
|
"preferences_category_visual": "Визуелна подешавања",
|
||||||
"preferences_player_style_label": "Стил плејера: ",
|
"preferences_player_style_label": "Стил плејера: ",
|
||||||
|
@ -246,7 +246,7 @@
|
||||||
"preferences_locale_label": "Језик: ",
|
"preferences_locale_label": "Језик: ",
|
||||||
"Persian": "Персијски",
|
"Persian": "Персијски",
|
||||||
"View `x` comments": {
|
"View `x` comments": {
|
||||||
"": "Погледај `x` коментаре",
|
"": "Погледај `x` коментара",
|
||||||
"([^.,0-9]|^)1([^.,0-9]|$)": "Погледај `x` коментар"
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Погледај `x` коментар"
|
||||||
},
|
},
|
||||||
"search_filters_type_option_channel": "Канал",
|
"search_filters_type_option_channel": "Канал",
|
||||||
|
@ -404,7 +404,7 @@
|
||||||
"generic_count_months_0": "{{count}} месец",
|
"generic_count_months_0": "{{count}} месец",
|
||||||
"generic_count_months_1": "{{count}} месеца",
|
"generic_count_months_1": "{{count}} месеца",
|
||||||
"generic_count_months_2": "{{count}} месеци",
|
"generic_count_months_2": "{{count}} месеци",
|
||||||
"search_message_use_another_instance": " Такође, можете <a href=\"`x`\">претраживати на другој инстанци</a>.",
|
"search_message_use_another_instance": "Такође, можете <a href=\"`x`\">претраживати на другој инстанци</a>.",
|
||||||
"generic_subscribers_count_0": "{{count}} пратилац",
|
"generic_subscribers_count_0": "{{count}} пратилац",
|
||||||
"generic_subscribers_count_1": "{{count}} пратиоца",
|
"generic_subscribers_count_1": "{{count}} пратиоца",
|
||||||
"generic_subscribers_count_2": "{{count}} пратилаца",
|
"generic_subscribers_count_2": "{{count}} пратилаца",
|
||||||
|
@ -505,5 +505,13 @@
|
||||||
"generic_views_count_1": "{{count}} прегледа",
|
"generic_views_count_1": "{{count}} прегледа",
|
||||||
"generic_views_count_2": "{{count}} прегледа",
|
"generic_views_count_2": "{{count}} прегледа",
|
||||||
"Import YouTube watch history (.json)": "Увези YouTube историју гледањa (.json)",
|
"Import YouTube watch history (.json)": "Увези YouTube историју гледањa (.json)",
|
||||||
"toggle_theme": "Укључи тему"
|
"toggle_theme": "Укључи тему",
|
||||||
|
"Add to playlist": "Додај на плејлисту",
|
||||||
|
"Answer": "Одговор",
|
||||||
|
"Search for videos": "Претражите видео снимке",
|
||||||
|
"carousel_go_to": "Иди на слајд `x`",
|
||||||
|
"Add to playlist: ": "Додајте на плејлисту: ",
|
||||||
|
"carousel_skip": "Прескочи карусел",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Администратор је онемогућио фид „Популарно“.",
|
||||||
|
"carousel_slide": "Слајд {{current}} од {{total}}"
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Importera och exportera data",
|
"Import and Export Data": "Importera och exportera data",
|
||||||
"Import": "Importera",
|
"Import": "Importera",
|
||||||
"Import Invidious data": "Importera Invidious JSON data",
|
"Import Invidious data": "Importera Invidious JSON data",
|
||||||
"Import YouTube subscriptions": "Importera YouTube/OPML prenumerationer",
|
"Import YouTube subscriptions": "Importera YouTube CSV eller OPML prenumerationer",
|
||||||
"Import FreeTube subscriptions (.db)": "Importera FreeTube-prenumerationer (.db)",
|
"Import FreeTube subscriptions (.db)": "Importera FreeTube-prenumerationer (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "Importera NewPipe-prenumerationer (.json)",
|
"Import NewPipe subscriptions (.json)": "Importera NewPipe-prenumerationer (.json)",
|
||||||
"Import NewPipe data (.zip)": "Importera NewPipe-data (.zip)",
|
"Import NewPipe data (.zip)": "Importera NewPipe-data (.zip)",
|
||||||
|
@ -320,13 +320,13 @@
|
||||||
"channel_tab_community_label": "Gemenskap",
|
"channel_tab_community_label": "Gemenskap",
|
||||||
"search_filters_sort_option_relevance": "Relevans",
|
"search_filters_sort_option_relevance": "Relevans",
|
||||||
"search_filters_sort_option_rating": "Rankning",
|
"search_filters_sort_option_rating": "Rankning",
|
||||||
"search_filters_sort_option_date": "Uppladdnings Datum",
|
"search_filters_sort_option_date": "Uppladdnings datum",
|
||||||
"search_filters_sort_option_views": "Visningar",
|
"search_filters_sort_option_views": "Visningar",
|
||||||
"search_filters_type_label": "Typ",
|
"search_filters_type_label": "Typ",
|
||||||
"search_filters_duration_label": "Varaktighet",
|
"search_filters_duration_label": "Varaktighet",
|
||||||
"search_filters_features_label": "Funktioner",
|
"search_filters_features_label": "Funktioner",
|
||||||
"search_filters_sort_label": "Sortera efter",
|
"search_filters_sort_label": "Sortera efter",
|
||||||
"search_filters_date_option_hour": "Senaste Timmen",
|
"search_filters_date_option_hour": "Senaste timmen",
|
||||||
"search_filters_date_option_today": "Idag",
|
"search_filters_date_option_today": "Idag",
|
||||||
"search_filters_date_option_week": "Denna vecka",
|
"search_filters_date_option_week": "Denna vecka",
|
||||||
"search_filters_date_option_month": "Denna månad",
|
"search_filters_date_option_month": "Denna månad",
|
||||||
|
@ -393,7 +393,7 @@
|
||||||
"Artist: ": "Artist: ",
|
"Artist: ": "Artist: ",
|
||||||
"generic_count_months": "{{count}}månad",
|
"generic_count_months": "{{count}}månad",
|
||||||
"generic_count_months_plural": "{{count}}månader",
|
"generic_count_months_plural": "{{count}}månader",
|
||||||
"search_message_use_another_instance": " Du kan också <a href=\"`x`\">söka på en annan instans</a>.",
|
"search_message_use_another_instance": "Du kan också <a href=\"`x`\">söka på en annan instans</a>.",
|
||||||
"generic_subscribers_count": "{{count}} prenumerant",
|
"generic_subscribers_count": "{{count}} prenumerant",
|
||||||
"generic_subscribers_count_plural": "{{count}} prenumeranter",
|
"generic_subscribers_count_plural": "{{count}} prenumeranter",
|
||||||
"download_subtitles": "Undertexter - `x` (.vtt)",
|
"download_subtitles": "Undertexter - `x` (.vtt)",
|
||||||
|
@ -488,5 +488,13 @@
|
||||||
"crash_page_you_found_a_bug": "Det verkar som att du har hittat en bugg i Invidious!",
|
"crash_page_you_found_a_bug": "Det verkar som att du har hittat en bugg i Invidious!",
|
||||||
"generic_views_count": "{{count}} visning",
|
"generic_views_count": "{{count}} visning",
|
||||||
"generic_views_count_plural": "{{count}} visningar",
|
"generic_views_count_plural": "{{count}} visningar",
|
||||||
"toggle_theme": "Växla tema"
|
"toggle_theme": "Växla tema",
|
||||||
|
"Add to playlist": "Lägg till i spellista",
|
||||||
|
"Add to playlist: ": "Lägg till i spellista: ",
|
||||||
|
"Answer": "Svara",
|
||||||
|
"Search for videos": "Sök efter videor",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Det populära flödet har inaktiverats av administratören.",
|
||||||
|
"carousel_slide": "Bildspel {{current}} av {{total}}",
|
||||||
|
"carousel_skip": "Hoppa över karusellen",
|
||||||
|
"carousel_go_to": "Gå till bildspel `x`"
|
||||||
}
|
}
|
||||||
|
|
7
locales/tk.json
Normal file
7
locales/tk.json
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
{
|
||||||
|
"Add to playlist": "Aýdym sanawyna goş",
|
||||||
|
"Add to playlist: ": "Pleýliste goş: ",
|
||||||
|
"Answer": "Jogap",
|
||||||
|
"Search for videos": "Wideo gözläň",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Trende bolan administrator tarapyndan ýapyldy."
|
||||||
|
}
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Verileri İçe ve Dışa Aktar",
|
"Import and Export Data": "Verileri İçe ve Dışa Aktar",
|
||||||
"Import": "İçe Aktar",
|
"Import": "İçe Aktar",
|
||||||
"Import Invidious data": "Invidious JSON Verilerini İçe Aktar",
|
"Import Invidious data": "Invidious JSON Verilerini İçe Aktar",
|
||||||
"Import YouTube subscriptions": "YouTube/OPML Aboneliklerini İçe Aktar",
|
"Import YouTube subscriptions": "YouTube CSV veya OPML Aboneliklerini İçe Aktar",
|
||||||
"Import FreeTube subscriptions (.db)": "FreeTube Aboneliklerini İçe Aktar (.db)",
|
"Import FreeTube subscriptions (.db)": "FreeTube Aboneliklerini İçe Aktar (.db)",
|
||||||
"Import NewPipe subscriptions (.json)": "NewPipe Aboneliklerini İçe Aktar (.json)",
|
"Import NewPipe subscriptions (.json)": "NewPipe Aboneliklerini İçe Aktar (.json)",
|
||||||
"Import NewPipe data (.zip)": "NewPipe Verilerini İçe Aktar (.zip)",
|
"Import NewPipe data (.zip)": "NewPipe Verilerini İçe Aktar (.zip)",
|
||||||
|
@ -322,13 +322,13 @@
|
||||||
"channel_tab_community_label": "Topluluk",
|
"channel_tab_community_label": "Topluluk",
|
||||||
"search_filters_sort_option_relevance": "İlgi",
|
"search_filters_sort_option_relevance": "İlgi",
|
||||||
"search_filters_sort_option_rating": "Değerlendirme",
|
"search_filters_sort_option_rating": "Değerlendirme",
|
||||||
"search_filters_sort_option_date": "Yükleme Tarihi",
|
"search_filters_sort_option_date": "Yükleme tarihi",
|
||||||
"search_filters_sort_option_views": "Görüntüleme Sayısı",
|
"search_filters_sort_option_views": "Görüntüleme Sayısı",
|
||||||
"search_filters_type_label": "Tür",
|
"search_filters_type_label": "Tür",
|
||||||
"search_filters_duration_label": "Süre",
|
"search_filters_duration_label": "Süre",
|
||||||
"search_filters_features_label": "Özellikler",
|
"search_filters_features_label": "Özellikler",
|
||||||
"search_filters_sort_label": "Sıralama Ölçütü",
|
"search_filters_sort_label": "Sıralama Ölçütü",
|
||||||
"search_filters_date_option_hour": "Son Saat",
|
"search_filters_date_option_hour": "Son saat",
|
||||||
"search_filters_date_option_today": "Bugün",
|
"search_filters_date_option_today": "Bugün",
|
||||||
"search_filters_date_option_week": "Bu Hafta",
|
"search_filters_date_option_week": "Bu Hafta",
|
||||||
"search_filters_date_option_month": "Bu Ay",
|
"search_filters_date_option_month": "Bu Ay",
|
||||||
|
@ -452,7 +452,7 @@
|
||||||
"Spanish (Spain)": "İspanyolca (İspanya)",
|
"Spanish (Spain)": "İspanyolca (İspanya)",
|
||||||
"Vietnamese (auto-generated)": "Vietnamca (Otomatik Oluşturuldu)",
|
"Vietnamese (auto-generated)": "Vietnamca (Otomatik Oluşturuldu)",
|
||||||
"preferences_watch_history_label": "İzleme Geçmişini Etkinleştir: ",
|
"preferences_watch_history_label": "İzleme Geçmişini Etkinleştir: ",
|
||||||
"search_message_use_another_instance": " Ayrıca <a href=\"`x`\">başka bir örnekte arayabilirsiniz</a>.",
|
"search_message_use_another_instance": "Ayrıca <a href=\"`x`\">başka bir örnekte arayabilirsiniz</a>.",
|
||||||
"search_filters_type_option_all": "Herhangi Bir Tür",
|
"search_filters_type_option_all": "Herhangi Bir Tür",
|
||||||
"search_filters_duration_option_none": "Herhangi Bir Süre",
|
"search_filters_duration_option_none": "Herhangi Bir Süre",
|
||||||
"search_message_no_results": "Sonuç bulunamadı.",
|
"search_message_no_results": "Sonuç bulunamadı.",
|
||||||
|
@ -488,5 +488,13 @@
|
||||||
"generic_channels_count": "{{count}} kanal",
|
"generic_channels_count": "{{count}} kanal",
|
||||||
"generic_channels_count_plural": "{{count}} kanal",
|
"generic_channels_count_plural": "{{count}} kanal",
|
||||||
"Import YouTube watch history (.json)": "YouTube İzleme Geçmişini İçe Aktar (.json)",
|
"Import YouTube watch history (.json)": "YouTube İzleme Geçmişini İçe Aktar (.json)",
|
||||||
"toggle_theme": "Temayı Değiştir"
|
"toggle_theme": "Temayı Değiştir",
|
||||||
|
"Add to playlist": "Oynatma listesine ekle",
|
||||||
|
"Add to playlist: ": "Oynatma listesine ekle: ",
|
||||||
|
"Answer": "Yanıt",
|
||||||
|
"Search for videos": "Video ara",
|
||||||
|
"carousel_slide": "Sunum {{current}} / {{total}}",
|
||||||
|
"carousel_skip": "Kayar menüyü atla",
|
||||||
|
"carousel_go_to": "`x` sunumuna git",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Popüler akışı yönetici tarafından devre dışı bırakıldı."
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,7 @@
|
||||||
"Import and Export Data": "Імпорт і експорт даних",
|
"Import and Export Data": "Імпорт і експорт даних",
|
||||||
"Import": "Імпорт",
|
"Import": "Імпорт",
|
||||||
"Import Invidious data": "Імпортувати JSON-дані Invidious",
|
"Import Invidious data": "Імпортувати JSON-дані Invidious",
|
||||||
"Import YouTube subscriptions": "Імпортувати підписки з YouTube чи OPML",
|
"Import YouTube subscriptions": "Імпортувати підписки YouTube з CSV чи OPML",
|
||||||
"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)",
|
||||||
|
@ -127,7 +127,7 @@
|
||||||
"Create playlist": "Створити список відтворення",
|
"Create playlist": "Створити список відтворення",
|
||||||
"Title": "Заголовок",
|
"Title": "Заголовок",
|
||||||
"Playlist privacy": "Конфіденційність списку відтворення",
|
"Playlist privacy": "Конфіденційність списку відтворення",
|
||||||
"Editing playlist `x`": "Редагування списку відтворення \"x\"",
|
"Editing playlist `x`": "Редагування списку відтворення `x`",
|
||||||
"Watch on YouTube": "Дивитися на YouTube",
|
"Watch on YouTube": "Дивитися на YouTube",
|
||||||
"Hide annotations": "Приховати анотації",
|
"Hide annotations": "Приховати анотації",
|
||||||
"Show annotations": "Показати анотації",
|
"Show annotations": "Показати анотації",
|
||||||
|
@ -455,7 +455,7 @@
|
||||||
"search_filters_date_option_week": "Цей тиждень",
|
"search_filters_date_option_week": "Цей тиждень",
|
||||||
"search_filters_type_label": "Тип",
|
"search_filters_type_label": "Тип",
|
||||||
"search_filters_type_option_channel": "Канал",
|
"search_filters_type_option_channel": "Канал",
|
||||||
"search_message_use_another_instance": " Можете також <a href=\"`x`\">пошукати іншим сервером</a>.",
|
"search_message_use_another_instance": "Можете також <a href=\"`x`\">пошукати на іншому сервері</a>.",
|
||||||
"search_filters_title": "Фільтри",
|
"search_filters_title": "Фільтри",
|
||||||
"search_filters_date_option_hour": "Остання година",
|
"search_filters_date_option_hour": "Остання година",
|
||||||
"search_filters_date_option_month": "Цей місяць",
|
"search_filters_date_option_month": "Цей місяць",
|
||||||
|
@ -472,7 +472,7 @@
|
||||||
"search_filters_features_option_three_sixty": "360°",
|
"search_filters_features_option_three_sixty": "360°",
|
||||||
"search_filters_features_option_hdr": "HDR",
|
"search_filters_features_option_hdr": "HDR",
|
||||||
"search_filters_sort_label": "Спершу",
|
"search_filters_sort_label": "Спершу",
|
||||||
"search_filters_sort_option_date": "Нещодавні",
|
"search_filters_sort_option_date": "Дата вивантаження",
|
||||||
"search_filters_apply_button": "Застосувати фільтри",
|
"search_filters_apply_button": "Застосувати фільтри",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
"search_filters_features_option_purchased": "Придбано",
|
"search_filters_features_option_purchased": "Придбано",
|
||||||
|
@ -505,5 +505,13 @@
|
||||||
"generic_channels_count_1": "{{count}} канали",
|
"generic_channels_count_1": "{{count}} канали",
|
||||||
"generic_channels_count_2": "{{count}} каналів",
|
"generic_channels_count_2": "{{count}} каналів",
|
||||||
"Import YouTube watch history (.json)": "Імпортувати історію переглядів YouTube (.json)",
|
"Import YouTube watch history (.json)": "Імпортувати історію переглядів YouTube (.json)",
|
||||||
"toggle_theme": "Перемкнути тему"
|
"toggle_theme": "Перемкнути тему",
|
||||||
|
"Add to playlist": "Додати до списку відтворення",
|
||||||
|
"Add to playlist: ": "Додати до списку відтворення: ",
|
||||||
|
"Answer": "Відповідь",
|
||||||
|
"Search for videos": "Шукати відео",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Стрічка Популярні вимкнена адміністратором.",
|
||||||
|
"carousel_slide": "Слайд {{current}} з {{total}}",
|
||||||
|
"carousel_skip": "Пропустити карусель",
|
||||||
|
"carousel_go_to": "Перейти до слайда `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,12 +33,12 @@
|
||||||
"Export data as JSON": "Xuất dữ liệu Invidious dưới dạng JSON",
|
"Export data as JSON": "Xuất dữ liệu Invidious dưới dạng JSON",
|
||||||
"Delete account?": "Xóa tài khoản?",
|
"Delete account?": "Xóa tài khoản?",
|
||||||
"History": "Lịch sử",
|
"History": "Lịch sử",
|
||||||
"An alternative front-end to YouTube": "Một front-end thay thế cho YouTube",
|
"An alternative front-end to YouTube": "Giao diện thay thế cho YouTube",
|
||||||
"JavaScript license information": "Thông tin giấy phép JavaScript",
|
"JavaScript license information": "Thông tin giấy phép JavaScript",
|
||||||
"source": "nguồn",
|
"source": "nguồn",
|
||||||
"Log in": "Đăng nhập",
|
"Log in": "Đăng nhập",
|
||||||
"Log in/register": "Đăng nhập / đăng ký",
|
"Log in/register": "Đăng nhập / đăng ký",
|
||||||
"User ID": "ID người dùng",
|
"User ID": "Mã nhận dạng người dùng",
|
||||||
"Password": "Mật khẩu",
|
"Password": "Mật khẩu",
|
||||||
"Time (h:mm:ss):": "Thời gian (h:mm:ss):",
|
"Time (h:mm:ss):": "Thời gian (h:mm:ss):",
|
||||||
"Text CAPTCHA": "CAPTCHA dạng chữ",
|
"Text CAPTCHA": "CAPTCHA dạng chữ",
|
||||||
|
@ -46,16 +46,16 @@
|
||||||
"Sign In": "Đăng nhập",
|
"Sign In": "Đăng nhập",
|
||||||
"Register": "Đăng ký",
|
"Register": "Đăng ký",
|
||||||
"E-mail": "E-mail",
|
"E-mail": "E-mail",
|
||||||
"Preferences": "Sở thích",
|
"Preferences": "Cài đặt",
|
||||||
"preferences_category_player": "Tùy chọn trình phát video",
|
"preferences_category_player": "Tùy chọn trình phát video",
|
||||||
"preferences_video_loop_label": "Luôn lặp lại: ",
|
"preferences_video_loop_label": "Luôn lặp lại: ",
|
||||||
"preferences_autoplay_label": "Tự động phát: ",
|
"preferences_autoplay_label": "Tự động phát: ",
|
||||||
"preferences_continue_label": "Phát kế tiếp theo mặc định: ",
|
"preferences_continue_label": "Phát kế tiếp theo mặc định: ",
|
||||||
"preferences_continue_autoplay_label": "Tự động phát video tiếp theo: ",
|
"preferences_continue_autoplay_label": "Tự động phát video tiếp theo: ",
|
||||||
"preferences_listen_label": "Nghe theo mặc định: ",
|
"preferences_listen_label": "Nghe theo mặc định: ",
|
||||||
"preferences_local_label": "Video proxy: ",
|
"preferences_local_label": "Máy chủ sử lý video: ",
|
||||||
"preferences_speed_label": "Tốc độ mặc định: ",
|
"preferences_speed_label": "Tốc độ mặc định: ",
|
||||||
"preferences_quality_label": "Chất lượng video ưa thích: ",
|
"preferences_quality_label": "Chất lượng video: ",
|
||||||
"preferences_volume_label": "Âm lượng video: ",
|
"preferences_volume_label": "Âm lượng video: ",
|
||||||
"preferences_comments_label": "Nhận xét mặc định: ",
|
"preferences_comments_label": "Nhận xét mặc định: ",
|
||||||
"youtube": "YouTube",
|
"youtube": "YouTube",
|
||||||
|
@ -341,13 +341,13 @@
|
||||||
"([^.,0-9]|^)1([^.,0-9]|$)": "Hiển thị `x`bình luận"
|
"([^.,0-9]|^)1([^.,0-9]|$)": "Hiển thị `x`bình luận"
|
||||||
},
|
},
|
||||||
"Song: ": "Ca khúc: ",
|
"Song: ": "Ca khúc: ",
|
||||||
"Premieres in `x`": "Trình chiếu lần đầu vào `x`",
|
"Premieres in `x`": "Trình chiếu ở `x`",
|
||||||
"preferences_quality_dash_option_worst": "Thấp nhất",
|
"preferences_quality_dash_option_worst": "Tệ nhất",
|
||||||
"preferences_watch_history_label": "Bật lịch sử video đã xem ",
|
"preferences_watch_history_label": "Bật lịch sử video đã xem ",
|
||||||
"preferences_quality_option_hd720": "HD720",
|
"preferences_quality_option_hd720": "HD720",
|
||||||
"unsubscribe": "hủy đăng kí",
|
"unsubscribe": "hủy đăng kí",
|
||||||
"revoke": "gỡ bỏ",
|
"revoke": "gỡ bỏ",
|
||||||
"preferences_quality_dash_label": "Chất lượng video DASH ưa thích ",
|
"preferences_quality_dash_label": "Chất lượng video DASH ",
|
||||||
"preferences_quality_dash_option_auto": "Tự động",
|
"preferences_quality_dash_option_auto": "Tự động",
|
||||||
"Subscriptions": "Thuê bao",
|
"Subscriptions": "Thuê bao",
|
||||||
"View YouTube comments": "Hiển thị bình luận từ YouTube",
|
"View YouTube comments": "Hiển thị bình luận từ YouTube",
|
||||||
|
@ -470,5 +470,14 @@
|
||||||
"search_filters_duration_option_medium": "Trung bình (4 - 20 phút)",
|
"search_filters_duration_option_medium": "Trung bình (4 - 20 phút)",
|
||||||
"generic_count_seconds_0": "{{count}} giây",
|
"generic_count_seconds_0": "{{count}} giây",
|
||||||
"search_filters_date_label": "Ngày tải lên",
|
"search_filters_date_label": "Ngày tải lên",
|
||||||
"crash_page_you_found_a_bug": "Có vẻ như bạn đã tìm ra lỗi trong Indivious!"
|
"crash_page_you_found_a_bug": "Có vẻ như bạn đã tìm ra lỗi trong Indivious!",
|
||||||
|
"Add to playlist": "Thêm vào danh sách phát",
|
||||||
|
"Add to playlist: ": "Thêm vào danh sách phát: ",
|
||||||
|
"Answer": "Trả lời",
|
||||||
|
"toggle_theme": "Bật/tắt diện mạo",
|
||||||
|
"carousel_slide": "Trang {{current}} trên tổng {{total}} trang",
|
||||||
|
"carousel_skip": "Bỏ qua Carousel",
|
||||||
|
"carousel_go_to": "Đi tới trang `x`",
|
||||||
|
"Search for videos": "Tìm kiếm video",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "Bảng tin phổ biến đã bị tắt bởi ban quản lý."
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
"Import and Export Data": "导入与导出数据",
|
"Import and Export Data": "导入与导出数据",
|
||||||
"Import": "导入",
|
"Import": "导入",
|
||||||
"Import Invidious data": "导入 Invidious JSON 数据",
|
"Import Invidious data": "导入 Invidious JSON 数据",
|
||||||
"Import YouTube subscriptions": "导入 YouTube/OPML 订阅",
|
"Import YouTube subscriptions": "导入 YouTube CSV 或 OPML 订阅",
|
||||||
"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)",
|
||||||
|
@ -436,7 +436,7 @@
|
||||||
"Turkish (auto-generated)": "土耳其语 (自动生成)",
|
"Turkish (auto-generated)": "土耳其语 (自动生成)",
|
||||||
"Spanish (Spain)": "西班牙语 (西班牙)",
|
"Spanish (Spain)": "西班牙语 (西班牙)",
|
||||||
"preferences_watch_history_label": "启用观看历史: ",
|
"preferences_watch_history_label": "启用观看历史: ",
|
||||||
"search_message_use_another_instance": " 你也可以 <a href=\"`x`\">在另一实例上搜索</a>。",
|
"search_message_use_another_instance": "你也可以 <a href=\"`x`\">在另一实例上搜索</a>。",
|
||||||
"search_filters_title": "过滤器",
|
"search_filters_title": "过滤器",
|
||||||
"search_filters_date_label": "上传日期",
|
"search_filters_date_label": "上传日期",
|
||||||
"search_filters_apply_button": "应用所选过滤器",
|
"search_filters_apply_button": "应用所选过滤器",
|
||||||
|
@ -471,5 +471,13 @@
|
||||||
"generic_button_rss": "RSS",
|
"generic_button_rss": "RSS",
|
||||||
"channel_tab_releases_label": "公告",
|
"channel_tab_releases_label": "公告",
|
||||||
"generic_channels_count_0": "{{count}} 个频道",
|
"generic_channels_count_0": "{{count}} 个频道",
|
||||||
"toggle_theme": "切换主题"
|
"toggle_theme": "切换主题",
|
||||||
|
"Add to playlist": "添加到播放列表",
|
||||||
|
"Add to playlist: ": "添加到播放列表: ",
|
||||||
|
"Answer": "响应",
|
||||||
|
"Search for videos": "搜索视频",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "“流行”源已被管理员禁用。",
|
||||||
|
"carousel_slide": "当前为第 {{current}} 张图,共 {{total}} 张图",
|
||||||
|
"carousel_skip": "跳过图集",
|
||||||
|
"carousel_go_to": "转到图 `x`"
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
"Import and Export Data": "匯入與匯出資料",
|
"Import and Export Data": "匯入與匯出資料",
|
||||||
"Import": "匯入",
|
"Import": "匯入",
|
||||||
"Import Invidious data": "匯入 Invidious JSON 資料",
|
"Import Invidious data": "匯入 Invidious JSON 資料",
|
||||||
"Import YouTube subscriptions": "匯入 YouTube/OPML 訂閱",
|
"Import YouTube subscriptions": "匯入 YouTube CSV 或 OPML 訂閱",
|
||||||
"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)",
|
||||||
|
@ -338,13 +338,13 @@
|
||||||
"channel_tab_community_label": "社群",
|
"channel_tab_community_label": "社群",
|
||||||
"search_filters_sort_option_relevance": "關聯",
|
"search_filters_sort_option_relevance": "關聯",
|
||||||
"search_filters_sort_option_rating": "評分",
|
"search_filters_sort_option_rating": "評分",
|
||||||
"search_filters_sort_option_date": "日期",
|
"search_filters_sort_option_date": "上傳日期",
|
||||||
"search_filters_sort_option_views": "檢視",
|
"search_filters_sort_option_views": "檢視",
|
||||||
"search_filters_type_label": "內容類型",
|
"search_filters_type_label": "內容類型",
|
||||||
"search_filters_duration_label": "時長",
|
"search_filters_duration_label": "時長",
|
||||||
"search_filters_features_label": "特色",
|
"search_filters_features_label": "特色",
|
||||||
"search_filters_sort_label": "排序",
|
"search_filters_sort_label": "排序",
|
||||||
"search_filters_date_option_hour": "小時",
|
"search_filters_date_option_hour": "最後一小時",
|
||||||
"search_filters_date_option_today": "今天",
|
"search_filters_date_option_today": "今天",
|
||||||
"search_filters_date_option_week": "週",
|
"search_filters_date_option_week": "週",
|
||||||
"search_filters_date_option_month": "月",
|
"search_filters_date_option_month": "月",
|
||||||
|
@ -442,7 +442,7 @@
|
||||||
"search_filters_duration_option_none": "任何時長",
|
"search_filters_duration_option_none": "任何時長",
|
||||||
"search_filters_duration_option_medium": "中等(4到20分鐘)",
|
"search_filters_duration_option_medium": "中等(4到20分鐘)",
|
||||||
"search_filters_features_option_vr180": "VR180",
|
"search_filters_features_option_vr180": "VR180",
|
||||||
"search_message_use_another_instance": " 您也可以<a href=\"`x`\">在其他站台上搜尋</a>。",
|
"search_message_use_another_instance": "您也可以<a href=\"`x`\">在其他站台上搜尋</a>。",
|
||||||
"search_filters_title": "過濾條件",
|
"search_filters_title": "過濾條件",
|
||||||
"search_filters_date_label": "上傳日期",
|
"search_filters_date_label": "上傳日期",
|
||||||
"search_filters_type_option_all": "任何類型",
|
"search_filters_type_option_all": "任何類型",
|
||||||
|
@ -471,5 +471,13 @@
|
||||||
"channel_tab_podcasts_label": "Podcast",
|
"channel_tab_podcasts_label": "Podcast",
|
||||||
"channel_tab_releases_label": "發布",
|
"channel_tab_releases_label": "發布",
|
||||||
"generic_channels_count_0": "{{count}} 個頻道",
|
"generic_channels_count_0": "{{count}} 個頻道",
|
||||||
"toggle_theme": "切換佈景主題"
|
"toggle_theme": "切換佈景主題",
|
||||||
|
"Add to playlist": "新增至播放清單",
|
||||||
|
"Add to playlist: ": "新增至播放清單: ",
|
||||||
|
"Answer": "答案",
|
||||||
|
"Search for videos": "搜尋影片",
|
||||||
|
"carousel_slide": "第 {{current}} 張投影片,共 {{total}} 張",
|
||||||
|
"carousel_skip": "略過輪播",
|
||||||
|
"carousel_go_to": "跳到投影片 `x`",
|
||||||
|
"The Popular feed has been disabled by the administrator.": "熱門 feed 已被管理員停用。"
|
||||||
}
|
}
|
||||||
|
|
18
nginx.conf
Normal file
18
nginx.conf
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
user www-data;
|
||||||
|
events {
|
||||||
|
worker_connections 1024;
|
||||||
|
}
|
||||||
|
http {
|
||||||
|
server {
|
||||||
|
listen 3000;
|
||||||
|
listen [::]:3000;
|
||||||
|
access_log off;
|
||||||
|
location / {
|
||||||
|
resolver 127.0.0.11;
|
||||||
|
set $backend "invidious";
|
||||||
|
proxy_pass http://$backend:3000;
|
||||||
|
proxy_http_version 1.1; # to keep alive
|
||||||
|
proxy_set_header Connection ""; # to keep alive
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -2,7 +2,7 @@ version: 2.0
|
||||||
shards:
|
shards:
|
||||||
ameba:
|
ameba:
|
||||||
git: https://github.com/crystal-ameba/ameba.git
|
git: https://github.com/crystal-ameba/ameba.git
|
||||||
version: 1.5.0
|
version: 1.6.1
|
||||||
|
|
||||||
athena-negotiation:
|
athena-negotiation:
|
||||||
git: https://github.com/athena-framework/negotiation.git
|
git: https://github.com/athena-framework/negotiation.git
|
||||||
|
|
|
@ -37,7 +37,7 @@ development_dependencies:
|
||||||
version: ~> 0.10.4
|
version: ~> 0.10.4
|
||||||
ameba:
|
ameba:
|
||||||
github: crystal-ameba/ameba
|
github: crystal-ameba/ameba
|
||||||
version: ~> 1.5.0
|
version: ~> 1.6.1
|
||||||
|
|
||||||
crystal: ">= 1.0.0, < 2.0.0"
|
crystal: ">= 1.0.0, < 2.0.0"
|
||||||
|
|
||||||
|
|
|
@ -27,8 +27,8 @@ Spectator.describe Invidious::Hashtag do
|
||||||
expect(video_11.length_seconds).to eq((56.minutes + 41.seconds).total_seconds.to_i32)
|
expect(video_11.length_seconds).to eq((56.minutes + 41.seconds).total_seconds.to_i32)
|
||||||
expect(video_11.views).to eq(40_504_893)
|
expect(video_11.views).to eq(40_504_893)
|
||||||
|
|
||||||
expect(video_11.live_now).to be_false
|
expect(video_11.badges.live_now?).to be_false
|
||||||
expect(video_11.premium).to be_false
|
expect(video_11.badges.premium?).to be_false
|
||||||
expect(video_11.premiere_timestamp).to be_nil
|
expect(video_11.premiere_timestamp).to be_nil
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -49,8 +49,8 @@ Spectator.describe Invidious::Hashtag do
|
||||||
expect(video_35.length_seconds).to eq((3.minutes + 14.seconds).total_seconds.to_i32)
|
expect(video_35.length_seconds).to eq((3.minutes + 14.seconds).total_seconds.to_i32)
|
||||||
expect(video_35.views).to eq(30_790_049)
|
expect(video_35.views).to eq(30_790_049)
|
||||||
|
|
||||||
expect(video_35.live_now).to be_false
|
expect(video_35.badges.live_now?).to be_false
|
||||||
expect(video_35.premium).to be_false
|
expect(video_35.badges.premium?).to be_false
|
||||||
expect(video_35.premiere_timestamp).to be_nil
|
expect(video_35.premiere_timestamp).to be_nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -80,8 +80,8 @@ Spectator.describe Invidious::Hashtag do
|
||||||
expect(video_41.length_seconds).to eq((1.hour).total_seconds.to_i32)
|
expect(video_41.length_seconds).to eq((1.hour).total_seconds.to_i32)
|
||||||
expect(video_41.views).to eq(63_240)
|
expect(video_41.views).to eq(63_240)
|
||||||
|
|
||||||
expect(video_41.live_now).to be_false
|
expect(video_41.badges.live_now?).to be_false
|
||||||
expect(video_41.premium).to be_false
|
expect(video_41.badges.premium?).to be_false
|
||||||
expect(video_41.premiere_timestamp).to be_nil
|
expect(video_41.premiere_timestamp).to be_nil
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -102,8 +102,8 @@ Spectator.describe Invidious::Hashtag do
|
||||||
expect(video_48.length_seconds).to eq((35.minutes + 46.seconds).total_seconds.to_i32)
|
expect(video_48.length_seconds).to eq((35.minutes + 46.seconds).total_seconds.to_i32)
|
||||||
expect(video_48.views).to eq(68_704)
|
expect(video_48.views).to eq(68_704)
|
||||||
|
|
||||||
expect(video_48.live_now).to be_false
|
expect(video_48.badges.live_now?).to be_false
|
||||||
expect(video_48.premium).to be_false
|
expect(video_48.badges.premium?).to be_false
|
||||||
expect(video_48.premiere_timestamp).to be_nil
|
expect(video_48.premiere_timestamp).to be_nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -301,7 +301,6 @@ Spectator.describe Invidious::Search::Filters do
|
||||||
|
|
||||||
it "Encodes features filter (single)" do
|
it "Encodes features filter (single)" do
|
||||||
Invidious::Search::Filters::Features.each do |value|
|
Invidious::Search::Filters::Features.each do |value|
|
||||||
string = described_class.format_features(value)
|
|
||||||
filters = described_class.new(features: value)
|
filters = described_class.new(features: value)
|
||||||
|
|
||||||
expect("#{filters.to_iv_params}")
|
expect("#{filters.to_iv_params}")
|
||||||
|
|
|
@ -67,7 +67,7 @@ Spectator.describe "parse_video_info" do
|
||||||
# Video metadata
|
# Video metadata
|
||||||
|
|
||||||
expect(info["genre"].as_s).to eq("Entertainment")
|
expect(info["genre"].as_s).to eq("Entertainment")
|
||||||
expect(info["genreUcid"].as_s).to be_empty
|
expect(info["genreUcid"].as_s?).to be_nil
|
||||||
expect(info["license"].as_s).to be_empty
|
expect(info["license"].as_s).to be_empty
|
||||||
|
|
||||||
# Author infos
|
# Author infos
|
||||||
|
@ -151,7 +151,7 @@ Spectator.describe "parse_video_info" do
|
||||||
# Video metadata
|
# Video metadata
|
||||||
|
|
||||||
expect(info["genre"].as_s).to eq("Music")
|
expect(info["genre"].as_s).to eq("Music")
|
||||||
expect(info["genreUcid"].as_s).to be_empty
|
expect(info["genreUcid"].as_s?).to be_nil
|
||||||
expect(info["license"].as_s).to be_empty
|
expect(info["license"].as_s).to be_empty
|
||||||
|
|
||||||
# Author infos
|
# Author infos
|
||||||
|
|
|
@ -94,7 +94,7 @@ Spectator.describe "parse_video_info" do
|
||||||
# Video metadata
|
# Video metadata
|
||||||
|
|
||||||
expect(info["genre"].as_s).to eq("Entertainment")
|
expect(info["genre"].as_s).to eq("Entertainment")
|
||||||
expect(info["genreUcid"].as_s).to be_empty
|
expect(info["genreUcid"].as_s?).to be_nil
|
||||||
expect(info["license"].as_s).to be_empty
|
expect(info["license"].as_s).to be_empty
|
||||||
|
|
||||||
# Author infos
|
# Author infos
|
||||||
|
|
|
@ -67,11 +67,12 @@ REDIS_DB = Redis::PooledClient.new(unixsocket: CONFIG.redis_socket || nil, url:
|
||||||
if REDIS_DB.ping
|
if REDIS_DB.ping
|
||||||
puts "Connected to redis"
|
puts "Connected to redis"
|
||||||
end
|
end
|
||||||
ARCHIVE_URL = URI.parse("https://archive.org")
|
ARCHIVE_URL = URI.parse("https://archive.org")
|
||||||
PUBSUB_URL = URI.parse("https://pubsubhubbub.appspot.com")
|
PUBSUB_URL = URI.parse("https://pubsubhubbub.appspot.com")
|
||||||
REDDIT_URL = URI.parse("https://www.reddit.com")
|
REDDIT_URL = URI.parse("https://www.reddit.com")
|
||||||
YT_URL = URI.parse("https://www.youtube.com")
|
YT_URL = URI.parse("https://www.youtube.com")
|
||||||
HOST_URL = make_host_url(Kemal.config)
|
HOST_URL = make_host_url(Kemal.config)
|
||||||
|
EXT_VIDEOP_LIST = gen_videoplayback_proxy_list()
|
||||||
|
|
||||||
CHARS_SAFE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_"
|
CHARS_SAFE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_"
|
||||||
TEST_IDS = {"AgbeGFYluEA", "BaW_jenozKc", "a9LDPn-MO4I", "ddFvjfvPnqk", "iqKdEhx-dD4"}
|
TEST_IDS = {"AgbeGFYluEA", "BaW_jenozKc", "a9LDPn-MO4I", "ddFvjfvPnqk", "iqKdEhx-dD4"}
|
||||||
|
@ -109,20 +110,15 @@ Kemal.config.extra_options do |parser|
|
||||||
exit
|
exit
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
parser.on("-f THREADS", "--feed-threads=THREADS", "Number of threads for refreshing feeds (default: #{CONFIG.feed_threads})") do |number|
|
|
||||||
begin
|
|
||||||
CONFIG.feed_threads = number.to_i
|
|
||||||
rescue ex
|
|
||||||
puts "THREADS must be integer"
|
|
||||||
exit
|
|
||||||
end
|
|
||||||
end
|
|
||||||
parser.on("-o OUTPUT", "--output=OUTPUT", "Redirect output (default: #{CONFIG.output})") do |output|
|
parser.on("-o OUTPUT", "--output=OUTPUT", "Redirect output (default: #{CONFIG.output})") do |output|
|
||||||
CONFIG.output = output
|
CONFIG.output = output
|
||||||
end
|
end
|
||||||
parser.on("-l LEVEL", "--log-level=LEVEL", "Log level, one of #{LogLevel.values} (default: #{CONFIG.log_level})") do |log_level|
|
parser.on("-l LEVEL", "--log-level=LEVEL", "Log level, one of #{LogLevel.values} (default: #{CONFIG.log_level})") do |log_level|
|
||||||
CONFIG.log_level = LogLevel.parse(log_level)
|
CONFIG.log_level = LogLevel.parse(log_level)
|
||||||
end
|
end
|
||||||
|
parser.on("-k", "--colorize", "Colorize logs") do
|
||||||
|
CONFIG.colorize_logs = true
|
||||||
|
end
|
||||||
parser.on("-v", "--version", "Print version") do
|
parser.on("-v", "--version", "Print version") do
|
||||||
puts SOFTWARE.to_pretty_json
|
puts SOFTWARE.to_pretty_json
|
||||||
exit
|
exit
|
||||||
|
@ -139,7 +135,7 @@ if CONFIG.output.upcase != "STDOUT"
|
||||||
FileUtils.mkdir_p(File.dirname(CONFIG.output))
|
FileUtils.mkdir_p(File.dirname(CONFIG.output))
|
||||||
end
|
end
|
||||||
OUTPUT = CONFIG.output.upcase == "STDOUT" ? STDOUT : File.open(CONFIG.output, mode: "a")
|
OUTPUT = CONFIG.output.upcase == "STDOUT" ? STDOUT : File.open(CONFIG.output, mode: "a")
|
||||||
LOGGER = Invidious::LogHandler.new(OUTPUT, CONFIG.log_level)
|
LOGGER = Invidious::LogHandler.new(OUTPUT, CONFIG.log_level, CONFIG.colorize_logs)
|
||||||
|
|
||||||
# Check table integrity
|
# Check table integrity
|
||||||
Invidious::Database.check_integrity(CONFIG)
|
Invidious::Database.check_integrity(CONFIG)
|
||||||
|
@ -159,21 +155,21 @@ Invidious::Database.check_integrity(CONFIG)
|
||||||
{% puts "\nDone checking player dependencies, now compiling Invidious...\n" %}
|
{% puts "\nDone checking player dependencies, now compiling Invidious...\n" %}
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
||||||
|
# Misc
|
||||||
|
|
||||||
|
DECRYPT_FUNCTION =
|
||||||
|
if sig_helper_address = CONFIG.signature_server.presence
|
||||||
|
IV::DecryptFunction.new(sig_helper_address)
|
||||||
|
else
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
|
||||||
# Start jobs
|
# Start jobs
|
||||||
|
|
||||||
if CONFIG.channel_threads > 0
|
if CONFIG.channel_threads > 0
|
||||||
Invidious::Jobs.register Invidious::Jobs::RefreshChannelsJob.new(PG_DB)
|
Invidious::Jobs.register Invidious::Jobs::RefreshChannelsJob.new(PG_DB)
|
||||||
end
|
end
|
||||||
|
|
||||||
if CONFIG.feed_threads > 0
|
|
||||||
Invidious::Jobs.register Invidious::Jobs::RefreshFeedsJob.new(PG_DB)
|
|
||||||
end
|
|
||||||
|
|
||||||
DECRYPT_FUNCTION = DecryptFunction.new(CONFIG.decrypt_polling)
|
|
||||||
if CONFIG.decrypt_polling
|
|
||||||
Invidious::Jobs.register Invidious::Jobs::UpdateDecryptFunctionJob.new
|
|
||||||
end
|
|
||||||
|
|
||||||
if CONFIG.statistics_enabled
|
if CONFIG.statistics_enabled
|
||||||
Invidious::Jobs.register Invidious::Jobs::StatisticsRefreshJob.new(PG_DB, SOFTWARE)
|
Invidious::Jobs.register Invidious::Jobs::StatisticsRefreshJob.new(PG_DB, SOFTWARE)
|
||||||
end
|
end
|
||||||
|
@ -191,6 +187,16 @@ Invidious::Jobs.register Invidious::Jobs::NotificationJob.new(CONNECTION_CHANNEL
|
||||||
|
|
||||||
Invidious::Jobs.register Invidious::Jobs::ClearExpiredItemsJob.new
|
Invidious::Jobs.register Invidious::Jobs::ClearExpiredItemsJob.new
|
||||||
|
|
||||||
|
Invidious::Jobs.register Invidious::Jobs::InstanceListRefreshJob.new
|
||||||
|
|
||||||
|
if !CONFIG.external_videoplayback_proxy.empty?
|
||||||
|
Invidious::Jobs.register Invidious::Jobs::CheckExternalProxy.new
|
||||||
|
end
|
||||||
|
|
||||||
|
if CONFIG.refresh_tokens
|
||||||
|
Invidious::Jobs.register Invidious::Jobs::RefreshTokens.new
|
||||||
|
end
|
||||||
|
|
||||||
Invidious::Jobs.start_all
|
Invidious::Jobs.start_all
|
||||||
|
|
||||||
def popular_videos
|
def popular_videos
|
||||||
|
|
|
@ -14,7 +14,9 @@ record AboutChannel,
|
||||||
is_family_friendly : Bool,
|
is_family_friendly : Bool,
|
||||||
allowed_regions : Array(String),
|
allowed_regions : Array(String),
|
||||||
tabs : Array(String),
|
tabs : Array(String),
|
||||||
verified : Bool
|
tags : Array(String),
|
||||||
|
verified : Bool,
|
||||||
|
is_age_gated : Bool
|
||||||
|
|
||||||
def get_about_info(ucid, locale) : AboutChannel
|
def get_about_info(ucid, locale) : AboutChannel
|
||||||
begin
|
begin
|
||||||
|
@ -43,37 +45,103 @@ def get_about_info(ucid, locale) : AboutChannel
|
||||||
auto_generated = true
|
auto_generated = true
|
||||||
end
|
end
|
||||||
|
|
||||||
if auto_generated
|
tags = [] of String
|
||||||
author = initdata["header"]["interactiveTabbedHeaderRenderer"]["title"]["simpleText"].as_s
|
tab_names = [] of String
|
||||||
author_url = initdata["microformat"]["microformatDataRenderer"]["urlCanonical"].as_s
|
total_views = 0_i64
|
||||||
author_thumbnail = initdata["header"]["interactiveTabbedHeaderRenderer"]["boxArt"]["thumbnails"][0]["url"].as_s
|
joined = Time.unix(0)
|
||||||
|
|
||||||
# Raises a KeyError on failure.
|
if age_gate_renderer = initdata.dig?("contents", "twoColumnBrowseResultsRenderer", "tabs", 0, "tabRenderer", "content", "sectionListRenderer", "contents", 0, "channelAgeGateRenderer")
|
||||||
banners = initdata["header"]["interactiveTabbedHeaderRenderer"]?.try &.["banner"]?.try &.["thumbnails"]?
|
description_node = nil
|
||||||
banner = banners.try &.[-1]?.try &.["url"].as_s?
|
author = age_gate_renderer["channelTitle"].as_s
|
||||||
|
ucid = initdata.dig("responseContext", "serviceTrackingParams", 0, "params", 0, "value").as_s
|
||||||
description_node = initdata["header"]["interactiveTabbedHeaderRenderer"]["description"]
|
author_url = "https://www.youtube.com/channel/#{ucid}"
|
||||||
|
author_thumbnail = age_gate_renderer.dig("avatar", "thumbnails", 0, "url").as_s
|
||||||
|
banner = nil
|
||||||
|
is_family_friendly = false
|
||||||
|
is_age_gated = true
|
||||||
|
tab_names = ["videos", "shorts", "streams"]
|
||||||
|
auto_generated = false
|
||||||
else
|
else
|
||||||
author = initdata["metadata"]["channelMetadataRenderer"]["title"].as_s
|
if auto_generated
|
||||||
author_url = initdata["metadata"]["channelMetadataRenderer"]["channelUrl"].as_s
|
author = initdata["header"]["interactiveTabbedHeaderRenderer"]["title"]["simpleText"].as_s
|
||||||
author_thumbnail = initdata["metadata"]["channelMetadataRenderer"]["avatar"]["thumbnails"][0]["url"].as_s
|
author_url = initdata["microformat"]["microformatDataRenderer"]["urlCanonical"].as_s
|
||||||
author_verified = has_verified_badge?(initdata.dig?("header", "c4TabbedHeaderRenderer", "badges"))
|
author_thumbnail = initdata["header"]["interactiveTabbedHeaderRenderer"]["boxArt"]["thumbnails"][0]["url"].as_s
|
||||||
|
|
||||||
ucid = initdata["metadata"]["channelMetadataRenderer"]["externalId"].as_s
|
# Raises a KeyError on failure.
|
||||||
|
banners = initdata["header"]["interactiveTabbedHeaderRenderer"]?.try &.["banner"]?.try &.["thumbnails"]?
|
||||||
|
banner = banners.try &.[-1]?.try &.["url"].as_s?
|
||||||
|
|
||||||
# Raises a KeyError on failure.
|
description_base_node = initdata["header"]["interactiveTabbedHeaderRenderer"]["description"]
|
||||||
banners = initdata["header"]["c4TabbedHeaderRenderer"]?.try &.["banner"]?.try &.["thumbnails"]?
|
# some channels have the description in a simpleText
|
||||||
banner = banners.try &.[-1]?.try &.["url"].as_s?
|
# ex: https://www.youtube.com/channel/UCQvWX73GQygcwXOTSf_VDVg/
|
||||||
|
description_node = description_base_node.dig?("simpleText") || description_base_node
|
||||||
|
|
||||||
# if banner.includes? "channels/c4/default_banner"
|
tags = initdata.dig?("header", "interactiveTabbedHeaderRenderer", "badges")
|
||||||
# banner = nil
|
.try &.as_a.map(&.["metadataBadgeRenderer"]["label"].as_s) || [] of String
|
||||||
# end
|
else
|
||||||
|
author = initdata["metadata"]["channelMetadataRenderer"]["title"].as_s
|
||||||
|
author_url = initdata["metadata"]["channelMetadataRenderer"]["channelUrl"].as_s
|
||||||
|
author_thumbnail = initdata["metadata"]["channelMetadataRenderer"]["avatar"]["thumbnails"][0]["url"].as_s
|
||||||
|
author_verified = has_verified_badge?(initdata.dig?("header", "c4TabbedHeaderRenderer", "badges"))
|
||||||
|
|
||||||
description_node = initdata["metadata"]["channelMetadataRenderer"]?.try &.["description"]?
|
ucid = initdata["metadata"]["channelMetadataRenderer"]["externalId"].as_s
|
||||||
|
|
||||||
|
# Raises a KeyError on failure.
|
||||||
|
banners = initdata["header"]["c4TabbedHeaderRenderer"]?.try &.["banner"]?.try &.["thumbnails"]?
|
||||||
|
banners ||= initdata.dig?("header", "pageHeaderRenderer", "content", "pageHeaderViewModel", "banner", "imageBannerViewModel", "image", "sources")
|
||||||
|
banner = banners.try &.[-1]?.try &.["url"].as_s?
|
||||||
|
|
||||||
|
# if banner.includes? "channels/c4/default_banner"
|
||||||
|
# banner = nil
|
||||||
|
# end
|
||||||
|
|
||||||
|
description_node = initdata["metadata"]["channelMetadataRenderer"]?.try &.["description"]?
|
||||||
|
tags = initdata.dig?("microformat", "microformatDataRenderer", "tags").try &.as_a.map(&.as_s) || [] of String
|
||||||
|
end
|
||||||
|
|
||||||
|
is_family_friendly = initdata["microformat"]["microformatDataRenderer"]["familySafe"].as_bool
|
||||||
|
if tabs_json = initdata["contents"]["twoColumnBrowseResultsRenderer"]["tabs"]?
|
||||||
|
# Get the name of the tabs available on this channel
|
||||||
|
tab_names = tabs_json.as_a.compact_map do |entry|
|
||||||
|
name = entry.dig?("tabRenderer", "title").try &.as_s.downcase
|
||||||
|
|
||||||
|
# This is a small fix to not add extra code on the HTML side
|
||||||
|
# I.e, the URL for the "live" tab is .../streams, so use "streams"
|
||||||
|
# everywhere for the sake of simplicity
|
||||||
|
(name == "live") ? "streams" : name
|
||||||
|
end
|
||||||
|
|
||||||
|
# Get the currently active tab ("About")
|
||||||
|
about_tab = extract_selected_tab(tabs_json)
|
||||||
|
|
||||||
|
# Try to find the about metadata section
|
||||||
|
channel_about_meta = about_tab.dig?(
|
||||||
|
"content",
|
||||||
|
"sectionListRenderer", "contents", 0,
|
||||||
|
"itemSectionRenderer", "contents", 0,
|
||||||
|
"channelAboutFullMetadataRenderer"
|
||||||
|
)
|
||||||
|
|
||||||
|
if !channel_about_meta.nil?
|
||||||
|
total_views = channel_about_meta.dig?("viewCountText", "simpleText").try &.as_s.gsub(/\D/, "").to_i64? || 0_i64
|
||||||
|
|
||||||
|
# The joined text is split to several sub strings. The reduce joins those strings before parsing the date.
|
||||||
|
joined = extract_text(channel_about_meta["joinedDateText"]?)
|
||||||
|
.try { |text| Time.parse(text, "Joined %b %-d, %Y", Time::Location.local) } || Time.unix(0)
|
||||||
|
|
||||||
|
# Normal Auto-generated channels
|
||||||
|
# https://support.google.com/youtube/answer/2579942
|
||||||
|
# For auto-generated channels, channel_about_meta only has
|
||||||
|
# ["description"]["simpleText"] and ["primaryLinks"][0]["title"]["simpleText"]
|
||||||
|
auto_generated = (
|
||||||
|
(channel_about_meta["primaryLinks"]?.try &.size) == 1 && \
|
||||||
|
extract_text(channel_about_meta.dig?("primaryLinks", 0, "title")) == "Auto-generated by YouTube" ||
|
||||||
|
channel_about_meta.dig?("links", 0, "channelExternalLinkViewModel", "title", "content").try &.as_s == "Auto-generated by YouTube"
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
is_family_friendly = initdata["microformat"]["microformatDataRenderer"]["familySafe"].as_bool
|
|
||||||
|
|
||||||
allowed_regions = initdata
|
allowed_regions = initdata
|
||||||
.dig?("microformat", "microformatDataRenderer", "availableCountries")
|
.dig?("microformat", "microformatDataRenderer", "availableCountries")
|
||||||
.try &.as_a.map(&.as_s) || [] of String
|
.try &.as_a.map(&.as_s) || [] of String
|
||||||
|
@ -91,55 +159,18 @@ def get_about_info(ucid, locale) : AboutChannel
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
total_views = 0_i64
|
sub_count = 0
|
||||||
joined = Time.unix(0)
|
|
||||||
|
|
||||||
tab_names = [] of String
|
if (metadata_rows = initdata.dig?("header", "pageHeaderRenderer", "content", "pageHeaderViewModel", "metadata", "contentMetadataViewModel", "metadataRows").try &.as_a)
|
||||||
|
metadata_rows.each do |row|
|
||||||
if tabs_json = initdata["contents"]["twoColumnBrowseResultsRenderer"]["tabs"]?
|
metadata_part = row.dig?("metadataParts").try &.as_a.find { |i| i.dig?("text", "content").try &.as_s.includes?("subscribers") }
|
||||||
# Get the name of the tabs available on this channel
|
if !metadata_part.nil?
|
||||||
tab_names = tabs_json.as_a.compact_map do |entry|
|
sub_count = short_text_to_number(metadata_part.dig("text", "content").as_s.split(" ")[0]).to_i32
|
||||||
name = entry.dig?("tabRenderer", "title").try &.as_s.downcase
|
end
|
||||||
|
break if sub_count != 0
|
||||||
# This is a small fix to not add extra code on the HTML side
|
|
||||||
# I.e, the URL for the "live" tab is .../streams, so use "streams"
|
|
||||||
# everywhere for the sake of simplicity
|
|
||||||
(name == "live") ? "streams" : name
|
|
||||||
end
|
|
||||||
|
|
||||||
# Get the currently active tab ("About")
|
|
||||||
about_tab = extract_selected_tab(tabs_json)
|
|
||||||
|
|
||||||
# Try to find the about metadata section
|
|
||||||
channel_about_meta = about_tab.dig?(
|
|
||||||
"content",
|
|
||||||
"sectionListRenderer", "contents", 0,
|
|
||||||
"itemSectionRenderer", "contents", 0,
|
|
||||||
"channelAboutFullMetadataRenderer"
|
|
||||||
)
|
|
||||||
|
|
||||||
if !channel_about_meta.nil?
|
|
||||||
total_views = channel_about_meta.dig?("viewCountText", "simpleText").try &.as_s.gsub(/\D/, "").to_i64? || 0_i64
|
|
||||||
|
|
||||||
# The joined text is split to several sub strings. The reduce joins those strings before parsing the date.
|
|
||||||
joined = extract_text(channel_about_meta["joinedDateText"]?)
|
|
||||||
.try { |text| Time.parse(text, "Joined %b %-d, %Y", Time::Location.local) } || Time.unix(0)
|
|
||||||
|
|
||||||
# Normal Auto-generated channels
|
|
||||||
# https://support.google.com/youtube/answer/2579942
|
|
||||||
# For auto-generated channels, channel_about_meta only has
|
|
||||||
# ["description"]["simpleText"] and ["primaryLinks"][0]["title"]["simpleText"]
|
|
||||||
auto_generated = (
|
|
||||||
(channel_about_meta["primaryLinks"]?.try &.size) == 1 && \
|
|
||||||
extract_text(channel_about_meta.dig?("primaryLinks", 0, "title")) == "Auto-generated by YouTube"
|
|
||||||
)
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
sub_count = initdata
|
|
||||||
.dig?("header", "c4TabbedHeaderRenderer", "subscriberCountText", "simpleText").try &.as_s?
|
|
||||||
.try { |text| short_text_to_number(text.split(" ")[0]).to_i32 } || 0
|
|
||||||
|
|
||||||
AboutChannel.new(
|
AboutChannel.new(
|
||||||
ucid: ucid,
|
ucid: ucid,
|
||||||
author: author,
|
author: author,
|
||||||
|
@ -155,7 +186,9 @@ def get_about_info(ucid, locale) : AboutChannel
|
||||||
is_family_friendly: is_family_friendly,
|
is_family_friendly: is_family_friendly,
|
||||||
allowed_regions: allowed_regions,
|
allowed_regions: allowed_regions,
|
||||||
tabs: tab_names,
|
tabs: tab_names,
|
||||||
|
tags: tags,
|
||||||
verified: author_verified || false,
|
verified: author_verified || false,
|
||||||
|
is_age_gated: is_age_gated || false,
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -223,7 +223,7 @@ def fetch_channel(ucid, pull_all_videos : Bool)
|
||||||
length_seconds = channel_video.try &.length_seconds
|
length_seconds = channel_video.try &.length_seconds
|
||||||
length_seconds ||= 0
|
length_seconds ||= 0
|
||||||
|
|
||||||
live_now = channel_video.try &.live_now
|
live_now = channel_video.try &.badges.live_now?
|
||||||
live_now ||= false
|
live_now ||= false
|
||||||
|
|
||||||
premiere_timestamp = channel_video.try &.premiere_timestamp
|
premiere_timestamp = channel_video.try &.premiere_timestamp
|
||||||
|
@ -232,7 +232,7 @@ def fetch_channel(ucid, pull_all_videos : Bool)
|
||||||
id: video_id,
|
id: video_id,
|
||||||
title: title,
|
title: title,
|
||||||
published: published,
|
published: published,
|
||||||
updated: Time.utc,
|
updated: updated,
|
||||||
ucid: ucid,
|
ucid: ucid,
|
||||||
author: author,
|
author: author,
|
||||||
length_seconds: length_seconds,
|
length_seconds: length_seconds,
|
||||||
|
@ -251,8 +251,8 @@ def fetch_channel(ucid, pull_all_videos : Bool)
|
||||||
LOGGER.trace("fetch_channel: #{ucid} : video #{video_id} : Inserted, updating subscriptions")
|
LOGGER.trace("fetch_channel: #{ucid} : video #{video_id} : Inserted, updating subscriptions")
|
||||||
if CONFIG.enable_user_notifications
|
if CONFIG.enable_user_notifications
|
||||||
Invidious::Database::Users.add_notification(video)
|
Invidious::Database::Users.add_notification(video)
|
||||||
else
|
# else
|
||||||
Invidious::Database::Users.feed_needs_update(video)
|
# Invidious::Database::Users.feed_needs_update(video)
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
LOGGER.trace("fetch_channel: #{ucid} : video #{video_id} : Updated")
|
LOGGER.trace("fetch_channel: #{ucid} : video #{video_id} : Updated")
|
||||||
|
@ -275,7 +275,7 @@ def fetch_channel(ucid, pull_all_videos : Bool)
|
||||||
ucid: video.ucid,
|
ucid: video.ucid,
|
||||||
author: video.author,
|
author: video.author,
|
||||||
length_seconds: video.length_seconds,
|
length_seconds: video.length_seconds,
|
||||||
live_now: video.live_now,
|
live_now: video.badges.live_now?,
|
||||||
premiere_timestamp: video.premiere_timestamp,
|
premiere_timestamp: video.premiere_timestamp,
|
||||||
views: video.views,
|
views: video.views,
|
||||||
})
|
})
|
||||||
|
@ -287,8 +287,8 @@ def fetch_channel(ucid, pull_all_videos : Bool)
|
||||||
if was_insert
|
if was_insert
|
||||||
if CONFIG.enable_user_notifications
|
if CONFIG.enable_user_notifications
|
||||||
Invidious::Database::Users.add_notification(video)
|
Invidious::Database::Users.add_notification(video)
|
||||||
else
|
# else
|
||||||
Invidious::Database::Users.feed_needs_update(video)
|
# Invidious::Database::Users.feed_needs_update(video)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
def produce_channel_videos_continuation(ucid, page = 1, auto_generated = nil, sort_by = "newest", v2 = false)
|
def produce_channel_content_continuation(ucid, content_type, page = 1, auto_generated = nil, sort_by = "newest", v2 = false)
|
||||||
object_inner_2 = {
|
object_inner_2 = {
|
||||||
"2:0:embedded" => {
|
"2:0:embedded" => {
|
||||||
"1:0:varint" => 0_i64,
|
"1:0:varint" => 0_i64,
|
||||||
|
@ -16,6 +16,13 @@ def produce_channel_videos_continuation(ucid, page = 1, auto_generated = nil, so
|
||||||
.try { |i| Base64.urlsafe_encode(i) }
|
.try { |i| Base64.urlsafe_encode(i) }
|
||||||
.try { |i| URI.encode_www_form(i) }
|
.try { |i| URI.encode_www_form(i) }
|
||||||
|
|
||||||
|
content_type_numerical =
|
||||||
|
case content_type
|
||||||
|
when "videos" then 15
|
||||||
|
when "livestreams" then 14
|
||||||
|
else 15 # Fallback to "videos"
|
||||||
|
end
|
||||||
|
|
||||||
sort_by_numerical =
|
sort_by_numerical =
|
||||||
case sort_by
|
case sort_by
|
||||||
when "newest" then 1_i64
|
when "newest" then 1_i64
|
||||||
|
@ -27,7 +34,7 @@ def produce_channel_videos_continuation(ucid, page = 1, auto_generated = nil, so
|
||||||
object_inner_1 = {
|
object_inner_1 = {
|
||||||
"110:embedded" => {
|
"110:embedded" => {
|
||||||
"3:embedded" => {
|
"3:embedded" => {
|
||||||
"15:embedded" => {
|
"#{content_type_numerical}:embedded" => {
|
||||||
"1:embedded" => {
|
"1:embedded" => {
|
||||||
"1:string" => object_inner_2_encoded,
|
"1:string" => object_inner_2_encoded,
|
||||||
},
|
},
|
||||||
|
@ -62,6 +69,10 @@ def produce_channel_videos_continuation(ucid, page = 1, auto_generated = nil, so
|
||||||
return continuation
|
return continuation
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def make_initial_content_ctoken(ucid, content_type, sort_by) : String
|
||||||
|
return produce_channel_content_continuation(ucid, content_type, sort_by: sort_by)
|
||||||
|
end
|
||||||
|
|
||||||
module Invidious::Channel::Tabs
|
module Invidious::Channel::Tabs
|
||||||
extend self
|
extend self
|
||||||
|
|
||||||
|
@ -69,10 +80,6 @@ module Invidious::Channel::Tabs
|
||||||
# Regular videos
|
# Regular videos
|
||||||
# -------------------
|
# -------------------
|
||||||
|
|
||||||
def make_initial_video_ctoken(ucid, sort_by) : String
|
|
||||||
return produce_channel_videos_continuation(ucid, sort_by: sort_by)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Wrapper for AboutChannel, as we still need to call get_videos with
|
# Wrapper for AboutChannel, as we still need to call get_videos with
|
||||||
# an author name and ucid directly (e.g in RSS feeds).
|
# an author name and ucid directly (e.g in RSS feeds).
|
||||||
# TODO: figure out how to get rid of that
|
# TODO: figure out how to get rid of that
|
||||||
|
@ -94,7 +101,7 @@ module Invidious::Channel::Tabs
|
||||||
end
|
end
|
||||||
|
|
||||||
def get_videos(author : String, ucid : String, *, continuation : String? = nil, sort_by = "newest")
|
def get_videos(author : String, ucid : String, *, continuation : String? = nil, sort_by = "newest")
|
||||||
continuation ||= make_initial_video_ctoken(ucid, sort_by)
|
continuation ||= make_initial_content_ctoken(ucid, "videos", sort_by)
|
||||||
initial_data = YoutubeAPI.browse(continuation: continuation)
|
initial_data = YoutubeAPI.browse(continuation: continuation)
|
||||||
|
|
||||||
return extract_items(initial_data, author, ucid)
|
return extract_items(initial_data, author, ucid)
|
||||||
|
@ -138,21 +145,18 @@ module Invidious::Channel::Tabs
|
||||||
# Livestreams
|
# Livestreams
|
||||||
# -------------------
|
# -------------------
|
||||||
|
|
||||||
def get_livestreams(channel : AboutChannel, continuation : String? = nil)
|
def get_livestreams(channel : AboutChannel, continuation : String? = nil, sort_by = "newest")
|
||||||
if continuation.nil?
|
continuation ||= make_initial_content_ctoken(channel.ucid, "livestreams", sort_by)
|
||||||
# EgdzdHJlYW1z8gYECgJ6AA%3D%3D is the protobuf object to load "streams"
|
|
||||||
initial_data = YoutubeAPI.browse(channel.ucid, params: "EgdzdHJlYW1z8gYECgJ6AA%3D%3D")
|
initial_data = YoutubeAPI.browse(continuation: continuation)
|
||||||
else
|
|
||||||
initial_data = YoutubeAPI.browse(continuation: continuation)
|
|
||||||
end
|
|
||||||
|
|
||||||
return extract_items(initial_data, channel.author, channel.ucid)
|
return extract_items(initial_data, channel.author, channel.ucid)
|
||||||
end
|
end
|
||||||
|
|
||||||
def get_60_livestreams(channel : AboutChannel, continuation : String? = nil)
|
def get_60_livestreams(channel : AboutChannel, *, continuation : String? = nil, sort_by = "newest")
|
||||||
if continuation.nil?
|
if continuation.nil?
|
||||||
# Fetch the first "page" of streams
|
# Fetch the first "page" of stream
|
||||||
items, next_continuation = get_livestreams(channel)
|
items, next_continuation = get_livestreams(channel, sort_by: sort_by)
|
||||||
else
|
else
|
||||||
# Fetch a "page" of streams using the given continuation token
|
# Fetch a "page" of streams using the given continuation token
|
||||||
items, next_continuation = get_livestreams(channel, continuation: continuation)
|
items, next_continuation = get_livestreams(channel, continuation: continuation)
|
||||||
|
|
|
@ -5,35 +5,35 @@ def text_to_parsed_content(text : String) : JSON::Any
|
||||||
# In first case line is just a simple node before
|
# In first case line is just a simple node before
|
||||||
# check patterns inside line
|
# check patterns inside line
|
||||||
# { 'text': line }
|
# { 'text': line }
|
||||||
currentNodes = [] of JSON::Any
|
current_nodes = [] of JSON::Any
|
||||||
initialNode = {"text" => line}
|
initial_node = {"text" => line}
|
||||||
currentNodes << (JSON.parse(initialNode.to_json))
|
current_nodes << (JSON.parse(initial_node.to_json))
|
||||||
|
|
||||||
# For each match with url pattern, get last node and preserve
|
# For each match with url pattern, get last node and preserve
|
||||||
# last node before create new node with url information
|
# last node before create new node with url information
|
||||||
# { 'text': match, 'navigationEndpoint': { 'urlEndpoint' : 'url': match } }
|
# { 'text': match, 'navigationEndpoint': { 'urlEndpoint' : 'url': match } }
|
||||||
line.scan(/https?:\/\/[^ ]*/).each do |urlMatch|
|
line.scan(/https?:\/\/[^ ]*/).each do |url_match|
|
||||||
# Retrieve last node and update node without match
|
# Retrieve last node and update node without match
|
||||||
lastNode = currentNodes[currentNodes.size - 1].as_h
|
last_node = current_nodes[-1].as_h
|
||||||
splittedLastNode = lastNode["text"].as_s.split(urlMatch[0])
|
splitted_last_node = last_node["text"].as_s.split(url_match[0])
|
||||||
lastNode["text"] = JSON.parse(splittedLastNode[0].to_json)
|
last_node["text"] = JSON.parse(splitted_last_node[0].to_json)
|
||||||
currentNodes[currentNodes.size - 1] = JSON.parse(lastNode.to_json)
|
current_nodes[-1] = JSON.parse(last_node.to_json)
|
||||||
# Create new node with match and navigation infos
|
# Create new node with match and navigation infos
|
||||||
currentNode = {"text" => urlMatch[0], "navigationEndpoint" => {"urlEndpoint" => {"url" => urlMatch[0]}}}
|
current_node = {"text" => url_match[0], "navigationEndpoint" => {"urlEndpoint" => {"url" => url_match[0]}}}
|
||||||
currentNodes << (JSON.parse(currentNode.to_json))
|
current_nodes << (JSON.parse(current_node.to_json))
|
||||||
# If text remain after match create new simple node with text after match
|
# If text remain after match create new simple node with text after match
|
||||||
afterNode = {"text" => splittedLastNode.size > 1 ? splittedLastNode[1] : ""}
|
after_node = {"text" => splitted_last_node.size > 1 ? splitted_last_node[1] : ""}
|
||||||
currentNodes << (JSON.parse(afterNode.to_json))
|
current_nodes << (JSON.parse(after_node.to_json))
|
||||||
end
|
end
|
||||||
|
|
||||||
# After processing of matches inside line
|
# After processing of matches inside line
|
||||||
# Add \n at end of last node for preserve carriage return
|
# Add \n at end of last node for preserve carriage return
|
||||||
lastNode = currentNodes[currentNodes.size - 1].as_h
|
last_node = current_nodes[-1].as_h
|
||||||
lastNode["text"] = JSON.parse("#{currentNodes[currentNodes.size - 1]["text"]}\n".to_json)
|
last_node["text"] = JSON.parse("#{last_node["text"]}\n".to_json)
|
||||||
currentNodes[currentNodes.size - 1] = JSON.parse(lastNode.to_json)
|
current_nodes[-1] = JSON.parse(last_node.to_json)
|
||||||
|
|
||||||
# Finally add final nodes to nodes returned
|
# Finally add final nodes to nodes returned
|
||||||
currentNodes.each do |node|
|
current_nodes.each do |node|
|
||||||
nodes << (node)
|
nodes << (node)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -53,8 +53,8 @@ def content_to_comment_html(content, video_id : String? = "")
|
||||||
|
|
||||||
text = HTML.escape(run["text"].as_s)
|
text = HTML.escape(run["text"].as_s)
|
||||||
|
|
||||||
if navigationEndpoint = run.dig?("navigationEndpoint")
|
if navigation_endpoint = run.dig?("navigationEndpoint")
|
||||||
text = parse_link_endpoint(navigationEndpoint, text, video_id)
|
text = parse_link_endpoint(navigation_endpoint, text, video_id)
|
||||||
end
|
end
|
||||||
|
|
||||||
text = "<b>#{text}</b>" if run["bold"]?
|
text = "<b>#{text}</b>" if run["bold"]?
|
||||||
|
@ -64,15 +64,15 @@ def content_to_comment_html(content, video_id : String? = "")
|
||||||
# check for custom emojis
|
# check for custom emojis
|
||||||
if run["emoji"]?
|
if run["emoji"]?
|
||||||
if run["emoji"]["isCustomEmoji"]?.try &.as_bool
|
if run["emoji"]["isCustomEmoji"]?.try &.as_bool
|
||||||
if emojiImage = run.dig?("emoji", "image")
|
if emoji_image = run.dig?("emoji", "image")
|
||||||
emojiAlt = emojiImage.dig?("accessibility", "accessibilityData", "label").try &.as_s || text
|
emoji_alt = emoji_image.dig?("accessibility", "accessibilityData", "label").try &.as_s || text
|
||||||
emojiThumb = emojiImage["thumbnails"][0]
|
emoji_thumb = emoji_image["thumbnails"][0]
|
||||||
text = String.build do |str|
|
text = String.build do |str|
|
||||||
str << %(<img alt=") << emojiAlt << "\" "
|
str << %(<img alt=") << emoji_alt << "\" "
|
||||||
str << %(src="/ggpht) << URI.parse(emojiThumb["url"].as_s).request_target << "\" "
|
str << %(src="/ggpht) << URI.parse(emoji_thumb["url"].as_s).request_target << "\" "
|
||||||
str << %(title=") << emojiAlt << "\" "
|
str << %(title=") << emoji_alt << "\" "
|
||||||
str << %(width=") << emojiThumb["width"] << "\" "
|
str << %(width=") << emoji_thumb["width"] << "\" "
|
||||||
str << %(height=") << emojiThumb["height"] << "\" "
|
str << %(height=") << emoji_thumb["height"] << "\" "
|
||||||
str << %(class="channel-emoji" />)
|
str << %(class="channel-emoji" />)
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
|
|
|
@ -57,7 +57,7 @@ module Invidious::Comments
|
||||||
return initial_data
|
return initial_data
|
||||||
end
|
end
|
||||||
|
|
||||||
def parse_youtube(id, response, format, locale, thin_mode, sort_by = "top", isPost = false)
|
def parse_youtube(id, response, format, locale, thin_mode, sort_by = "top", is_post = false)
|
||||||
contents = nil
|
contents = nil
|
||||||
|
|
||||||
if on_response_received_endpoints = response["onResponseReceivedEndpoints"]?
|
if on_response_received_endpoints = response["onResponseReceivedEndpoints"]?
|
||||||
|
@ -104,6 +104,8 @@ module Invidious::Comments
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
mutations = response.dig?("frameworkUpdates", "entityBatchUpdate", "mutations").try &.as_a || [] of JSON::Any
|
||||||
|
|
||||||
response = JSON.build do |json|
|
response = JSON.build do |json|
|
||||||
json.object do
|
json.object do
|
||||||
if header
|
if header
|
||||||
|
@ -113,7 +115,7 @@ module Invidious::Comments
|
||||||
json.field "commentCount", comment_count
|
json.field "commentCount", comment_count
|
||||||
end
|
end
|
||||||
|
|
||||||
if isPost
|
if is_post
|
||||||
json.field "postId", id
|
json.field "postId", id
|
||||||
else
|
else
|
||||||
json.field "videoId", id
|
json.field "videoId", id
|
||||||
|
@ -131,73 +133,138 @@ module Invidious::Comments
|
||||||
node_replies = node["replies"]["commentRepliesRenderer"]
|
node_replies = node["replies"]["commentRepliesRenderer"]
|
||||||
end
|
end
|
||||||
|
|
||||||
if node["comment"]?
|
if cvm = node["commentViewModel"]?
|
||||||
node_comment = node["comment"]["commentRenderer"]
|
# two commentViewModels for inital request
|
||||||
|
# one commentViewModel when getting a replies to a comment
|
||||||
|
cvm = cvm["commentViewModel"] if cvm["commentViewModel"]?
|
||||||
|
|
||||||
|
comment_key = cvm["commentKey"]
|
||||||
|
toolbar_key = cvm["toolbarStateKey"]
|
||||||
|
comment_mutation = mutations.find { |i| i.dig?("payload", "commentEntityPayload", "key") == comment_key }
|
||||||
|
toolbar_mutation = mutations.find { |i| i.dig?("entityKey") == toolbar_key }
|
||||||
|
|
||||||
|
if !comment_mutation.nil? && !toolbar_mutation.nil?
|
||||||
|
# todo parse styleRuns, commandRuns and attachmentRuns for comments
|
||||||
|
html_content = parse_description(comment_mutation.dig("payload", "commentEntityPayload", "properties", "content"), id)
|
||||||
|
comment_author = comment_mutation.dig("payload", "commentEntityPayload", "author")
|
||||||
|
json.field "authorId", comment_author["channelId"].as_s
|
||||||
|
json.field "authorUrl", "/channel/#{comment_author["channelId"].as_s}"
|
||||||
|
json.field "author", comment_author["displayName"].as_s
|
||||||
|
json.field "verified", comment_author["isVerified"].as_bool
|
||||||
|
json.field "authorThumbnails" do
|
||||||
|
json.array do
|
||||||
|
comment_mutation.dig?("payload", "commentEntityPayload", "avatar", "image", "sources").try &.as_a.each do |thumbnail|
|
||||||
|
json.object do
|
||||||
|
json.field "url", thumbnail["url"]
|
||||||
|
json.field "width", thumbnail["width"]
|
||||||
|
json.field "height", thumbnail["height"]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
json.field "authorIsChannelOwner", comment_author["isCreator"].as_bool
|
||||||
|
json.field "isSponsor", (comment_author["sponsorBadgeUrl"]? != nil)
|
||||||
|
|
||||||
|
if sponsor_badge_url = comment_author["sponsorBadgeUrl"]?
|
||||||
|
# Sponsor icon thumbnails always have one object and there's only ever the url property in it
|
||||||
|
json.field "sponsorIconUrl", sponsor_badge_url
|
||||||
|
end
|
||||||
|
|
||||||
|
comment_toolbar = comment_mutation.dig("payload", "commentEntityPayload", "toolbar")
|
||||||
|
json.field "likeCount", short_text_to_number(comment_toolbar["likeCountNotliked"].as_s)
|
||||||
|
reply_count = short_text_to_number(comment_toolbar["replyCount"]?.try &.as_s || "0")
|
||||||
|
|
||||||
|
if heart_state = toolbar_mutation.dig?("payload", "engagementToolbarStateEntityPayload", "heartState")
|
||||||
|
if heart_state.as_s == "TOOLBAR_HEART_STATE_HEARTED"
|
||||||
|
json.field "creatorHeart" do
|
||||||
|
json.object do
|
||||||
|
json.field "creatorThumbnail", comment_toolbar["creatorThumbnailUrl"].as_s
|
||||||
|
json.field "creatorName", comment_toolbar["heartActiveTooltip"].as_s.sub("❤ by ", "")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
published_text = comment_mutation.dig?("payload", "commentEntityPayload", "properties", "publishedTime").try &.as_s
|
||||||
|
end
|
||||||
|
|
||||||
|
json.field "isPinned", (cvm.dig?("pinnedText") != nil)
|
||||||
|
json.field "commentId", cvm["commentId"]
|
||||||
else
|
else
|
||||||
node_comment = node["commentRenderer"]
|
if node["comment"]?
|
||||||
end
|
node_comment = node["comment"]["commentRenderer"]
|
||||||
|
else
|
||||||
|
node_comment = node["commentRenderer"]
|
||||||
|
end
|
||||||
|
json.field "commentId", node_comment["commentId"]
|
||||||
|
html_content = node_comment["contentText"]?.try { |t| parse_content(t, id) }
|
||||||
|
|
||||||
content_html = node_comment["contentText"]?.try { |t| parse_content(t, id) } || ""
|
json.field "verified", (node_comment["authorCommentBadge"]? != nil)
|
||||||
author = node_comment["authorText"]?.try &.["simpleText"]? || ""
|
|
||||||
|
|
||||||
json.field "verified", (node_comment["authorCommentBadge"]? != nil)
|
json.field "author", node_comment["authorText"]?.try &.["simpleText"]? || ""
|
||||||
|
json.field "authorThumbnails" do
|
||||||
json.field "author", author
|
json.array do
|
||||||
json.field "authorThumbnails" do
|
node_comment["authorThumbnail"]["thumbnails"].as_a.each do |thumbnail|
|
||||||
json.array do
|
json.object do
|
||||||
node_comment["authorThumbnail"]["thumbnails"].as_a.each do |thumbnail|
|
json.field "url", thumbnail["url"]
|
||||||
json.object do
|
json.field "width", thumbnail["width"]
|
||||||
json.field "url", thumbnail["url"]
|
json.field "height", thumbnail["height"]
|
||||||
json.field "width", thumbnail["width"]
|
end
|
||||||
json.field "height", thumbnail["height"]
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if comment_action_buttons_renderer = node_comment.dig?("actionButtons", "commentActionButtonsRenderer")
|
||||||
|
json.field "likeCount", comment_action_buttons_renderer["likeButton"]["toggleButtonRenderer"]["accessibilityData"]["accessibilityData"]["label"].as_s.scan(/\d/).map(&.[0]).join.to_i
|
||||||
|
if comment_action_buttons_renderer["creatorHeart"]?
|
||||||
|
heart_data = comment_action_buttons_renderer["creatorHeart"]["creatorHeartRenderer"]["creatorThumbnail"]
|
||||||
|
json.field "creatorHeart" do
|
||||||
|
json.object do
|
||||||
|
json.field "creatorThumbnail", heart_data["thumbnails"][-1]["url"]
|
||||||
|
json.field "creatorName", heart_data["accessibility"]["accessibilityData"]["label"]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
if node_comment["authorEndpoint"]?
|
||||||
|
json.field "authorId", node_comment["authorEndpoint"]["browseEndpoint"]["browseId"]
|
||||||
|
json.field "authorUrl", node_comment["authorEndpoint"]["browseEndpoint"]["canonicalBaseUrl"]
|
||||||
|
else
|
||||||
|
json.field "authorId", ""
|
||||||
|
json.field "authorUrl", ""
|
||||||
|
end
|
||||||
|
|
||||||
|
json.field "authorIsChannelOwner", node_comment["authorIsChannelOwner"]
|
||||||
|
json.field "isPinned", (node_comment["pinnedCommentBadge"]? != nil)
|
||||||
|
published_text = node_comment["publishedTimeText"]["runs"][0]["text"].as_s
|
||||||
|
|
||||||
|
json.field "isSponsor", (node_comment["sponsorCommentBadge"]? != nil)
|
||||||
|
if node_comment["sponsorCommentBadge"]?
|
||||||
|
# Sponsor icon thumbnails always have one object and there's only ever the url property in it
|
||||||
|
json.field "sponsorIconUrl", node_comment.dig("sponsorCommentBadge", "sponsorCommentBadgeRenderer", "customBadge", "thumbnails", 0, "url").to_s
|
||||||
|
end
|
||||||
|
|
||||||
|
reply_count = node_comment["replyCount"]?
|
||||||
end
|
end
|
||||||
|
|
||||||
if node_comment["authorEndpoint"]?
|
content_html = html_content || ""
|
||||||
json.field "authorId", node_comment["authorEndpoint"]["browseEndpoint"]["browseId"]
|
|
||||||
json.field "authorUrl", node_comment["authorEndpoint"]["browseEndpoint"]["canonicalBaseUrl"]
|
|
||||||
else
|
|
||||||
json.field "authorId", ""
|
|
||||||
json.field "authorUrl", ""
|
|
||||||
end
|
|
||||||
|
|
||||||
published_text = node_comment["publishedTimeText"]["runs"][0]["text"].as_s
|
|
||||||
published = decode_date(published_text.rchop(" (edited)"))
|
|
||||||
|
|
||||||
if published_text.includes?(" (edited)")
|
|
||||||
json.field "isEdited", true
|
|
||||||
else
|
|
||||||
json.field "isEdited", false
|
|
||||||
end
|
|
||||||
|
|
||||||
json.field "content", html_to_content(content_html)
|
json.field "content", html_to_content(content_html)
|
||||||
json.field "contentHtml", content_html
|
json.field "contentHtml", content_html
|
||||||
|
|
||||||
json.field "isPinned", (node_comment["pinnedCommentBadge"]? != nil)
|
if published_text != nil
|
||||||
json.field "isSponsor", (node_comment["sponsorCommentBadge"]? != nil)
|
published_text = published_text.to_s
|
||||||
if node_comment["sponsorCommentBadge"]?
|
if published_text.includes?(" (edited)")
|
||||||
# Sponsor icon thumbnails always have one object and there's only ever the url property in it
|
json.field "isEdited", true
|
||||||
json.field "sponsorIconUrl", node_comment.dig("sponsorCommentBadge", "sponsorCommentBadgeRenderer", "customBadge", "thumbnails", 0, "url").to_s
|
published = decode_date(published_text.rchop(" (edited)"))
|
||||||
end
|
else
|
||||||
json.field "published", published.to_unix
|
json.field "isEdited", false
|
||||||
json.field "publishedText", translate(locale, "`x` ago", recode_date(published, locale))
|
published = decode_date(published_text)
|
||||||
|
|
||||||
comment_action_buttons_renderer = node_comment["actionButtons"]["commentActionButtonsRenderer"]
|
|
||||||
|
|
||||||
json.field "likeCount", comment_action_buttons_renderer["likeButton"]["toggleButtonRenderer"]["accessibilityData"]["accessibilityData"]["label"].as_s.scan(/\d/).map(&.[0]).join.to_i
|
|
||||||
json.field "commentId", node_comment["commentId"]
|
|
||||||
json.field "authorIsChannelOwner", node_comment["authorIsChannelOwner"]
|
|
||||||
|
|
||||||
if comment_action_buttons_renderer["creatorHeart"]?
|
|
||||||
hearth_data = comment_action_buttons_renderer["creatorHeart"]["creatorHeartRenderer"]["creatorThumbnail"]
|
|
||||||
json.field "creatorHeart" do
|
|
||||||
json.object do
|
|
||||||
json.field "creatorThumbnail", hearth_data["thumbnails"][-1]["url"]
|
|
||||||
json.field "creatorName", hearth_data["accessibility"]["accessibilityData"]["label"]
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
json.field "published", published.to_unix
|
||||||
|
json.field "publishedText", translate(locale, "`x` ago", recode_date(published, locale))
|
||||||
end
|
end
|
||||||
|
|
||||||
if node_replies && !response["commentRepliesContinuation"]?
|
if node_replies && !response["commentRepliesContinuation"]?
|
||||||
|
@ -210,7 +277,7 @@ module Invidious::Comments
|
||||||
|
|
||||||
json.field "replies" do
|
json.field "replies" do
|
||||||
json.object do
|
json.object do
|
||||||
json.field "replyCount", node_comment["replyCount"]? || 1
|
json.field "replyCount", reply_count || 1
|
||||||
json.field "continuation", continuation
|
json.field "continuation", continuation
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -236,7 +303,6 @@ module Invidious::Comments
|
||||||
if format == "html"
|
if format == "html"
|
||||||
response = JSON.parse(response)
|
response = JSON.parse(response)
|
||||||
content_html = Frontend::Comments.template_youtube(response, locale, thin_mode)
|
content_html = Frontend::Comments.template_youtube(response, locale, thin_mode)
|
||||||
|
|
||||||
response = 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
|
||||||
|
|
|
@ -13,6 +13,7 @@ struct ConfigPreferences
|
||||||
|
|
||||||
property annotations : Bool = false
|
property annotations : Bool = false
|
||||||
property annotations_subscribed : Bool = false
|
property annotations_subscribed : Bool = false
|
||||||
|
property preload : Bool = true
|
||||||
property autoplay : Bool = false
|
property autoplay : Bool = false
|
||||||
property captions : Array(String) = ["", "", ""]
|
property captions : Array(String) = ["", "", ""]
|
||||||
property comments : Array(String) = ["youtube", ""]
|
property comments : Array(String) = ["youtube", ""]
|
||||||
|
@ -44,6 +45,8 @@ struct ConfigPreferences
|
||||||
property vr_mode : Bool = true
|
property vr_mode : Bool = true
|
||||||
property show_nick : Bool = true
|
property show_nick : Bool = true
|
||||||
property save_player_pos : Bool = false
|
property save_player_pos : Bool = false
|
||||||
|
property po_token : String = ""
|
||||||
|
property visitor_data : String = ""
|
||||||
|
|
||||||
def to_tuple
|
def to_tuple
|
||||||
{% begin %}
|
{% begin %}
|
||||||
|
@ -62,12 +65,12 @@ class Config
|
||||||
# Time interval between two executions of the job that crawls channel videos (subscriptions update).
|
# Time interval between two executions of the job that crawls channel videos (subscriptions update).
|
||||||
@[YAML::Field(converter: Preferences::TimeSpanConverter)]
|
@[YAML::Field(converter: Preferences::TimeSpanConverter)]
|
||||||
property channel_refresh_interval : Time::Span = 30.minutes
|
property channel_refresh_interval : Time::Span = 30.minutes
|
||||||
# Number of threads to use for updating feeds
|
|
||||||
property feed_threads : Int32 = 1
|
|
||||||
# Log file path or STDOUT
|
# Log file path or STDOUT
|
||||||
property output : String = "STDOUT"
|
property output : String = "STDOUT"
|
||||||
# Default log level, valid YAML values are ints and strings, see src/invidious/helpers/logger.cr
|
# Default log level, valid YAML values are ints and strings, see src/invidious/helpers/logger.cr
|
||||||
property log_level : LogLevel = LogLevel::Info
|
property log_level : LogLevel = LogLevel::Info
|
||||||
|
# Enables colors in logs. Useful for debugging purposes
|
||||||
|
property colorize_logs : Bool = false
|
||||||
# Database configuration with separate parameters (username, hostname, etc)
|
# Database configuration with separate parameters (username, hostname, etc)
|
||||||
property db : DBConfig? = nil
|
property db : DBConfig? = nil
|
||||||
|
|
||||||
|
@ -86,10 +89,18 @@ class Config
|
||||||
|
|
||||||
# Used to tell Invidious it is behind a proxy, so links to resources should be https://
|
# Used to tell Invidious it is behind a proxy, so links to resources should be https://
|
||||||
property https_only : Bool?
|
property https_only : Bool?
|
||||||
|
# Enable or disable CSP
|
||||||
|
property csp : Bool? = true
|
||||||
# HMAC signing key for CSRF tokens and verifying pubsub subscriptions
|
# HMAC signing key for CSRF tokens and verifying pubsub subscriptions
|
||||||
property hmac_key : String = ""
|
property hmac_key : String = ""
|
||||||
# Domain to be used for links to resources on the site where an absolute URL is required
|
# Domain to be used for links to resources on the site where an absolute URL is required
|
||||||
property domain : String?
|
property domain : String?
|
||||||
|
# Alternative domains. You can add other domains, like TOR and I2P addresses
|
||||||
|
property alternative_domains : Array(String) = [] of String
|
||||||
|
property donation_url : String?
|
||||||
|
property contact_url : String?
|
||||||
|
property home_domain : String?
|
||||||
|
|
||||||
# Subscribe to channels using PubSubHubbub (requires domain, hmac_key)
|
# Subscribe to channels using PubSubHubbub (requires domain, hmac_key)
|
||||||
property use_pubsub_feeds : Bool | Int32 = false
|
property use_pubsub_feeds : Bool | Int32 = false
|
||||||
property popular_enabled : Bool = true
|
property popular_enabled : Bool = true
|
||||||
|
@ -106,8 +117,6 @@ class Config
|
||||||
property check_tables : Bool = false
|
property check_tables : Bool = false
|
||||||
# Cache annotations requested from IA, will not cache empty annotations or annotations that only contain cards
|
# Cache annotations requested from IA, will not cache empty annotations or annotations that only contain cards
|
||||||
property cache_annotations : Bool = false
|
property cache_annotations : Bool = false
|
||||||
# Optional banner to be displayed along top of page for announcements, etc.
|
|
||||||
property banner : String? = nil
|
|
||||||
# Enables 'Strict-Transport-Security'. Ensure that `domain` and all subdomains are served securely
|
# Enables 'Strict-Transport-Security'. Ensure that `domain` and all subdomains are served securely
|
||||||
property hsts : Bool? = true
|
property hsts : Bool? = true
|
||||||
# Disable proxying server-wide: options: 'dash', 'livestreams', 'downloads', 'local'
|
# Disable proxying server-wide: options: 'dash', 'livestreams', 'downloads', 'local'
|
||||||
|
@ -115,13 +124,31 @@ class Config
|
||||||
# Enable the user notifications for all users
|
# Enable the user notifications for all users
|
||||||
property enable_user_notifications : Bool = true
|
property enable_user_notifications : Bool = true
|
||||||
|
|
||||||
|
# Optional banner to be displayed along top of page for announcements, etc.
|
||||||
|
property banner : String? = nil
|
||||||
|
# Optional footer text to be displayed within Invidious' footer. Can be used for maintainer contact info, etc.
|
||||||
|
property footer : String? = nil
|
||||||
|
# Email to contact the instance maintainer. This is used within the footer as an mailto link.
|
||||||
|
property instance_maintainer_email : String? = nil
|
||||||
# URL to the modified source code to be easily AGPL compliant
|
# URL to the modified source code to be easily AGPL compliant
|
||||||
# Will display in the footer, next to the main source code link
|
# Will display in the footer
|
||||||
property modified_source_code_url : String? = nil
|
property modified_source_code_url : String? = nil
|
||||||
|
# Link to the terms of service of the instance (if any). Will be displayed in the footer.
|
||||||
|
property footer_instance_tos_link : String? = nil
|
||||||
|
# Link to the privacy policy of the instance (if any). Will be displayed in the footer.
|
||||||
|
property footer_instance_privacy_policy_link : String? = nil
|
||||||
|
# Instance donation URL displayed in the "Instance" section of the footer
|
||||||
|
property footer_instance_donate_link : String? = nil
|
||||||
|
# Custom fields to be displayed within the footer's instance section
|
||||||
|
property footer_instance_section_custom_fields : Array(Array(String)) = [] of Array(String)
|
||||||
|
|
||||||
# Connect to YouTube over 'ipv6', 'ipv4'. Will sometimes resolve fix issues with rate-limiting (see https://github.com/ytdl-org/youtube-dl/issues/21729)
|
# Connect to YouTube over 'ipv6', 'ipv4'. Will sometimes resolve fix issues with rate-limiting (see https://github.com/ytdl-org/youtube-dl/issues/21729)
|
||||||
@[YAML::Field(converter: Preferences::FamilyConverter)]
|
@[YAML::Field(converter: Preferences::FamilyConverter)]
|
||||||
property force_resolve : Socket::Family = Socket::Family::UNSPEC
|
property force_resolve : Socket::Family = Socket::Family::UNSPEC
|
||||||
|
|
||||||
|
# External signature solver server socket (either a path to a UNIX domain socket or "<IP>:<Port>")
|
||||||
|
property signature_server : String? = nil
|
||||||
|
|
||||||
# Port to listen for connections (overridden by command line argument)
|
# Port to listen for connections (overridden by command line argument)
|
||||||
property port : Int32 = 3000
|
property port : Int32 = 3000
|
||||||
# Host to bind (overridden by command line argument)
|
# Host to bind (overridden by command line argument)
|
||||||
|
@ -132,6 +159,11 @@ class Config
|
||||||
# Use Innertube's transcripts API instead of timedtext for closed captions
|
# Use Innertube's transcripts API instead of timedtext for closed captions
|
||||||
property use_innertube_for_captions : Bool = false
|
property use_innertube_for_captions : Bool = false
|
||||||
|
|
||||||
|
# visitor data ID for Google session
|
||||||
|
property visitor_data : String? = nil
|
||||||
|
# poToken for passing bot attestation
|
||||||
|
property po_token : String? = nil
|
||||||
|
|
||||||
# Saved cookies in "name1=value1; name2=value2..." format
|
# Saved cookies in "name1=value1; name2=value2..." format
|
||||||
@[YAML::Field(converter: Preferences::StringToCookies)]
|
@[YAML::Field(converter: Preferences::StringToCookies)]
|
||||||
property cookies : HTTP::Cookies = HTTP::Cookies.new
|
property cookies : HTTP::Cookies = HTTP::Cookies.new
|
||||||
|
@ -139,6 +171,25 @@ class Config
|
||||||
# Playlist length limit
|
# Playlist length limit
|
||||||
property playlist_length_limit : Int32 = 500
|
property playlist_length_limit : Int32 = 500
|
||||||
|
|
||||||
|
# The max resolution the Instance can offer
|
||||||
|
property max_dash_resolution : Int32?
|
||||||
|
|
||||||
|
# List of names of the backends
|
||||||
|
property backends : Array(String) = [] of String
|
||||||
|
# Character used to separate the backend number from the description/note
|
||||||
|
# of the backend
|
||||||
|
property backends_delimiter : String = "|"
|
||||||
|
|
||||||
|
# External videoplayback proxies list. They should include `https://`
|
||||||
|
# at the start of the URI
|
||||||
|
property external_videoplayback_proxy : Array(String) = [] of String
|
||||||
|
|
||||||
|
# Job to refresh tokens from a Redis compatible DB
|
||||||
|
property refresh_tokens : Bool = true
|
||||||
|
|
||||||
|
# Materialious redirects
|
||||||
|
property materialious_domain : String?
|
||||||
|
|
||||||
def disabled?(option)
|
def disabled?(option)
|
||||||
case disabled = CONFIG.disable_proxy
|
case disabled = CONFIG.disable_proxy
|
||||||
when Bool
|
when Bool
|
||||||
|
|
|
@ -140,6 +140,7 @@ module Invidious::Database::Playlists
|
||||||
request = <<-SQL
|
request = <<-SQL
|
||||||
SELECT id,title FROM playlists
|
SELECT id,title FROM playlists
|
||||||
WHERE author = $1 AND id LIKE 'IV%'
|
WHERE author = $1 AND id LIKE 'IV%'
|
||||||
|
ORDER BY title
|
||||||
SQL
|
SQL
|
||||||
|
|
||||||
PG_DB.query_all(request, email, as: {String, String})
|
PG_DB.query_all(request, email, as: {String, String})
|
||||||
|
|
|
@ -154,15 +154,17 @@ module Invidious::Database::Users
|
||||||
# Update (misc)
|
# Update (misc)
|
||||||
# -------------------
|
# -------------------
|
||||||
|
|
||||||
def feed_needs_update(video : ChannelVideo)
|
# Feeds never need update. PubSubHubBub is the one that sends videos to
|
||||||
request = <<-SQL
|
# invidious.
|
||||||
UPDATE users
|
# def feed_needs_update(video : ChannelVideo)
|
||||||
SET feed_needs_update = true
|
# request = <<-SQL
|
||||||
WHERE $1 = ANY(subscriptions)
|
# UPDATE users
|
||||||
SQL
|
# SET feed_needs_update = true
|
||||||
|
# WHERE $1 = ANY(subscriptions)
|
||||||
|
# SQL
|
||||||
|
|
||||||
PG_DB.exec(request, video.ucid)
|
# PG_DB.exec(request, video.ucid)
|
||||||
end
|
# end
|
||||||
|
|
||||||
def update_preferences(user : User)
|
def update_preferences(user : User)
|
||||||
request = <<-SQL
|
request = <<-SQL
|
||||||
|
|
|
@ -10,8 +10,8 @@ module Invidious::Database::Videos
|
||||||
ON CONFLICT (id) DO NOTHING
|
ON CONFLICT (id) DO NOTHING
|
||||||
SQL
|
SQL
|
||||||
|
|
||||||
REDIS_DB.set(video.id, video.info.to_json, ex: 3600)
|
REDIS_DB.set(video.id, video.info.to_json, ex: 14400)
|
||||||
REDIS_DB.set(video.id + ":time", video.updated, ex: 3600)
|
REDIS_DB.set(video.id + ":time", video.updated, ex: 14400)
|
||||||
end
|
end
|
||||||
|
|
||||||
def delete(id)
|
def delete(id)
|
||||||
|
|
|
@ -149,12 +149,12 @@ module Invidious::Frontend::Comments
|
||||||
|
|
||||||
if comments["videoId"]?
|
if comments["videoId"]?
|
||||||
html << <<-END_HTML
|
html << <<-END_HTML
|
||||||
<a href="https://www.youtube.com/watch?v=#{comments["videoId"]}&lc=#{child["commentId"]}" title="#{translate(locale, "YouTube comment permalink")}">[YT]</a>
|
<a rel="noreferrer noopener" href="https://www.youtube.com/watch?v=#{comments["videoId"]}&lc=#{child["commentId"]}" title="#{translate(locale, "YouTube comment permalink")}">[YT]</a>
|
||||||
|
|
|
|
||||||
END_HTML
|
END_HTML
|
||||||
elsif comments["authorId"]?
|
elsif comments["authorId"]?
|
||||||
html << <<-END_HTML
|
html << <<-END_HTML
|
||||||
<a href="https://www.youtube.com/channel/#{comments["authorId"]}/community?lb=#{child["commentId"]}" title="#{translate(locale, "YouTube comment permalink")}">[YT]</a>
|
<a rel="noreferrer noopener" 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
|
end
|
||||||
|
|
|
@ -6,9 +6,9 @@ module Invidious::Frontend::Misc
|
||||||
|
|
||||||
if prefs.automatic_instance_redirect
|
if prefs.automatic_instance_redirect
|
||||||
current_page = env.get?("current_page").as(String)
|
current_page = env.get?("current_page").as(String)
|
||||||
redirect_url = "/redirect?referer=#{current_page}"
|
return "/redirect?referer=#{current_page}"
|
||||||
else
|
else
|
||||||
redirect_url = "https://redirect.invidious.io#{env.request.resource}"
|
return "https://redirect.invidious.io#{env.request.resource}"
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,9 +3,9 @@
|
||||||
# IPv6 addresses.
|
# IPv6 addresses.
|
||||||
#
|
#
|
||||||
class TCPSocket
|
class TCPSocket
|
||||||
def initialize(host : String, port, dns_timeout = nil, connect_timeout = nil, family = Socket::Family::UNSPEC)
|
def initialize(host, port, dns_timeout = nil, connect_timeout = nil, blocking = false, family = Socket::Family::UNSPEC)
|
||||||
Addrinfo.tcp(host, port, timeout: dns_timeout, family: family) do |addrinfo|
|
Addrinfo.tcp(host, port, timeout: dns_timeout, family: family) do |addrinfo|
|
||||||
super(addrinfo.family, addrinfo.type, addrinfo.protocol)
|
super(addrinfo.family, addrinfo.type, addrinfo.protocol, blocking)
|
||||||
connect(addrinfo, timeout: connect_timeout) do |error|
|
connect(addrinfo, timeout: connect_timeout) do |error|
|
||||||
close
|
close
|
||||||
error
|
error
|
||||||
|
@ -26,7 +26,7 @@ class HTTP::Client
|
||||||
end
|
end
|
||||||
|
|
||||||
hostname = @host.starts_with?('[') && @host.ends_with?(']') ? @host[1..-2] : @host
|
hostname = @host.starts_with?('[') && @host.ends_with?(']') ? @host[1..-2] : @host
|
||||||
io = TCPSocket.new hostname, @port, @dns_timeout, @connect_timeout, @family
|
io = TCPSocket.new hostname, @port, @dns_timeout, @connect_timeout, family: @family
|
||||||
io.read_timeout = @read_timeout if @read_timeout
|
io.read_timeout = @read_timeout if @read_timeout
|
||||||
io.write_timeout = @write_timeout if @write_timeout
|
io.write_timeout = @write_timeout if @write_timeout
|
||||||
io.sync = false
|
io.sync = false
|
||||||
|
@ -35,7 +35,7 @@ class HTTP::Client
|
||||||
if tls = @tls
|
if tls = @tls
|
||||||
tcp_socket = io
|
tcp_socket = io
|
||||||
begin
|
begin
|
||||||
io = OpenSSL::SSL::Socket::Client.new(tcp_socket, context: tls, sync_close: true, hostname: @host)
|
io = OpenSSL::SSL::Socket::Client.new(tcp_socket, context: tls, sync_close: true, hostname: @host.rchop('.'))
|
||||||
rescue exc
|
rescue exc
|
||||||
# don't leak the TCP socket when the SSL connection failed
|
# don't leak the TCP socket when the SSL connection failed
|
||||||
tcp_socket.close
|
tcp_socket.close
|
||||||
|
|
|
@ -43,6 +43,8 @@ def error_template_helper(env : HTTP::Server::Context, status_code : Int32, exce
|
||||||
# URLs for the error message below
|
# URLs for the error message below
|
||||||
url_faq = "https://github.com/iv-org/documentation/blob/master/docs/faq.md"
|
url_faq = "https://github.com/iv-org/documentation/blob/master/docs/faq.md"
|
||||||
url_search_issues = "https://github.com/iv-org/invidious/issues"
|
url_search_issues = "https://github.com/iv-org/invidious/issues"
|
||||||
|
url_search_issues += "?q=is:issue+is:open+"
|
||||||
|
url_search_issues += URI.encode_www_form("[Bug] #{issue_title}")
|
||||||
|
|
||||||
url_switch = "https://redirect.invidious.io" + env.request.resource
|
url_switch = "https://redirect.invidious.io" + env.request.resource
|
||||||
|
|
||||||
|
@ -190,7 +192,7 @@ def error_redirect_helper(env : HTTP::Server::Context)
|
||||||
<a href="/redirect?referer=#{env.get("current_page")}">#{switch_instance}</a>
|
<a href="/redirect?referer=#{env.get("current_page")}">#{switch_instance}</a>
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
<a href="https://youtube.com#{env.request.resource}">#{go_to_youtube}</a>
|
<a rel="noreferrer noopener" href="https://youtube.com#{env.request.resource}">#{go_to_youtube}</a>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
END_HTML
|
END_HTML
|
||||||
|
|
|
@ -97,7 +97,7 @@ class AuthHandler < Kemal::Handler
|
||||||
if token = env.request.headers["Authorization"]?
|
if token = env.request.headers["Authorization"]?
|
||||||
token = JSON.parse(URI.decode_www_form(token.lchop("Bearer ")))
|
token = JSON.parse(URI.decode_www_form(token.lchop("Bearer ")))
|
||||||
session = URI.decode_www_form(token["session"].as_s)
|
session = URI.decode_www_form(token["session"].as_s)
|
||||||
scopes, expire, signature = validate_request(token, session, env.request, HMAC_KEY, nil)
|
scopes, _, _ = validate_request(token, session, env.request, HMAC_KEY, nil)
|
||||||
|
|
||||||
if email = Invidious::Database::SessionIDs.select_email(session)
|
if email = Invidious::Database::SessionIDs.select_email(session)
|
||||||
user = Invidious::Database::Users.select!(email: email)
|
user = Invidious::Database::Users.select!(email: email)
|
||||||
|
|
|
@ -95,7 +95,6 @@ module I18next::Plurals
|
||||||
"hr" => PluralForms::Special_Hungarian_Serbian,
|
"hr" => PluralForms::Special_Hungarian_Serbian,
|
||||||
"it" => PluralForms::Special_Spanish_Italian,
|
"it" => PluralForms::Special_Spanish_Italian,
|
||||||
"pt" => PluralForms::Special_French_Portuguese,
|
"pt" => PluralForms::Special_French_Portuguese,
|
||||||
"pt" => PluralForms::Special_French_Portuguese,
|
|
||||||
"sr" => PluralForms::Special_Hungarian_Serbian,
|
"sr" => PluralForms::Special_Hungarian_Serbian,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -189,7 +188,7 @@ module I18next::Plurals
|
||||||
|
|
||||||
# Emulate the `rule.numbers.size == 2 && rule.numbers[0] == 1` check
|
# Emulate the `rule.numbers.size == 2 && rule.numbers[0] == 1` check
|
||||||
# from original i18next code
|
# from original i18next code
|
||||||
private def is_simple_plural(form : PluralForms) : Bool
|
private def simple_plural?(form : PluralForms) : Bool
|
||||||
case form
|
case form
|
||||||
when .single_gt_one? then return true
|
when .single_gt_one? then return true
|
||||||
when .single_not_one? then return true
|
when .single_not_one? then return true
|
||||||
|
@ -211,7 +210,7 @@ module I18next::Plurals
|
||||||
idx = SuffixIndex.get_index(plural_form, count)
|
idx = SuffixIndex.get_index(plural_form, count)
|
||||||
|
|
||||||
# Simple plurals are handled differently in all versions (but v4)
|
# Simple plurals are handled differently in all versions (but v4)
|
||||||
if @simplify_plural_suffix && is_simple_plural(plural_form)
|
if @simplify_plural_suffix && simple_plural?(plural_form)
|
||||||
return (idx == 1) ? "_plural" : ""
|
return (idx == 1) ? "_plural" : ""
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -262,9 +261,9 @@ module I18next::Plurals
|
||||||
when .special_hebrew? then return special_hebrew(count)
|
when .special_hebrew? then return special_hebrew(count)
|
||||||
when .special_odia? then return special_odia(count)
|
when .special_odia? then return special_odia(count)
|
||||||
# Mixed v3/v4 forms
|
# Mixed v3/v4 forms
|
||||||
when .special_spanish_italian? then return special_cldr_Spanish_Italian(count)
|
when .special_spanish_italian? then return special_cldr_spanish_italian(count)
|
||||||
when .special_french_portuguese? then return special_cldr_French_Portuguese(count)
|
when .special_french_portuguese? then return special_cldr_french_portuguese(count)
|
||||||
when .special_hungarian_serbian? then return special_cldr_Hungarian_Serbian(count)
|
when .special_hungarian_serbian? then return special_cldr_hungarian_serbian(count)
|
||||||
else
|
else
|
||||||
# default, if nothing matched above
|
# default, if nothing matched above
|
||||||
return 0_u8
|
return 0_u8
|
||||||
|
@ -535,7 +534,7 @@ module I18next::Plurals
|
||||||
#
|
#
|
||||||
# This rule is mostly compliant to CLDR v42
|
# This rule is mostly compliant to CLDR v42
|
||||||
#
|
#
|
||||||
def self.special_cldr_Spanish_Italian(count : Int) : UInt8
|
def self.special_cldr_spanish_italian(count : Int) : UInt8
|
||||||
return 0_u8 if (count == 1) # one
|
return 0_u8 if (count == 1) # one
|
||||||
return 1_u8 if (count != 0 && count % 1_000_000 == 0) # many
|
return 1_u8 if (count != 0 && count % 1_000_000 == 0) # many
|
||||||
return 2_u8 # other
|
return 2_u8 # other
|
||||||
|
@ -545,7 +544,7 @@ module I18next::Plurals
|
||||||
#
|
#
|
||||||
# This rule is mostly compliant to CLDR v42
|
# This rule is mostly compliant to CLDR v42
|
||||||
#
|
#
|
||||||
def self.special_cldr_French_Portuguese(count : Int) : UInt8
|
def self.special_cldr_french_portuguese(count : Int) : UInt8
|
||||||
return 0_u8 if (count == 0 || count == 1) # one
|
return 0_u8 if (count == 0 || count == 1) # one
|
||||||
return 1_u8 if (count % 1_000_000 == 0) # many
|
return 1_u8 if (count % 1_000_000 == 0) # many
|
||||||
return 2_u8 # other
|
return 2_u8 # other
|
||||||
|
@ -555,7 +554,7 @@ module I18next::Plurals
|
||||||
#
|
#
|
||||||
# This rule is mostly compliant to CLDR v42
|
# This rule is mostly compliant to CLDR v42
|
||||||
#
|
#
|
||||||
def self.special_cldr_Hungarian_Serbian(count : Int) : UInt8
|
def self.special_cldr_hungarian_serbian(count : Int) : UInt8
|
||||||
n_mod_10 = count % 10
|
n_mod_10 = count % 10
|
||||||
n_mod_100 = count % 100
|
n_mod_100 = count % 100
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,5 @@
|
||||||
|
require "colorize"
|
||||||
|
|
||||||
enum LogLevel
|
enum LogLevel
|
||||||
All = 0
|
All = 0
|
||||||
Trace = 1
|
Trace = 1
|
||||||
|
@ -10,7 +12,7 @@ enum LogLevel
|
||||||
end
|
end
|
||||||
|
|
||||||
class Invidious::LogHandler < Kemal::BaseLogHandler
|
class Invidious::LogHandler < Kemal::BaseLogHandler
|
||||||
def initialize(@io : IO = STDOUT, @level = LogLevel::Debug)
|
def initialize(@io : IO = STDOUT, @level = LogLevel::Debug, @color : Bool = true)
|
||||||
end
|
end
|
||||||
|
|
||||||
def call(context : HTTP::Server::Context)
|
def call(context : HTTP::Server::Context)
|
||||||
|
@ -34,28 +36,28 @@ class Invidious::LogHandler < Kemal::BaseLogHandler
|
||||||
context
|
context
|
||||||
end
|
end
|
||||||
|
|
||||||
def puts(message : String)
|
|
||||||
@io << message << '\n'
|
|
||||||
@io.flush
|
|
||||||
end
|
|
||||||
|
|
||||||
def write(message : String)
|
def write(message : String)
|
||||||
@io << message
|
@io << message
|
||||||
@io.flush
|
@io.flush
|
||||||
end
|
end
|
||||||
|
|
||||||
def set_log_level(level : String)
|
def color(level)
|
||||||
@level = LogLevel.parse(level)
|
case level
|
||||||
end
|
when LogLevel::Trace then :cyan
|
||||||
|
when LogLevel::Debug then :green
|
||||||
def set_log_level(level : LogLevel)
|
when LogLevel::Info then :white
|
||||||
@level = level
|
when LogLevel::Warn then :yellow
|
||||||
|
when LogLevel::Error then :red
|
||||||
|
when LogLevel::Fatal then :magenta
|
||||||
|
else :default
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
{% for level in %w(trace debug info warn error fatal) %}
|
{% for level in %w(trace debug info warn error fatal) %}
|
||||||
def {{level.id}}(message : String)
|
def {{level.id}}(message : String)
|
||||||
if LogLevel::{{level.id.capitalize}} >= @level
|
if LogLevel::{{level.id.capitalize}} >= @level
|
||||||
puts("#{Time.utc} [{{level.id}}] #{message}")
|
puts("#{Time.utc} [{{level.id}}] #{message}".colorize(color(LogLevel::{{level.id.capitalize}})).toggle(@color))
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
|
@ -48,8 +48,9 @@ module JSON::Serializable
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
macro templated(_filename, template = "template", navbar_search = true)
|
macro templated(_filename, template = "template", navbar_search = true, buffer_footer = false)
|
||||||
navbar_search = {{navbar_search}}
|
navbar_search = {{navbar_search}}
|
||||||
|
buffer_footer = {{buffer_footer}}
|
||||||
|
|
||||||
{{ filename = "src/invidious/views/" + _filename + ".ecr" }}
|
{{ filename = "src/invidious/views/" + _filename + ".ecr" }}
|
||||||
{{ layout = "src/invidious/views/" + template + ".ecr" }}
|
{{ layout = "src/invidious/views/" + template + ".ecr" }}
|
||||||
|
|
25
src/invidious/helpers/redis_tokens.cr
Normal file
25
src/invidious/helpers/redis_tokens.cr
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
module Tokens
|
||||||
|
extend self
|
||||||
|
@@po_token : String | Nil
|
||||||
|
@@visitor_data : String | Nil
|
||||||
|
|
||||||
|
def refresh_tokens
|
||||||
|
@@po_token = REDIS_DB.get("invidious:po_token")
|
||||||
|
@@visitor_data = REDIS_DB.get("invidious:visitor_data")
|
||||||
|
LOGGER.debug("RefreshTokens: Tokens are:")
|
||||||
|
LOGGER.debug("RefreshTokens: po_token: #{@@po_token}")
|
||||||
|
LOGGER.debug("RefreshTokens: visitor_data: #{@@visitor_data}")
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_tokens
|
||||||
|
return {@@po_token, @@visitor_data}
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_po_token
|
||||||
|
return @@po_token
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_visitor_data
|
||||||
|
return @@visitor_data
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,3 +1,16 @@
|
||||||
|
@[Flags]
|
||||||
|
enum VideoBadges
|
||||||
|
LiveNow
|
||||||
|
Premium
|
||||||
|
ThreeD
|
||||||
|
FourK
|
||||||
|
New
|
||||||
|
EightK
|
||||||
|
VR180
|
||||||
|
VR360
|
||||||
|
ClosedCaptions
|
||||||
|
end
|
||||||
|
|
||||||
struct SearchVideo
|
struct SearchVideo
|
||||||
include DB::Serializable
|
include DB::Serializable
|
||||||
|
|
||||||
|
@ -9,10 +22,9 @@ struct SearchVideo
|
||||||
property views : Int64
|
property views : Int64
|
||||||
property description_html : String
|
property description_html : String
|
||||||
property length_seconds : Int32
|
property length_seconds : Int32
|
||||||
property live_now : Bool
|
|
||||||
property premium : Bool
|
|
||||||
property premiere_timestamp : Time?
|
property premiere_timestamp : Time?
|
||||||
property author_verified : Bool
|
property author_verified : Bool
|
||||||
|
property badges : VideoBadges
|
||||||
|
|
||||||
def to_xml(auto_generated, query_params, xml : XML::Builder)
|
def to_xml(auto_generated, query_params, xml : XML::Builder)
|
||||||
query_params["v"] = self.id
|
query_params["v"] = self.id
|
||||||
|
@ -88,13 +100,20 @@ struct SearchVideo
|
||||||
json.field "published", self.published.to_unix
|
json.field "published", self.published.to_unix
|
||||||
json.field "publishedText", translate(locale, "`x` ago", recode_date(self.published, locale))
|
json.field "publishedText", translate(locale, "`x` ago", recode_date(self.published, locale))
|
||||||
json.field "lengthSeconds", self.length_seconds
|
json.field "lengthSeconds", self.length_seconds
|
||||||
json.field "liveNow", self.live_now
|
json.field "liveNow", self.badges.live_now?
|
||||||
json.field "premium", self.premium
|
json.field "premium", self.badges.premium?
|
||||||
json.field "isUpcoming", self.is_upcoming
|
json.field "isUpcoming", self.upcoming?
|
||||||
|
|
||||||
if self.premiere_timestamp
|
if self.premiere_timestamp
|
||||||
json.field "premiereTimestamp", self.premiere_timestamp.try &.to_unix
|
json.field "premiereTimestamp", self.premiere_timestamp.try &.to_unix
|
||||||
end
|
end
|
||||||
|
json.field "isNew", self.badges.new?
|
||||||
|
json.field "is4k", self.badges.four_k?
|
||||||
|
json.field "is8k", self.badges.eight_k?
|
||||||
|
json.field "isVr180", self.badges.vr180?
|
||||||
|
json.field "isVr360", self.badges.vr360?
|
||||||
|
json.field "is3d", self.badges.three_d?
|
||||||
|
json.field "hasCaptions", self.badges.closed_captions?
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -109,7 +128,7 @@ struct SearchVideo
|
||||||
to_json(nil, json)
|
to_json(nil, json)
|
||||||
end
|
end
|
||||||
|
|
||||||
def is_upcoming
|
def upcoming?
|
||||||
premiere_timestamp ? true : false
|
premiere_timestamp ? true : false
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
350
src/invidious/helpers/sig_helper.cr
Normal file
350
src/invidious/helpers/sig_helper.cr
Normal file
|
@ -0,0 +1,350 @@
|
||||||
|
require "uri"
|
||||||
|
require "socket"
|
||||||
|
require "socket/tcp_socket"
|
||||||
|
require "socket/unix_socket"
|
||||||
|
|
||||||
|
{% if flag?(:advanced_debug) %}
|
||||||
|
require "io/hexdump"
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
private alias NetworkEndian = IO::ByteFormat::NetworkEndian
|
||||||
|
|
||||||
|
module Invidious::SigHelper
|
||||||
|
enum UpdateStatus
|
||||||
|
Updated
|
||||||
|
UpdateNotRequired
|
||||||
|
Error
|
||||||
|
end
|
||||||
|
|
||||||
|
# -------------------
|
||||||
|
# Payload types
|
||||||
|
# -------------------
|
||||||
|
|
||||||
|
abstract struct Payload
|
||||||
|
end
|
||||||
|
|
||||||
|
struct StringPayload < Payload
|
||||||
|
getter string : String
|
||||||
|
|
||||||
|
def initialize(str : String)
|
||||||
|
raise Exception.new("SigHelper: String can't be empty") if str.empty?
|
||||||
|
@string = str
|
||||||
|
end
|
||||||
|
|
||||||
|
def self.from_bytes(slice : Bytes)
|
||||||
|
size = IO::ByteFormat::NetworkEndian.decode(UInt16, slice)
|
||||||
|
if size == 0 # Error code
|
||||||
|
raise Exception.new("SigHelper: Server encountered an error")
|
||||||
|
end
|
||||||
|
|
||||||
|
if (slice.bytesize - 2) != size
|
||||||
|
raise Exception.new("SigHelper: String size mismatch")
|
||||||
|
end
|
||||||
|
|
||||||
|
if str = String.new(slice[2..])
|
||||||
|
return self.new(str)
|
||||||
|
else
|
||||||
|
raise Exception.new("SigHelper: Can't read string from socket")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def to_io(io)
|
||||||
|
# `.to_u16` raises if there is an overflow during the conversion
|
||||||
|
io.write_bytes(@string.bytesize.to_u16, NetworkEndian)
|
||||||
|
io.write(@string.to_slice)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private enum Opcode
|
||||||
|
FORCE_UPDATE = 0
|
||||||
|
DECRYPT_N_SIGNATURE = 1
|
||||||
|
DECRYPT_SIGNATURE = 2
|
||||||
|
GET_SIGNATURE_TIMESTAMP = 3
|
||||||
|
GET_PLAYER_STATUS = 4
|
||||||
|
PLAYER_UPDATE_TIMESTAMP = 5
|
||||||
|
end
|
||||||
|
|
||||||
|
private record Request,
|
||||||
|
opcode : Opcode,
|
||||||
|
payload : Payload?
|
||||||
|
|
||||||
|
# ----------------------
|
||||||
|
# High-level functions
|
||||||
|
# ----------------------
|
||||||
|
|
||||||
|
class Client
|
||||||
|
@mux : Multiplexor
|
||||||
|
|
||||||
|
def initialize(uri_or_path)
|
||||||
|
@mux = Multiplexor.new(uri_or_path)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Forces the server to re-fetch the YouTube player, and extract the necessary
|
||||||
|
# components from it (nsig function code, sig function code, signature timestamp).
|
||||||
|
def force_update : UpdateStatus
|
||||||
|
request = Request.new(Opcode::FORCE_UPDATE, nil)
|
||||||
|
|
||||||
|
value = send_request(request) do |bytes|
|
||||||
|
IO::ByteFormat::NetworkEndian.decode(UInt16, bytes)
|
||||||
|
end
|
||||||
|
|
||||||
|
case value
|
||||||
|
when 0x0000 then return UpdateStatus::Error
|
||||||
|
when 0xFFFF then return UpdateStatus::UpdateNotRequired
|
||||||
|
when 0xF44F then return UpdateStatus::Updated
|
||||||
|
else
|
||||||
|
code = value.nil? ? "nil" : value.to_s(base: 16)
|
||||||
|
raise Exception.new("SigHelper: Invalid status code received #{code}")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Decrypt a provided n signature using the server's current nsig function
|
||||||
|
# code, and return the result (or an error).
|
||||||
|
def decrypt_n_param(n : String) : String?
|
||||||
|
request = Request.new(Opcode::DECRYPT_N_SIGNATURE, StringPayload.new(n))
|
||||||
|
|
||||||
|
n_dec = self.send_request(request) do |bytes|
|
||||||
|
StringPayload.from_bytes(bytes).string
|
||||||
|
end
|
||||||
|
|
||||||
|
return n_dec
|
||||||
|
end
|
||||||
|
|
||||||
|
# Decrypt a provided s signature using the server's current sig function
|
||||||
|
# code, and return the result (or an error).
|
||||||
|
def decrypt_sig(sig : String) : String?
|
||||||
|
request = Request.new(Opcode::DECRYPT_SIGNATURE, StringPayload.new(sig))
|
||||||
|
|
||||||
|
sig_dec = self.send_request(request) do |bytes|
|
||||||
|
StringPayload.from_bytes(bytes).string
|
||||||
|
end
|
||||||
|
|
||||||
|
return sig_dec
|
||||||
|
end
|
||||||
|
|
||||||
|
# Return the signature timestamp from the server's current player
|
||||||
|
def get_signature_timestamp : UInt64?
|
||||||
|
request = Request.new(Opcode::GET_SIGNATURE_TIMESTAMP, nil)
|
||||||
|
|
||||||
|
return self.send_request(request) do |bytes|
|
||||||
|
IO::ByteFormat::NetworkEndian.decode(UInt64, bytes)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Return the current player's version
|
||||||
|
def get_player : UInt32?
|
||||||
|
request = Request.new(Opcode::GET_PLAYER_STATUS, nil)
|
||||||
|
|
||||||
|
return self.send_request(request) do |bytes|
|
||||||
|
has_player = (bytes[0] == 0xFF)
|
||||||
|
player_version = IO::ByteFormat::NetworkEndian.decode(UInt32, bytes[1..4])
|
||||||
|
has_player ? player_version : nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Return when the player was last updated
|
||||||
|
def get_player_timestamp : UInt64?
|
||||||
|
request = Request.new(Opcode::PLAYER_UPDATE_TIMESTAMP, nil)
|
||||||
|
|
||||||
|
return self.send_request(request) do |bytes|
|
||||||
|
IO::ByteFormat::NetworkEndian.decode(UInt64, bytes)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private def send_request(request : Request, &)
|
||||||
|
channel = @mux.send(request)
|
||||||
|
slice = channel.receive
|
||||||
|
return yield slice
|
||||||
|
rescue ex
|
||||||
|
LOGGER.debug("SigHelper: Error when sending a request")
|
||||||
|
LOGGER.trace(ex.inspect_with_backtrace)
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# ---------------------
|
||||||
|
# Low level functions
|
||||||
|
# ---------------------
|
||||||
|
|
||||||
|
class Multiplexor
|
||||||
|
alias TransactionID = UInt32
|
||||||
|
record Transaction, channel = ::Channel(Bytes).new
|
||||||
|
|
||||||
|
@prng = Random.new
|
||||||
|
@mutex = Mutex.new
|
||||||
|
@queue = {} of TransactionID => Transaction
|
||||||
|
|
||||||
|
@conn : Connection
|
||||||
|
|
||||||
|
@uri_or_path : String
|
||||||
|
|
||||||
|
def initialize(@uri_or_path)
|
||||||
|
@conn = Connection.new(uri_or_path)
|
||||||
|
listen
|
||||||
|
end
|
||||||
|
|
||||||
|
def listen : Nil
|
||||||
|
raise "Socket is closed" if @conn.closed?
|
||||||
|
|
||||||
|
LOGGER.debug("SigHelper: Multiplexor listening")
|
||||||
|
|
||||||
|
spawn do
|
||||||
|
loop do
|
||||||
|
begin
|
||||||
|
receive_data
|
||||||
|
rescue ex
|
||||||
|
LOGGER.info("SigHelper: Connection to helper died with '#{ex.message}' trying to reconnect...")
|
||||||
|
# We close the socket because for some reason is not closed.
|
||||||
|
@conn.close
|
||||||
|
loop do
|
||||||
|
begin
|
||||||
|
@conn = Connection.new(@uri_or_path)
|
||||||
|
LOGGER.info("SigHelper: Reconnected to SigHelper!")
|
||||||
|
rescue ex
|
||||||
|
LOGGER.debug("SigHelper: Reconnection to helper unsuccessful with error '#{ex.message}' retrying")
|
||||||
|
sleep 500.milliseconds
|
||||||
|
next
|
||||||
|
end
|
||||||
|
break if !@conn.closed?
|
||||||
|
end
|
||||||
|
end
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def send(request : Request)
|
||||||
|
transaction = Transaction.new
|
||||||
|
transaction_id = @prng.rand(TransactionID)
|
||||||
|
|
||||||
|
# Add transaction to queue
|
||||||
|
@mutex.synchronize do
|
||||||
|
# On a 32-bits random integer, this should never happen. Though, just in case, ...
|
||||||
|
if @queue[transaction_id]?
|
||||||
|
raise Exception.new("SigHelper: Duplicate transaction ID! You got a shiny pokemon!")
|
||||||
|
end
|
||||||
|
|
||||||
|
@queue[transaction_id] = transaction
|
||||||
|
end
|
||||||
|
|
||||||
|
write_packet(transaction_id, request)
|
||||||
|
|
||||||
|
return transaction.channel
|
||||||
|
end
|
||||||
|
|
||||||
|
def receive_data
|
||||||
|
transaction_id, slice = read_packet
|
||||||
|
|
||||||
|
@mutex.synchronize do
|
||||||
|
if transaction = @queue.delete(transaction_id)
|
||||||
|
# Remove transaction from queue and send data to the channel
|
||||||
|
transaction.channel.send(slice)
|
||||||
|
LOGGER.trace("SigHelper: Transaction unqueued and data sent to channel")
|
||||||
|
else
|
||||||
|
raise Exception.new("SigHelper: Received transaction was not in queue")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
# Read a single packet from the socket
|
||||||
|
private def read_packet : {TransactionID, Bytes}
|
||||||
|
# Header
|
||||||
|
transaction_id = @conn.read_bytes(UInt32, NetworkEndian)
|
||||||
|
length = @conn.read_bytes(UInt32, NetworkEndian)
|
||||||
|
|
||||||
|
LOGGER.trace("SigHelper: Recv transaction 0x#{transaction_id.to_s(base: 16)} / length #{length}")
|
||||||
|
|
||||||
|
if length > 67_000
|
||||||
|
raise Exception.new("SigHelper: Packet longer than expected (#{length})")
|
||||||
|
end
|
||||||
|
|
||||||
|
# Payload
|
||||||
|
slice = Bytes.new(length)
|
||||||
|
@conn.read(slice) if length > 0
|
||||||
|
|
||||||
|
LOGGER.trace("SigHelper: payload = #{slice}")
|
||||||
|
LOGGER.trace("SigHelper: Recv transaction 0x#{transaction_id.to_s(base: 16)} - Done")
|
||||||
|
|
||||||
|
return transaction_id, slice
|
||||||
|
end
|
||||||
|
|
||||||
|
# Write a single packet to the socket
|
||||||
|
private def write_packet(transaction_id : TransactionID, request : Request)
|
||||||
|
LOGGER.trace("SigHelper: Send transaction 0x#{transaction_id.to_s(base: 16)} / opcode #{request.opcode}")
|
||||||
|
|
||||||
|
io = IO::Memory.new(1024)
|
||||||
|
io.write_bytes(request.opcode.to_u8, NetworkEndian)
|
||||||
|
io.write_bytes(transaction_id, NetworkEndian)
|
||||||
|
|
||||||
|
if payload = request.payload
|
||||||
|
payload.to_io(io)
|
||||||
|
end
|
||||||
|
|
||||||
|
@conn.send(io)
|
||||||
|
@conn.flush
|
||||||
|
|
||||||
|
LOGGER.trace("SigHelper: Send transaction 0x#{transaction_id.to_s(base: 16)} - Done")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class Connection
|
||||||
|
@socket : UNIXSocket | TCPSocket
|
||||||
|
|
||||||
|
{% if flag?(:advanced_debug) %}
|
||||||
|
@io : IO::Hexdump
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
def initialize(host_or_path : String)
|
||||||
|
case host_or_path
|
||||||
|
when .starts_with?('/')
|
||||||
|
# Make sure that the file exists
|
||||||
|
if File.exists?(host_or_path)
|
||||||
|
@socket = UNIXSocket.new(host_or_path)
|
||||||
|
else
|
||||||
|
raise Exception.new("SigHelper: '#{host_or_path}' no such file")
|
||||||
|
end
|
||||||
|
when .starts_with?("tcp://")
|
||||||
|
uri = URI.parse(host_or_path)
|
||||||
|
@socket = TCPSocket.new(uri.host.not_nil!, uri.port.not_nil!)
|
||||||
|
else
|
||||||
|
uri = URI.parse("tcp://#{host_or_path}")
|
||||||
|
@socket = TCPSocket.new(uri.host.not_nil!, uri.port.not_nil!)
|
||||||
|
end
|
||||||
|
LOGGER.info("SigHelper: Using helper at '#{host_or_path}'")
|
||||||
|
|
||||||
|
{% if flag?(:advanced_debug) %}
|
||||||
|
@io = IO::Hexdump.new(@socket, output: STDERR, read: true, write: true)
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
@socket.sync = false
|
||||||
|
@socket.blocking = false
|
||||||
|
end
|
||||||
|
|
||||||
|
def closed? : Bool
|
||||||
|
return @socket.closed?
|
||||||
|
end
|
||||||
|
|
||||||
|
def close : Nil
|
||||||
|
@socket.close if !@socket.closed?
|
||||||
|
end
|
||||||
|
|
||||||
|
def flush(*args, **options)
|
||||||
|
@socket.flush(*args, **options)
|
||||||
|
end
|
||||||
|
|
||||||
|
def send(*args, **options)
|
||||||
|
@socket.send(*args, **options)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Wrap IO functions, with added debug tooling if needed
|
||||||
|
{% for function in %w(read read_bytes write write_bytes) %}
|
||||||
|
def {{function.id}}(*args, **options)
|
||||||
|
{% if flag?(:advanced_debug) %}
|
||||||
|
@io.{{function.id}}(*args, **options)
|
||||||
|
{% else %}
|
||||||
|
@socket.{{function.id}}(*args, **options)
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
{% end %}
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,73 +1,53 @@
|
||||||
alias SigProc = Proc(Array(String), Int32, Array(String))
|
require "http/params"
|
||||||
|
require "./sig_helper"
|
||||||
|
|
||||||
struct DecryptFunction
|
class Invidious::DecryptFunction
|
||||||
@decrypt_function = [] of {SigProc, Int32}
|
@last_update : Time = Time.utc - 42.days
|
||||||
@decrypt_time = Time.monotonic
|
|
||||||
|
|
||||||
def initialize(@use_polling = true)
|
def initialize(uri_or_path)
|
||||||
|
@client = SigHelper::Client.new(uri_or_path)
|
||||||
|
self.check_update
|
||||||
end
|
end
|
||||||
|
|
||||||
def update_decrypt_function
|
def check_update
|
||||||
@decrypt_function = fetch_decrypt_function
|
# If we have updated in the last 5 minutes, do nothing
|
||||||
|
return if (Time.utc - @last_update) < 5.minutes
|
||||||
|
|
||||||
|
# Get the amount of time elapsed since when the player was updated, in the
|
||||||
|
# event where multiple invidious processes are run in parallel.
|
||||||
|
update_time_elapsed = (@client.get_player_timestamp || 301).seconds
|
||||||
|
|
||||||
|
if update_time_elapsed > 5.minutes
|
||||||
|
LOGGER.debug("Signature: Player might be outdated, updating")
|
||||||
|
@client.force_update
|
||||||
|
@last_update = Time.utc
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
private def fetch_decrypt_function(id = "CvFH_6DNRCY")
|
def decrypt_nsig(n : String) : String?
|
||||||
document = YT_POOL.client &.get("/watch?v=#{id}&gl=US&hl=en").body
|
self.check_update
|
||||||
url = document.match(/src="(?<url>\/s\/player\/[^\/]+\/player_ias[^\/]+\/en_US\/base.js)"/).not_nil!["url"]
|
return @client.decrypt_n_param(n)
|
||||||
player = YT_POOL.client &.get(url).body
|
rescue ex
|
||||||
|
LOGGER.debug(ex.message || "Signature: Unknown error")
|
||||||
function_name = player.match(/^(?<name>[^=]+)=function\(\w\){\w=\w\.split\(""\);[^\. ]+\.[^( ]+/m).not_nil!["name"]
|
LOGGER.trace(ex.inspect_with_backtrace)
|
||||||
function_body = player.match(/^#{Regex.escape(function_name)}=function\(\w\){(?<body>[^}]+)}/m).not_nil!["body"]
|
return nil
|
||||||
function_body = function_body.split(";")[1..-2]
|
|
||||||
|
|
||||||
var_name = function_body[0][0, 2]
|
|
||||||
var_body = player.delete("\n").match(/var #{Regex.escape(var_name)}={(?<body>(.*?))};/).not_nil!["body"]
|
|
||||||
|
|
||||||
operations = {} of String => SigProc
|
|
||||||
var_body.split("},").each do |operation|
|
|
||||||
op_name = operation.match(/^[^:]+/).not_nil![0]
|
|
||||||
op_body = operation.match(/\{[^}]+/).not_nil![0]
|
|
||||||
|
|
||||||
case op_body
|
|
||||||
when "{a.reverse()"
|
|
||||||
operations[op_name] = ->(a : Array(String), _b : Int32) { a.reverse }
|
|
||||||
when "{a.splice(0,b)"
|
|
||||||
operations[op_name] = ->(a : Array(String), b : Int32) { a.delete_at(0..(b - 1)); a }
|
|
||||||
else
|
|
||||||
operations[op_name] = ->(a : Array(String), b : Int32) { c = a[0]; a[0] = a[b % a.size]; a[b % a.size] = c; a }
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
decrypt_function = [] of {SigProc, Int32}
|
|
||||||
function_body.each do |function|
|
|
||||||
function = function.lchop(var_name).delete("[].")
|
|
||||||
|
|
||||||
op_name = function.match(/[^\(]+/).not_nil![0]
|
|
||||||
value = function.match(/\(\w,(?<value>[\d]+)\)/).not_nil!["value"].to_i
|
|
||||||
|
|
||||||
decrypt_function << {operations[op_name], value}
|
|
||||||
end
|
|
||||||
|
|
||||||
return decrypt_function
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def decrypt_signature(fmt : Hash(String, JSON::Any))
|
def decrypt_signature(str : String) : String?
|
||||||
return "" if !fmt["s"]? || !fmt["sp"]?
|
self.check_update
|
||||||
|
return @client.decrypt_sig(str)
|
||||||
|
rescue ex
|
||||||
|
LOGGER.debug(ex.message || "Signature: Unknown error")
|
||||||
|
LOGGER.trace(ex.inspect_with_backtrace)
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
sp = fmt["sp"].as_s
|
def get_sts : UInt64?
|
||||||
sig = fmt["s"].as_s.split("")
|
self.check_update
|
||||||
if !@use_polling
|
return @client.get_signature_timestamp
|
||||||
now = Time.monotonic
|
rescue ex
|
||||||
if now - @decrypt_time > 60.seconds || @decrypt_function.size == 0
|
LOGGER.debug(ex.message || "Signature: Unknown error")
|
||||||
@decrypt_function = fetch_decrypt_function
|
LOGGER.trace(ex.inspect_with_backtrace)
|
||||||
@decrypt_time = Time.monotonic
|
return nil
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
@decrypt_function.each do |proc, value|
|
|
||||||
sig = proc.call(sig, value)
|
|
||||||
end
|
|
||||||
|
|
||||||
return "&#{sp}=#{sig.join("")}"
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -52,9 +52,9 @@ def recode_length_seconds(time)
|
||||||
end
|
end
|
||||||
|
|
||||||
def decode_interval(string : String) : Time::Span
|
def decode_interval(string : String) : Time::Span
|
||||||
rawMinutes = string.try &.to_i32?
|
raw_minutes = string.try &.to_i32?
|
||||||
|
|
||||||
if !rawMinutes
|
if !raw_minutes
|
||||||
hours = /(?<hours>\d+)h/.match(string).try &.["hours"].try &.to_i32
|
hours = /(?<hours>\d+)h/.match(string).try &.["hours"].try &.to_i32
|
||||||
hours ||= 0
|
hours ||= 0
|
||||||
|
|
||||||
|
@ -63,7 +63,7 @@ def decode_interval(string : String) : Time::Span
|
||||||
|
|
||||||
time = Time::Span.new(hours: hours, minutes: minutes)
|
time = Time::Span.new(hours: hours, minutes: minutes)
|
||||||
else
|
else
|
||||||
time = Time::Span.new(minutes: rawMinutes)
|
time = Time::Span.new(minutes: raw_minutes)
|
||||||
end
|
end
|
||||||
|
|
||||||
return time
|
return time
|
||||||
|
@ -323,68 +323,6 @@ def parse_range(range)
|
||||||
return 0_i64, nil
|
return 0_i64, nil
|
||||||
end
|
end
|
||||||
|
|
||||||
def fetch_random_instance
|
|
||||||
begin
|
|
||||||
instance_api_client = make_client(URI.parse("https://api.invidious.io"))
|
|
||||||
|
|
||||||
# Timeouts
|
|
||||||
instance_api_client.connect_timeout = 10.seconds
|
|
||||||
instance_api_client.dns_timeout = 10.seconds
|
|
||||||
|
|
||||||
instance_list = JSON.parse(instance_api_client.get("/instances.json").body).as_a
|
|
||||||
instance_api_client.close
|
|
||||||
rescue Socket::ConnectError | IO::TimeoutError | JSON::ParseException
|
|
||||||
instance_list = [] of JSON::Any
|
|
||||||
end
|
|
||||||
|
|
||||||
filtered_instance_list = [] of String
|
|
||||||
|
|
||||||
instance_list.each do |data|
|
|
||||||
# TODO Check if current URL is onion instance and use .onion types if so.
|
|
||||||
if data[1]["type"] == "https"
|
|
||||||
# Instances can have statistics disabled, which is an requirement of version validation.
|
|
||||||
# as_nil? doesn't exist. Thus we'll have to handle the error raised if as_nil fails.
|
|
||||||
begin
|
|
||||||
data[1]["stats"].as_nil
|
|
||||||
next
|
|
||||||
rescue TypeCastError
|
|
||||||
end
|
|
||||||
|
|
||||||
# stats endpoint could also lack the software dict.
|
|
||||||
next if data[1]["stats"]["software"]?.nil?
|
|
||||||
|
|
||||||
# Makes sure the instance isn't too outdated.
|
|
||||||
if remote_version = data[1]["stats"]?.try &.["software"]?.try &.["version"]
|
|
||||||
remote_commit_date = remote_version.as_s.match(/\d{4}\.\d{2}\.\d{2}/)
|
|
||||||
next if !remote_commit_date
|
|
||||||
|
|
||||||
remote_commit_date = Time.parse(remote_commit_date[0], "%Y.%m.%d", Time::Location::UTC)
|
|
||||||
local_commit_date = Time.parse(CURRENT_VERSION, "%Y.%m.%d", Time::Location::UTC)
|
|
||||||
|
|
||||||
next if (remote_commit_date - local_commit_date).abs.days > 30
|
|
||||||
|
|
||||||
begin
|
|
||||||
data[1]["monitor"].as_nil
|
|
||||||
health = data[1]["monitor"].as_h["dailyRatios"][0].as_h["ratio"]
|
|
||||||
filtered_instance_list << data[0].as_s if health.to_s.to_f > 90
|
|
||||||
rescue TypeCastError
|
|
||||||
# We can't check the health if the monitoring is broken. Thus we'll just add it to the list
|
|
||||||
# and move on. Ideally we'll ignore any instance that has broken health monitoring but due to the fact that
|
|
||||||
# it's an error that often occurs with all the instances at the same time, we have to just skip the check.
|
|
||||||
filtered_instance_list << data[0].as_s
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
# If for some reason no instances managed to get fetched successfully then we'll just redirect to redirect.invidious.io
|
|
||||||
if filtered_instance_list.size == 0
|
|
||||||
return "redirect.invidious.io"
|
|
||||||
end
|
|
||||||
|
|
||||||
return filtered_instance_list.sample(1)[0]
|
|
||||||
end
|
|
||||||
|
|
||||||
def reduce_uri(uri : URI | String, max_length : Int32 = 50, suffix : String = "…") : String
|
def reduce_uri(uri : URI | String, max_length : Int32 = 50, suffix : String = "…") : String
|
||||||
str = uri.to_s.sub(/^https?:\/\//, "")
|
str = uri.to_s.sub(/^https?:\/\//, "")
|
||||||
if str.size > max_length
|
if str.size > max_length
|
||||||
|
@ -445,3 +383,17 @@ def parse_link_endpoint(endpoint : JSON::Any, text : String, video_id : String)
|
||||||
end
|
end
|
||||||
return text
|
return text
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Generates a list of external videoplayback proxies for
|
||||||
|
# CSP
|
||||||
|
def gen_videoplayback_proxy_list
|
||||||
|
if !CONFIG.external_videoplayback_proxy.empty?
|
||||||
|
external_videoplayback_proxy = ""
|
||||||
|
CONFIG.external_videoplayback_proxy.each do |proxy|
|
||||||
|
external_videoplayback_proxy += " #{proxy}"
|
||||||
|
end
|
||||||
|
else
|
||||||
|
external_videoplayback_proxy = ""
|
||||||
|
end
|
||||||
|
return external_videoplayback_proxy
|
||||||
|
end
|
||||||
|
|
|
@ -4,6 +4,27 @@ module Invidious::HttpServer
|
||||||
module Utils
|
module Utils
|
||||||
extend self
|
extend self
|
||||||
|
|
||||||
|
@@proxy_alive : String = ""
|
||||||
|
|
||||||
|
def check_external_proxy
|
||||||
|
CONFIG.external_videoplayback_proxy.each do |proxy|
|
||||||
|
begin
|
||||||
|
response = HTTP::Client.get(proxy)
|
||||||
|
if response.status_code == 200
|
||||||
|
@@proxy_alive = proxy
|
||||||
|
LOGGER.debug("CheckExternalProxy: Proxy set to: '#{proxy}'")
|
||||||
|
break
|
||||||
|
end
|
||||||
|
rescue
|
||||||
|
LOGGER.debug("CheckExternalProxy: Proxy '#{proxy}' is not available")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def get_external_proxy
|
||||||
|
return @@proxy_alive
|
||||||
|
end
|
||||||
|
|
||||||
def proxy_video_url(raw_url : String, *, region : String? = nil, absolute : Bool = false)
|
def proxy_video_url(raw_url : String, *, region : String? = nil, absolute : Bool = false)
|
||||||
url = URI.parse(raw_url)
|
url = URI.parse(raw_url)
|
||||||
|
|
||||||
|
@ -11,11 +32,16 @@ module Invidious::HttpServer
|
||||||
params = url.query_params
|
params = url.query_params
|
||||||
params["host"] = url.host.not_nil! # Should never be nil, in theory
|
params["host"] = url.host.not_nil! # Should never be nil, in theory
|
||||||
params["region"] = region if !region.nil?
|
params["region"] = region if !region.nil?
|
||||||
|
url.query_params = params
|
||||||
|
|
||||||
if absolute
|
if absolute
|
||||||
return "#{HOST_URL}#{url.request_target}?#{params}"
|
if !@@proxy_alive.empty?
|
||||||
|
return "#{@@proxy_alive}#{url.request_target}"
|
||||||
|
else
|
||||||
|
return "#{HOST_URL}#{url.request_target}"
|
||||||
|
end
|
||||||
else
|
else
|
||||||
return "#{url.request_target}?#{params}"
|
return url.request_target
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue