Compare commits
53 Commits
3fdbf13d95
...
feature/is
| Author | SHA1 | Date | |
|---|---|---|---|
| 9b030b24f4 | |||
| 44b651cace | |||
| eb2edc3710 | |||
| f63497090b | |||
| 215142dee4 | |||
| e391f64cc5 | |||
| 4216d153fd | |||
| 9241ecfee8 | |||
| 4bc7a6d980 | |||
| b50833f19f | |||
| f5cccc30dc | |||
| 8a8f02c5bd | |||
| 292d71edc5 | |||
| c9fb8fda93 | |||
| 1fae03e70b | |||
| 6ed94db8cf | |||
| 02a7c29dba | |||
| cf8e2955c2 | |||
| 7aa06b7f83 | |||
| f7a690405b | |||
| 93ea11cd0e | |||
| 263281df3c | |||
|
|
b953933e6f | ||
| 7db84492b5 | |||
| c7cd5e0601 | |||
| c05f8fad35 | |||
| b9c7535812 | |||
| cdcd3bce1f | |||
| 6eb02cc95f | |||
| 2454d26d6b | |||
| 8054a4a983 | |||
| afefce0a11 | |||
| 334c82d098 | |||
| ddd536c958 | |||
| ae5ba0d044 | |||
| 83133218a4 | |||
| d412d983bd | |||
| 9859229101 | |||
| dc8f9d91e7 | |||
| f3a86b8070 | |||
| b814434c48 | |||
| ba05ecdd94 | |||
| 1751158cbd | |||
| c45cebe497 | |||
| dc61a7ee92 | |||
| d70a0a8630 | |||
| 4f37283e68 | |||
| 0e948e7f55 | |||
| 79c95a9ddb | |||
| 0ca0fb90d6 | |||
| 603a7f5942 | |||
| 63f8541a82 | |||
| dc5af8da28 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -174,3 +174,4 @@ cython_debug/
|
|||||||
.pypirc
|
.pypirc
|
||||||
|
|
||||||
dist
|
dist
|
||||||
|
context.json
|
||||||
@@ -4,6 +4,8 @@ dependencies = [
|
|||||||
"watchdog~=6.0.0",
|
"watchdog~=6.0.0",
|
||||||
"markdown~=3.3.6",
|
"markdown~=3.3.6",
|
||||||
"bs4~=0.0.2",
|
"bs4~=0.0.2",
|
||||||
|
"toml~0.10.2",
|
||||||
|
"jinja2~=3.1.6",
|
||||||
]
|
]
|
||||||
dynamic = []
|
dynamic = []
|
||||||
authors = []
|
authors = []
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Navê Gotarê
|
# Navê Gotarê example
|
||||||
Ev gotar, mînakek placeholder ya bi zimanê Kurdî ye. Nivîs, weşan û rûpelên nûçe di vê belgeyê de têne pêşniyar kirin. Ev metn ji bo testkirina layout an jî demo-yê hatiye çêkirin û çend bingehên cihanî yên naveroka gotarê nîşan dide.
|
Ev gotar, mînakek placeholder ya bi zimanê Kurdî ye. Nivîs, weşan û rûpelên nûçe di vê belgeyê de têne pêşniyar kirin. Ev metn ji bo testkirina layout an jî demo-yê hatiye çêkirin û çend bingehên cihanî yên naveroka gotarê nîşan dide.
|
||||||
|
|
||||||
## Destpêk
|
## Destpêk
|
||||||
2
src/articles/index.toml
Normal file
2
src/articles/index.toml
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
name="index_page"
|
||||||
|
datetime="2024-04-15 13:45:12.123456"
|
||||||
@@ -1,4 +1,7 @@
|
|||||||
# Navê Gotarê
|
# Navê Gotarê
|
||||||
|
|
||||||
|
[{article_title:example}]({article_url:example})
|
||||||
|
|
||||||
Ev gotar, mînakek placeholder ya bi zimanê Kurdî ye. Nivîs, weşan û rûpelên nûçe di vê belgeyê de têne pêşniyar kirin. Ev metn ji bo testkirina layout an jî demo-yê hatiye çêkirin û çend bingehên cihanî yên naveroka gotarê nîşan dide.
|
Ev gotar, mînakek placeholder ya bi zimanê Kurdî ye. Nivîs, weşan û rûpelên nûçe di vê belgeyê de têne pêşniyar kirin. Ev metn ji bo testkirina layout an jî demo-yê hatiye çêkirin û çend bingehên cihanî yên naveroka gotarê nîşan dide.
|
||||||
|
|
||||||
## Destpêk
|
## Destpêk
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="{article_language_code}">
|
<html>
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<title>{article_language_flag} {article_title}</title>
|
<title>{{slug}}</title>
|
||||||
<link rel="stylesheet" href="/static/bulma.min.css" />
|
<link rel="stylesheet" href="/static/bulma.min.css" />
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
@@ -14,17 +14,59 @@
|
|||||||
aria-label="main navigation"
|
aria-label="main navigation"
|
||||||
>
|
>
|
||||||
<div class="navbar-brand">
|
<div class="navbar-brand">
|
||||||
<a class="navbar-item" href="{article_overview_url}">
|
<a class="navbar-item" href="#">
|
||||||
<strong>{article_language_flag} {article_title}</strong>
|
<strong>Static Translated Site Generator</strong>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
|
|
||||||
<!-- Main Content -->
|
|
||||||
<section class="section">
|
<section class="section">
|
||||||
<div class="content">
|
{% if translations|length %}
|
||||||
{article_content}
|
<div class="container content">
|
||||||
|
<div class="column is-half is-offset-one-quarter">
|
||||||
|
<h1>Translations</h1>
|
||||||
|
</div>
|
||||||
|
<div class="column is-half is-offset-one-quarter">
|
||||||
|
{% for t in translations %}
|
||||||
|
<div class="card mb-4" lang="{{t.language.code}}">
|
||||||
|
<a href="{{t.url}}" hreflang="{{t.language.code}}" class="card mb-4" style="color: inherit; text-decoration: none;">
|
||||||
|
<div class="card-content">
|
||||||
|
<p class="title">{{t.language.flag}} {{t.title}} </p>
|
||||||
|
<p class="content">
|
||||||
|
{{t.preview}}
|
||||||
|
<br />
|
||||||
|
<time datetime="{{iso_date}}">{{date}}</time>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
|
||||||
|
{% if children|length %}
|
||||||
|
<div class="container content">
|
||||||
|
<div class="column is-half is-offset-one-quarter">
|
||||||
|
<h1>Child Articles</h1>
|
||||||
|
</div>
|
||||||
|
<div class="column is-half is-offset-one-quarter">
|
||||||
|
{% for c in children %}
|
||||||
|
<div class="card mb-4" >
|
||||||
|
<a href="{{c.url}}" class="card mb-4" style="color: inherit; text-decoration: none;">
|
||||||
|
<div class="card-content">
|
||||||
|
<p class="title">{{c.slug}} </p>
|
||||||
|
<p class="content">
|
||||||
|
<time datetime="{{iso_date}}">{{date}}</time>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<!-- Footer -->
|
<!-- Footer -->
|
||||||
@@ -34,4 +76,23 @@
|
|||||||
</div>
|
</div>
|
||||||
</footer>
|
</footer>
|
||||||
</body>
|
</body>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
const userLang = navigator.language || navigator.userLanguage;
|
||||||
|
|
||||||
|
// Normalize and check if the language is not English or German
|
||||||
|
if (!["en", "de", "de-DE"].includes(userLang)) {
|
||||||
|
// Try to find a matching card by language attribute
|
||||||
|
const cardToMove =
|
||||||
|
document.querySelector(`.card[lang^="${userLang.replace("_", "-").toLowerCase()}"]`) ||
|
||||||
|
document.querySelector(`.card[lang^="${userLang.split("-")[0]}"]`);
|
||||||
|
|
||||||
|
if (cardToMove) {
|
||||||
|
const container = cardToMove.parentNode;
|
||||||
|
container.insertBefore(cardToMove, container.firstChild);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
38
src/templates/article_translation.html
Normal file
38
src/templates/article_translation.html
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="{article_language_code}">
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
<title>{{language.flag}} {{title}}</title>
|
||||||
|
<link rel="stylesheet" href="/static/bulma.min.css" />
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<!-- Header (Navbar) -->
|
||||||
|
<nav
|
||||||
|
class="navbar is-primary"
|
||||||
|
role="navigation"
|
||||||
|
aria-label="main navigation"
|
||||||
|
>
|
||||||
|
<div class="navbar-brand">
|
||||||
|
<a class="navbar-item" href="{{article_url}}">
|
||||||
|
<strong>{{language.flag}} {{title}}</strong>
|
||||||
|
<time datetime="{{meta.iso_date}}">{{meta.date}}</time>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<!-- Main Content -->
|
||||||
|
<section class="section">
|
||||||
|
<div class="content">
|
||||||
|
{{content}}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<!-- Footer -->
|
||||||
|
<footer class="footer">
|
||||||
|
<div class="content has-text-centered">
|
||||||
|
<p><strong>STSG</strong> by Hazel. © 2025</p>
|
||||||
|
</div>
|
||||||
|
</footer>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
||||||
<title>STSG</title>
|
|
||||||
<link rel="stylesheet" href="/static/bulma.min.css" />
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<!-- Header (Navbar) -->
|
|
||||||
<nav
|
|
||||||
class="navbar is-primary"
|
|
||||||
role="navigation"
|
|
||||||
aria-label="main navigation"
|
|
||||||
>
|
|
||||||
<div class="navbar-brand">
|
|
||||||
<a class="navbar-item" href="#">
|
|
||||||
<strong>Static Translated Site Generator</strong>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
|
|
||||||
<section class="section">
|
|
||||||
<div class="container">
|
|
||||||
<div class="column is-half is-offset-one-quarter">{overview_cards}</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<!-- Footer -->
|
|
||||||
<footer class="footer">
|
|
||||||
<div class="content has-text-centered">
|
|
||||||
<p><strong>STSG</strong> by Hazel. © 2025</p>
|
|
||||||
</div>
|
|
||||||
</footer>
|
|
||||||
</body>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
document.addEventListener("DOMContentLoaded", function () {
|
|
||||||
const userLang = navigator.language || navigator.userLanguage;
|
|
||||||
|
|
||||||
// Normalize and check if the language is not English or German
|
|
||||||
if (!["en", "de", "de-DE"].includes(userLang)) {
|
|
||||||
// Try to find a matching card by language attribute
|
|
||||||
const cardToMove =
|
|
||||||
document.querySelector(`.card[lang^="${userLang.replace("_", "-").toLowerCase()}"]`) ||
|
|
||||||
document.querySelector(`.card[lang^="${userLang.split("-")[0]}"]`);
|
|
||||||
|
|
||||||
if (cardToMove) {
|
|
||||||
const container = cardToMove.parentNode;
|
|
||||||
container.insertBefore(cardToMove, container.firstChild);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
</html>
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
<div class="card mb-4" lang="{article_language_code}">
|
|
||||||
<a href="{article_href}" hreflang="{article_language_code}" class="card mb-4" style="color: inherit; text-decoration: none;">
|
|
||||||
<div class="card-content" href="{article_href}">
|
|
||||||
<p class="title">{article_language_flag} {article_title}</p>
|
|
||||||
<p class="content">{article_preview}</p>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
25
stsg.toml
Normal file
25
stsg.toml
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
[setup]
|
||||||
|
source_directory = "src"
|
||||||
|
dist_directory = "dist"
|
||||||
|
|
||||||
|
[formatting]
|
||||||
|
article_preview_length = 400
|
||||||
|
datetime_format = "%d. %B %Y"
|
||||||
|
default_language = "de"
|
||||||
|
|
||||||
|
[languages]
|
||||||
|
[languages.de]
|
||||||
|
native_name = "Schland"
|
||||||
|
priority = 100
|
||||||
|
|
||||||
|
[languages.en]
|
||||||
|
priority = 90
|
||||||
|
|
||||||
|
[languages.ar_sy]
|
||||||
|
priority = 50
|
||||||
|
|
||||||
|
[languages.ku]
|
||||||
|
priority = 49
|
||||||
|
|
||||||
|
[languages.tr]
|
||||||
|
priority = 60
|
||||||
793
stsg/__init__.py
793
stsg/__init__.py
@@ -0,0 +1,793 @@
|
|||||||
|
class config:
|
||||||
|
class setup:
|
||||||
|
source_directory = "src"
|
||||||
|
dist_directory = "dist"
|
||||||
|
|
||||||
|
class formatting:
|
||||||
|
article_preview_length = 200
|
||||||
|
datetime_format = "%d. %B %Y"
|
||||||
|
fallback_language = "en"
|
||||||
|
|
||||||
|
languages = {
|
||||||
|
"af": {
|
||||||
|
"flag": "🇿🇦",
|
||||||
|
"name": "Afrikaans",
|
||||||
|
"native_name": "Afrikaans"
|
||||||
|
},
|
||||||
|
"am": {
|
||||||
|
"flag": "🇪🇹",
|
||||||
|
"name": "Amharic",
|
||||||
|
"native_name": "አማርኛ"
|
||||||
|
},
|
||||||
|
"an": {
|
||||||
|
"flag": "🇪🇸",
|
||||||
|
"name": "Aragonese",
|
||||||
|
"native_name": "aragonés"
|
||||||
|
},
|
||||||
|
"ar": {
|
||||||
|
"flag": "🇸🇦",
|
||||||
|
"name": "Arabic",
|
||||||
|
"native_name": "العربية"
|
||||||
|
},
|
||||||
|
"ar_ae": {
|
||||||
|
"flag": "🇦🇪",
|
||||||
|
"name": "Arabic (UAE)",
|
||||||
|
"native_name": "العربية (الإمارات)"
|
||||||
|
},
|
||||||
|
"ar_bh": {
|
||||||
|
"flag": "🇧🇭",
|
||||||
|
"name": "Arabic (Bahrain)",
|
||||||
|
"native_name": "العربية (البحرين)"
|
||||||
|
},
|
||||||
|
"ar_dz": {
|
||||||
|
"flag": "🇩🇿",
|
||||||
|
"name": "Arabic (Algeria)",
|
||||||
|
"native_name": "العربية (الجزائر)"
|
||||||
|
},
|
||||||
|
"ar_eg": {
|
||||||
|
"flag": "🇪🇬",
|
||||||
|
"name": "Arabic (Egypt)",
|
||||||
|
"native_name": "العربية (مصر)"
|
||||||
|
},
|
||||||
|
"ar_iq": {
|
||||||
|
"flag": "🇮🇶",
|
||||||
|
"name": "Arabic (Iraq)",
|
||||||
|
"native_name": "العربية (العراق)"
|
||||||
|
},
|
||||||
|
"ar_jo": {
|
||||||
|
"flag": "🇯🇴",
|
||||||
|
"name": "Arabic (Jordan)",
|
||||||
|
"native_name": "العربية (الأردن)"
|
||||||
|
},
|
||||||
|
"ar_kw": {
|
||||||
|
"flag": "🇰🇼",
|
||||||
|
"name": "Arabic (Kuwait)",
|
||||||
|
"native_name": "العربية (الكويت)"
|
||||||
|
},
|
||||||
|
"ar_lb": {
|
||||||
|
"flag": "🇱🇧",
|
||||||
|
"name": "Arabic (Lebanon)",
|
||||||
|
"native_name": "العربية (لبنان)"
|
||||||
|
},
|
||||||
|
"ar_ly": {
|
||||||
|
"flag": "🇱🇾",
|
||||||
|
"name": "Arabic (Libya)",
|
||||||
|
"native_name": "العربية (ليبيا)"
|
||||||
|
},
|
||||||
|
"ar_ma": {
|
||||||
|
"flag": "🇲🇦",
|
||||||
|
"name": "Arabic (Morocco)",
|
||||||
|
"native_name": "العربية (المغرب)"
|
||||||
|
},
|
||||||
|
"ar_om": {
|
||||||
|
"flag": "🇴🇲",
|
||||||
|
"name": "Arabic (Oman)",
|
||||||
|
"native_name": "العربية (عُمان)"
|
||||||
|
},
|
||||||
|
"ar_qa": {
|
||||||
|
"flag": "🇶🇦",
|
||||||
|
"name": "Arabic (Qatar)",
|
||||||
|
"native_name": "العربية (قطر)"
|
||||||
|
},
|
||||||
|
"ar_sa": {
|
||||||
|
"flag": "🇸🇦",
|
||||||
|
"name": "Arabic (Saudi Arabia)",
|
||||||
|
"native_name": "العربية (السعودية)"
|
||||||
|
},
|
||||||
|
"ar_sd": {
|
||||||
|
"flag": "🇸🇩",
|
||||||
|
"name": "Arabic (Sudan)",
|
||||||
|
"native_name": "العربية (السودان)"
|
||||||
|
},
|
||||||
|
"ar_sy": {
|
||||||
|
"flag": "🇸🇾",
|
||||||
|
"name": "Arabic (Syria)",
|
||||||
|
"native_name": "العربية (سوريا)",
|
||||||
|
},
|
||||||
|
"ar_tn": {
|
||||||
|
"flag": "🇹🇳",
|
||||||
|
"name": "Arabic (Tunisia)",
|
||||||
|
"native_name": "العربية (تونس)"
|
||||||
|
},
|
||||||
|
"ar_ye": {
|
||||||
|
"flag": "🇾🇪",
|
||||||
|
"name": "Arabic (Yemen)",
|
||||||
|
"native_name": "العربية (اليمن)"
|
||||||
|
},
|
||||||
|
"ars_ae": {
|
||||||
|
"flag": "🇦🇪",
|
||||||
|
"name": "Najdi Arabic (UAE)",
|
||||||
|
"native_name": "نَجْدِيّ"
|
||||||
|
},
|
||||||
|
"ars_arab_sa": {
|
||||||
|
"flag": "🇸🇦",
|
||||||
|
"name": "Najdi Arabic (Saudi Arabia, Arabic Script)",
|
||||||
|
"native_name": "نَجْدِيّ"
|
||||||
|
},
|
||||||
|
"ars_sa": {
|
||||||
|
"flag": "🇸🇦",
|
||||||
|
"name": "Najdi Arabic (Saudi Arabia)",
|
||||||
|
"native_name": "نَجْدِيّ"
|
||||||
|
},
|
||||||
|
"as": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "Assamese",
|
||||||
|
"native_name": "অসমীয়া"
|
||||||
|
},
|
||||||
|
"az": {
|
||||||
|
"flag": "🇦🇿",
|
||||||
|
"name": "Azerbaijani",
|
||||||
|
"native_name": "Azərbaycan"
|
||||||
|
},
|
||||||
|
"be": {
|
||||||
|
"flag": "🇧🇾",
|
||||||
|
"name": "Belarusian",
|
||||||
|
"native_name": "Беларуская"
|
||||||
|
},
|
||||||
|
"bg": {
|
||||||
|
"flag": "🇧🇬",
|
||||||
|
"name": "Bulgarian",
|
||||||
|
"native_name": "Български"
|
||||||
|
},
|
||||||
|
"bm": {
|
||||||
|
"flag": "🇲🇱",
|
||||||
|
"name": "Bambara",
|
||||||
|
"native_name": "bamanankan"
|
||||||
|
},
|
||||||
|
"bn": {
|
||||||
|
"flag": "🇧🇩",
|
||||||
|
"name": "Bengali",
|
||||||
|
"native_name": "বাংলা"
|
||||||
|
},
|
||||||
|
"bn_in": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "Bengali (India)",
|
||||||
|
"native_name": "বাংলা (ভারত)"
|
||||||
|
},
|
||||||
|
"br": {
|
||||||
|
"flag": "🏴",
|
||||||
|
"name": "Breton",
|
||||||
|
"native_name": "brezhoneg"
|
||||||
|
},
|
||||||
|
"bs": {
|
||||||
|
"flag": "🇧🇦",
|
||||||
|
"name": "Bosnian",
|
||||||
|
"native_name": "Bosanski"
|
||||||
|
},
|
||||||
|
"ca": {
|
||||||
|
"flag": "🇪🇸",
|
||||||
|
"name": "Catalan",
|
||||||
|
"native_name": "Català"
|
||||||
|
},
|
||||||
|
"crh": {
|
||||||
|
"flag": "🇺🇦",
|
||||||
|
"name": "Crimean Tatar",
|
||||||
|
"native_name": "qırımtatarca"
|
||||||
|
},
|
||||||
|
"cs": {
|
||||||
|
"flag": "🇨🇿",
|
||||||
|
"name": "Czech",
|
||||||
|
"native_name": "Čeština"
|
||||||
|
},
|
||||||
|
"cv": {
|
||||||
|
"flag": "🇷🇺",
|
||||||
|
"name": "Chuvash",
|
||||||
|
"native_name": "чӑваш чӗлхи"
|
||||||
|
},
|
||||||
|
"cy": {
|
||||||
|
"flag": "🏴",
|
||||||
|
"name": "Welsh",
|
||||||
|
"native_name": "Cymraeg"
|
||||||
|
},
|
||||||
|
"da": {
|
||||||
|
"flag": "🇩🇰",
|
||||||
|
"name": "Danish",
|
||||||
|
"native_name": "Dansk"
|
||||||
|
},
|
||||||
|
"de": {
|
||||||
|
"flag": "🇩🇪",
|
||||||
|
"name": "German",
|
||||||
|
"native_name": "Deutsch",
|
||||||
|
},
|
||||||
|
"de_at": {
|
||||||
|
"flag": "🇦🇹",
|
||||||
|
"name": "German (Austria)",
|
||||||
|
"native_name": "Deutsch (Österreich)"
|
||||||
|
},
|
||||||
|
"de_be": {
|
||||||
|
"flag": "🇧🇪",
|
||||||
|
"name": "German (Belgium)",
|
||||||
|
"native_name": "Deutsch (Belgien)"
|
||||||
|
},
|
||||||
|
"de_ch": {
|
||||||
|
"flag": "🇨🇭",
|
||||||
|
"name": "German (Switzerland)",
|
||||||
|
"native_name": "Deutsch (Schweiz)"
|
||||||
|
},
|
||||||
|
"dv": {
|
||||||
|
"flag": "🇲🇻",
|
||||||
|
"name": "Dhivehi",
|
||||||
|
"native_name": "ދިވެހި"
|
||||||
|
},
|
||||||
|
"dz": {
|
||||||
|
"flag": "🇧🇹",
|
||||||
|
"name": "Dzongkha",
|
||||||
|
"native_name": "རྫོང་ཁ"
|
||||||
|
},
|
||||||
|
"el": {
|
||||||
|
"flag": "🇬🇷",
|
||||||
|
"name": "Greek",
|
||||||
|
"native_name": "Ελληνικά"
|
||||||
|
},
|
||||||
|
"en": {
|
||||||
|
"flag": "🇺🇸",
|
||||||
|
"name": "English",
|
||||||
|
"native_name": "English",
|
||||||
|
},
|
||||||
|
"en_au": {
|
||||||
|
"flag": "🇦🇺",
|
||||||
|
"name": "English (Australia)",
|
||||||
|
"native_name": "English (Australia)"
|
||||||
|
},
|
||||||
|
"en_ca": {
|
||||||
|
"flag": "🇨🇦",
|
||||||
|
"name": "English (Canada)",
|
||||||
|
"native_name": "English (Canada)"
|
||||||
|
},
|
||||||
|
"en_gb": {
|
||||||
|
"flag": "🇬🇧",
|
||||||
|
"name": "English (UK)",
|
||||||
|
"native_name": "English (UK)",
|
||||||
|
},
|
||||||
|
"en_ie": {
|
||||||
|
"flag": "🇮🇪",
|
||||||
|
"name": "English (Ireland)",
|
||||||
|
"native_name": "English (Ireland)"
|
||||||
|
},
|
||||||
|
"en_in": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "English (India)",
|
||||||
|
"native_name": "English (India)"
|
||||||
|
},
|
||||||
|
"en_nz": {
|
||||||
|
"flag": "🇳🇿",
|
||||||
|
"name": "English (New Zealand)",
|
||||||
|
"native_name": "English (New Zealand)"
|
||||||
|
},
|
||||||
|
"en_us": {
|
||||||
|
"flag": "🇺🇸",
|
||||||
|
"name": "English (US)",
|
||||||
|
"native_name": "English (US)",
|
||||||
|
},
|
||||||
|
"es": {
|
||||||
|
"flag": "🇪🇸",
|
||||||
|
"name": "Spanish",
|
||||||
|
"native_name": "Español"
|
||||||
|
},
|
||||||
|
"es_ar": {
|
||||||
|
"flag": "🇦🇷",
|
||||||
|
"name": "Spanish (Argentina)",
|
||||||
|
"native_name": "Español (Argentina)"
|
||||||
|
},
|
||||||
|
"es_mx": {
|
||||||
|
"flag": "🇲🇽",
|
||||||
|
"name": "Spanish (Mexico)",
|
||||||
|
"native_name": "Español (México)"
|
||||||
|
},
|
||||||
|
"et": {
|
||||||
|
"flag": "🇪🇪",
|
||||||
|
"name": "Estonian",
|
||||||
|
"native_name": "Eesti"
|
||||||
|
},
|
||||||
|
"fa": {
|
||||||
|
"flag": "🇮🇷",
|
||||||
|
"name": "Persian",
|
||||||
|
"native_name": "فارسی"
|
||||||
|
},
|
||||||
|
"ff": {
|
||||||
|
"flag": "🌍",
|
||||||
|
"name": "Fula",
|
||||||
|
"native_name": "Fulfulde"
|
||||||
|
},
|
||||||
|
"fi": {
|
||||||
|
"flag": "🇫🇮",
|
||||||
|
"name": "Finnish",
|
||||||
|
"native_name": "Suomi"
|
||||||
|
},
|
||||||
|
"fo": {
|
||||||
|
"flag": "🇫🇴",
|
||||||
|
"name": "Faroese",
|
||||||
|
"native_name": "føroyskt"
|
||||||
|
},
|
||||||
|
"fr": {
|
||||||
|
"flag": "🇫🇷",
|
||||||
|
"name": "French",
|
||||||
|
"native_name": "Français"
|
||||||
|
},
|
||||||
|
"fr_ca": {
|
||||||
|
"flag": "🇨🇦",
|
||||||
|
"name": "French (Canada)",
|
||||||
|
"native_name": "Français (Canada)"
|
||||||
|
},
|
||||||
|
"fr_ch": {
|
||||||
|
"flag": "🇨🇭",
|
||||||
|
"name": "French (Switzerland)",
|
||||||
|
"native_name": "Français (Suisse)"
|
||||||
|
},
|
||||||
|
"ga": {
|
||||||
|
"flag": "🇮🇪",
|
||||||
|
"name": "Irish",
|
||||||
|
"native_name": "Gaeilge"
|
||||||
|
},
|
||||||
|
"gl": {
|
||||||
|
"flag": "🇪🇸",
|
||||||
|
"name": "Galician",
|
||||||
|
"native_name": "Galego"
|
||||||
|
},
|
||||||
|
"gn": {
|
||||||
|
"flag": "🇵🇾",
|
||||||
|
"name": "Guarani",
|
||||||
|
"native_name": "Avañe'ẽ"
|
||||||
|
},
|
||||||
|
"gu": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "Gujarati",
|
||||||
|
"native_name": "ગુજરાતી"
|
||||||
|
},
|
||||||
|
"ha": {
|
||||||
|
"flag": "🇳🇬",
|
||||||
|
"name": "Hausa",
|
||||||
|
"native_name": "هَوُسَ"
|
||||||
|
},
|
||||||
|
"he": {
|
||||||
|
"flag": "🇮🇱",
|
||||||
|
"name": "Hebrew",
|
||||||
|
"native_name": "עברית"
|
||||||
|
},
|
||||||
|
"hi": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "Hindi",
|
||||||
|
"native_name": "हिन्दी"
|
||||||
|
},
|
||||||
|
"hr": {
|
||||||
|
"flag": "🇭🇷",
|
||||||
|
"name": "Croatian",
|
||||||
|
"native_name": "Hrvatski"
|
||||||
|
},
|
||||||
|
"ht": {
|
||||||
|
"flag": "🇭🇹",
|
||||||
|
"name": "Haitian Creole",
|
||||||
|
"native_name": "Kreyòl ayisyen"
|
||||||
|
},
|
||||||
|
"hu": {
|
||||||
|
"flag": "🇭🇺",
|
||||||
|
"name": "Hungarian",
|
||||||
|
"native_name": "Magyar"
|
||||||
|
},
|
||||||
|
"id": {
|
||||||
|
"flag": "🇮🇩",
|
||||||
|
"name": "Indonesian",
|
||||||
|
"native_name": "Bahasa Indonesia"
|
||||||
|
},
|
||||||
|
"io": {
|
||||||
|
"flag": "🌍",
|
||||||
|
"name": "Ido",
|
||||||
|
"native_name": "Ido"
|
||||||
|
},
|
||||||
|
"is": {
|
||||||
|
"flag": "🇮🇸",
|
||||||
|
"name": "Icelandic",
|
||||||
|
"native_name": "Íslenska"
|
||||||
|
},
|
||||||
|
"it": {
|
||||||
|
"flag": "🇮🇹",
|
||||||
|
"name": "Italian",
|
||||||
|
"native_name": "Italiano"
|
||||||
|
},
|
||||||
|
"ja": {
|
||||||
|
"flag": "🇯🇵",
|
||||||
|
"name": "Japanese",
|
||||||
|
"native_name": "日本語"
|
||||||
|
},
|
||||||
|
"jv_id": {
|
||||||
|
"flag": "🇮🇩",
|
||||||
|
"name": "Javanese (Indonesia)",
|
||||||
|
"native_name": "basa jawa"
|
||||||
|
},
|
||||||
|
"ka": {
|
||||||
|
"flag": "🇬🇪",
|
||||||
|
"name": "Georgian",
|
||||||
|
"native_name": "ქართული"
|
||||||
|
},
|
||||||
|
"kg": {
|
||||||
|
"flag": "🇨🇬",
|
||||||
|
"name": "Kongo",
|
||||||
|
"native_name": "KiKongo"
|
||||||
|
},
|
||||||
|
"kj": {
|
||||||
|
"flag": "🇳🇦",
|
||||||
|
"name": "Kuanyama",
|
||||||
|
"native_name": "Oshikwanyama"
|
||||||
|
},
|
||||||
|
"kk": {
|
||||||
|
"flag": "🇰🇿",
|
||||||
|
"name": "Kazakh",
|
||||||
|
"native_name": "Қазақ"
|
||||||
|
},
|
||||||
|
"kl": {
|
||||||
|
"flag": "🇬🇱",
|
||||||
|
"name": "Kalaallisut",
|
||||||
|
"native_name": "kalaallisut"
|
||||||
|
},
|
||||||
|
"km": {
|
||||||
|
"flag": "🇰🇭",
|
||||||
|
"name": "Khmer",
|
||||||
|
"native_name": "ខ្មែរ"
|
||||||
|
},
|
||||||
|
"ko": {
|
||||||
|
"flag": "🇰🇷",
|
||||||
|
"name": "Korean",
|
||||||
|
"native_name": "한국어"
|
||||||
|
},
|
||||||
|
"ks": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "Kashmiri",
|
||||||
|
"native_name": "کٲشُر"
|
||||||
|
},
|
||||||
|
"ku": {
|
||||||
|
"flag": "🇮🇶",
|
||||||
|
"name": "Kurdish",
|
||||||
|
"native_name": "Kurdî",
|
||||||
|
},
|
||||||
|
"lo": {
|
||||||
|
"flag": "🇱🇦",
|
||||||
|
"name": "Lao",
|
||||||
|
"native_name": "ລາວ"
|
||||||
|
},
|
||||||
|
"lt": {
|
||||||
|
"flag": "🇱🇹",
|
||||||
|
"name": "Lithuanian",
|
||||||
|
"native_name": "Lietuvių"
|
||||||
|
},
|
||||||
|
"lv": {
|
||||||
|
"flag": "🇱🇻",
|
||||||
|
"name": "Latvian",
|
||||||
|
"native_name": "Latviešu"
|
||||||
|
},
|
||||||
|
"mg": {
|
||||||
|
"flag": "🇲🇬",
|
||||||
|
"name": "Malagasy",
|
||||||
|
"native_name": "Malagasy"
|
||||||
|
},
|
||||||
|
"mg_mg": {
|
||||||
|
"flag": "🇲🇬",
|
||||||
|
"name": "Malagasy (Madagascar)",
|
||||||
|
"native_name": "malagasy"
|
||||||
|
},
|
||||||
|
"mh": {
|
||||||
|
"flag": "🇲🇭",
|
||||||
|
"name": "Marshallese",
|
||||||
|
"native_name": "Kajin M̧ajeļ"
|
||||||
|
},
|
||||||
|
"mk": {
|
||||||
|
"flag": "🇲🇰",
|
||||||
|
"name": "Macedonian",
|
||||||
|
"native_name": "Македонски"
|
||||||
|
},
|
||||||
|
"mn_mn": {
|
||||||
|
"flag": "🇲🇳",
|
||||||
|
"name": "Mongolian (Mongolia)",
|
||||||
|
"native_name": "Монгол хэл"
|
||||||
|
},
|
||||||
|
"mr_in": {
|
||||||
|
"flag": "🇮🇳",
|
||||||
|
"name": "Marathi (India)",
|
||||||
|
"native_name": "मराठी"
|
||||||
|
},
|
||||||
|
"ms": {
|
||||||
|
"flag": "🇲🇾",
|
||||||
|
"name": "Malay",
|
||||||
|
"native_name": "Bahasa Melayu"
|
||||||
|
},
|
||||||
|
"my": {
|
||||||
|
"flag": "🇲🇲",
|
||||||
|
"name": "Burmese",
|
||||||
|
"native_name": "မြန်မာဘာသာ"
|
||||||
|
},
|
||||||
|
"na": {
|
||||||
|
"flag": "🇳🇷",
|
||||||
|
"name": "Nauruan",
|
||||||
|
"native_name": "Dorerin Naoero"
|
||||||
|
},
|
||||||
|
"nb": {
|
||||||
|
"flag": "🇳🇴",
|
||||||
|
"name": "Norwegian Bokmål",
|
||||||
|
"native_name": "Norsk Bokmål"
|
||||||
|
},
|
||||||
|
"ng": {
|
||||||
|
"flag": "🇳🇦",
|
||||||
|
"name": "Ndonga",
|
||||||
|
"native_name": "Oshindonga"
|
||||||
|
},
|
||||||
|
"nl": {
|
||||||
|
"flag": "🇳🇱",
|
||||||
|
"name": "Dutch",
|
||||||
|
"native_name": "Nederlands"
|
||||||
|
},
|
||||||
|
"om": {
|
||||||
|
"flag": "🇪🇹",
|
||||||
|
"name": "Oromo",
|
||||||
|
"native_name": "Afaan Oromoo"
|
||||||
|
},
|
||||||
|
"os": {
|
||||||
|
"flag": "🇷🇺",
|
||||||
|
"name": "Ossetian",
|
||||||
|
"native_name": "ирон æвзаг"
|
||||||
|
},
|
||||||
|
"pl": {
|
||||||
|
"flag": "🇵🇱",
|
||||||
|
"name": "Polish",
|
||||||
|
"native_name": "Polski"
|
||||||
|
},
|
||||||
|
"pt": {
|
||||||
|
"flag": "🇵🇹",
|
||||||
|
"name": "Portuguese",
|
||||||
|
"native_name": "Português"
|
||||||
|
},
|
||||||
|
"pt_br": {
|
||||||
|
"flag": "🇧🇷",
|
||||||
|
"name": "Portuguese (Brazil)",
|
||||||
|
"native_name": "Português (Brasil)"
|
||||||
|
},
|
||||||
|
"qu": {
|
||||||
|
"flag": "🇵🇪",
|
||||||
|
"name": "Quechua",
|
||||||
|
"native_name": "Runa Simi"
|
||||||
|
},
|
||||||
|
"ro": {
|
||||||
|
"flag": "🇷🇴",
|
||||||
|
"name": "Romanian",
|
||||||
|
"native_name": "Română"
|
||||||
|
},
|
||||||
|
"ru": {
|
||||||
|
"flag": "🇷🇺",
|
||||||
|
"name": "Russian",
|
||||||
|
"native_name": "Русский"
|
||||||
|
},
|
||||||
|
"rw": {
|
||||||
|
"flag": "🇷🇼",
|
||||||
|
"name": "Kinyarwanda",
|
||||||
|
"native_name": "Ikinyarwanda"
|
||||||
|
},
|
||||||
|
"sc": {
|
||||||
|
"flag": "🇮🇹",
|
||||||
|
"name": "Sardinian",
|
||||||
|
"native_name": "sardu"
|
||||||
|
},
|
||||||
|
"sg": {
|
||||||
|
"flag": "🇨🇫",
|
||||||
|
"name": "Sango",
|
||||||
|
"native_name": "yângâ tî sängö"
|
||||||
|
},
|
||||||
|
"sk": {
|
||||||
|
"flag": "🇸🇰",
|
||||||
|
"name": "Slovak",
|
||||||
|
"native_name": "Slovenčina"
|
||||||
|
},
|
||||||
|
"sl": {
|
||||||
|
"flag": "🇸🇮",
|
||||||
|
"name": "Slovenian",
|
||||||
|
"native_name": "Slovenščina"
|
||||||
|
},
|
||||||
|
"sm": {
|
||||||
|
"flag": "🇼🇸",
|
||||||
|
"name": "Samoan",
|
||||||
|
"native_name": "Gagana Samoa"
|
||||||
|
},
|
||||||
|
"sn": {
|
||||||
|
"flag": "🇿🇼",
|
||||||
|
"name": "Shona",
|
||||||
|
"native_name": "chiShona"
|
||||||
|
},
|
||||||
|
"so": {
|
||||||
|
"flag": "🇸🇴",
|
||||||
|
"name": "Somali",
|
||||||
|
"native_name": "Soomaaliga"
|
||||||
|
},
|
||||||
|
"sr": {
|
||||||
|
"flag": "🇷🇸",
|
||||||
|
"name": "Serbian",
|
||||||
|
"native_name": "Српски"
|
||||||
|
},
|
||||||
|
"ss": {
|
||||||
|
"flag": "🇸🇿",
|
||||||
|
"name": "Swati",
|
||||||
|
"native_name": "SiSwati"
|
||||||
|
},
|
||||||
|
"st": {
|
||||||
|
"flag": "🇱🇸",
|
||||||
|
"name": "Southern Sotho",
|
||||||
|
"native_name": "Sesotho"
|
||||||
|
},
|
||||||
|
"su_id": {
|
||||||
|
"flag": "🇮🇩",
|
||||||
|
"name": "Sundanese (Indonesia)",
|
||||||
|
"native_name": "basa sunda"
|
||||||
|
},
|
||||||
|
"sv": {
|
||||||
|
"flag": "🇸🇪",
|
||||||
|
"name": "Swedish",
|
||||||
|
"native_name": "Svenska"
|
||||||
|
},
|
||||||
|
"th": {
|
||||||
|
"flag": "🇹🇭",
|
||||||
|
"name": "Thai",
|
||||||
|
"native_name": "ไทย"
|
||||||
|
},
|
||||||
|
"tk": {
|
||||||
|
"flag": "🇹🇲",
|
||||||
|
"name": "Turkmen",
|
||||||
|
"native_name": "Türkmen"
|
||||||
|
},
|
||||||
|
"tn": {
|
||||||
|
"flag": "🇧🇼",
|
||||||
|
"name": "Tswana",
|
||||||
|
"native_name": "Setswana"
|
||||||
|
},
|
||||||
|
"to": {
|
||||||
|
"flag": "🇹🇴",
|
||||||
|
"name": "Tongan",
|
||||||
|
"native_name": "faka-Tonga"
|
||||||
|
},
|
||||||
|
"tr": {
|
||||||
|
"flag": "🇹🇷",
|
||||||
|
"name": "Turkish",
|
||||||
|
"native_name": "Türkçe",
|
||||||
|
},
|
||||||
|
"ts": {
|
||||||
|
"flag": "🇿🇦",
|
||||||
|
"name": "Tsonga",
|
||||||
|
"native_name": "Xitsonga"
|
||||||
|
},
|
||||||
|
"ts_zw": {
|
||||||
|
"flag": "🇿🇼",
|
||||||
|
"name": "Tsonga (Zimbabwe)",
|
||||||
|
"native_name": "xitsonga"
|
||||||
|
},
|
||||||
|
"ty": {
|
||||||
|
"flag": "🇵🇫",
|
||||||
|
"name": "Tahitian",
|
||||||
|
"native_name": "Reo Tahiti"
|
||||||
|
},
|
||||||
|
"uk": {
|
||||||
|
"flag": "🇺🇦",
|
||||||
|
"name": "Ukrainian",
|
||||||
|
"native_name": "Українська"
|
||||||
|
},
|
||||||
|
"ur": {
|
||||||
|
"flag": "🇵🇰",
|
||||||
|
"name": "Urdu",
|
||||||
|
"native_name": "اردو"
|
||||||
|
},
|
||||||
|
"uz": {
|
||||||
|
"flag": "🇺🇿",
|
||||||
|
"name": "Uzbek",
|
||||||
|
"native_name": "oʻzbek"
|
||||||
|
},
|
||||||
|
"ve": {
|
||||||
|
"flag": "🇿🇦",
|
||||||
|
"name": "Venda",
|
||||||
|
"native_name": "Tshivenda"
|
||||||
|
},
|
||||||
|
"vi": {
|
||||||
|
"flag": "🇻🇳",
|
||||||
|
"name": "Vietnamese",
|
||||||
|
"native_name": "Tiếng Việt"
|
||||||
|
},
|
||||||
|
"vo": {
|
||||||
|
"flag": "🌍",
|
||||||
|
"name": "Volapük",
|
||||||
|
"native_name": "Volapük"
|
||||||
|
},
|
||||||
|
"wa": {
|
||||||
|
"flag": "🇧🇪",
|
||||||
|
"name": "Walloon",
|
||||||
|
"native_name": "walon"
|
||||||
|
},
|
||||||
|
"xh": {
|
||||||
|
"flag": "🇿🇦",
|
||||||
|
"name": "Xhosa",
|
||||||
|
"native_name": "isiXhosa"
|
||||||
|
},
|
||||||
|
"yi": {
|
||||||
|
"flag": "🌍",
|
||||||
|
"name": "Yiddish",
|
||||||
|
"native_name": "ייִדיש"
|
||||||
|
},
|
||||||
|
"yo": {
|
||||||
|
"flag": "🇳🇬",
|
||||||
|
"name": "Yoruba",
|
||||||
|
"native_name": "Yorùbá"
|
||||||
|
},
|
||||||
|
"zh": {
|
||||||
|
"flag": "🇨🇳",
|
||||||
|
"name": "Chinese",
|
||||||
|
"native_name": "中文"
|
||||||
|
},
|
||||||
|
"zh_hk": {
|
||||||
|
"flag": "🇭🇰",
|
||||||
|
"name": "Chinese (Hong Kong)",
|
||||||
|
"native_name": "中文(香港)"
|
||||||
|
},
|
||||||
|
"zh_tw": {
|
||||||
|
"flag": "🇹🇼",
|
||||||
|
"name": "Chinese (Taiwan)",
|
||||||
|
"native_name": "中文(台灣)"
|
||||||
|
},
|
||||||
|
"zu": {
|
||||||
|
"flag": "🇿🇦",
|
||||||
|
"name": "Zulu",
|
||||||
|
"native_name": "isiZulu"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
if True:
|
||||||
|
import toml
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
def process_data_dict(data: dict, existing: dict):
|
||||||
|
for key, value in data.items():
|
||||||
|
if key not in existing:
|
||||||
|
existing[key] = value
|
||||||
|
continue
|
||||||
|
|
||||||
|
if isinstance(value, dict):
|
||||||
|
if isinstance(existing[key], dict):
|
||||||
|
process_data_dict(value, existing[key])
|
||||||
|
continue
|
||||||
|
|
||||||
|
existing[key] = value
|
||||||
|
|
||||||
|
def process_data(data: dict, obj):
|
||||||
|
for key, value in data.items():
|
||||||
|
if not hasattr(obj, key):
|
||||||
|
continue
|
||||||
|
|
||||||
|
existing = getattr(obj, key)
|
||||||
|
if isinstance(existing, dict):
|
||||||
|
process_data_dict(value, existing)
|
||||||
|
continue
|
||||||
|
|
||||||
|
|
||||||
|
if isinstance(value, dict):
|
||||||
|
process_data(value, getattr(obj, key))
|
||||||
|
continue
|
||||||
|
|
||||||
|
setattr(obj, key, value)
|
||||||
|
|
||||||
|
|
||||||
|
config_file = Path("stsg.toml")
|
||||||
|
if config_file.exists() and config_file.is_file():
|
||||||
|
with config_file.open("r") as f:
|
||||||
|
process_data(toml.loads(f.read()), config)
|
||||||
|
|||||||
@@ -5,9 +5,14 @@ import time
|
|||||||
import os
|
import os
|
||||||
import subprocess
|
import subprocess
|
||||||
|
|
||||||
from .config import SOURCE_DIRECTORY, CODE_DIRECTORY
|
|
||||||
|
from . import config
|
||||||
from .build import build as complete_build
|
from .build import build as complete_build
|
||||||
|
|
||||||
|
|
||||||
|
CODE_DIRECTORY = "stsg"
|
||||||
|
|
||||||
|
|
||||||
logging.basicConfig(level=logging.INFO)
|
logging.basicConfig(level=logging.INFO)
|
||||||
logger = logging.getLogger("stsg")
|
logger = logging.getLogger("stsg")
|
||||||
|
|
||||||
@@ -69,7 +74,7 @@ def build_on_change():
|
|||||||
build()
|
build()
|
||||||
|
|
||||||
observer = Observer()
|
observer = Observer()
|
||||||
observer.schedule(MarkdownChangeHandler(), path=SOURCE_DIRECTORY, recursive=True)
|
observer.schedule(MarkdownChangeHandler(), path=config.setup.source_directory, recursive=True)
|
||||||
observer.start()
|
observer.start()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -97,3 +102,4 @@ def hot_reload():
|
|||||||
observer.stop()
|
observer.stop()
|
||||||
|
|
||||||
observer.join()
|
observer.join()
|
||||||
|
|
||||||
|
|||||||
339
stsg/build.py
339
stsg/build.py
@@ -4,13 +4,14 @@ import shutil
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
import os
|
import os
|
||||||
import markdown
|
import markdown
|
||||||
from typing import Optional, Union, Dict, Generator, List
|
from typing import Optional, Union, Dict, Generator, List, DefaultDict, Any
|
||||||
from bs4 import BeautifulSoup
|
from bs4 import BeautifulSoup
|
||||||
|
from collections import defaultdict
|
||||||
|
import toml
|
||||||
|
from datetime import datetime
|
||||||
|
import jinja2
|
||||||
|
|
||||||
from .config import SOURCE_DIRECTORY, DIST_DIRECTORY, LANGUAGE_INFORMATION, ARTICLE_PREVIEW_LENGTH, DEFAULT_LANGUAGE
|
from . import config
|
||||||
|
|
||||||
|
|
||||||
logger = logging.getLogger("stsg.build")
|
|
||||||
|
|
||||||
|
|
||||||
def replace_values(template: str, values: Dict[str, str]) -> str:
|
def replace_values(template: str, values: Dict[str, str]) -> str:
|
||||||
@@ -20,160 +21,260 @@ def replace_values(template: str, values: Dict[str, str]) -> str:
|
|||||||
return template
|
return template
|
||||||
|
|
||||||
|
|
||||||
def get_first_header_content(content):
|
def get_first_header_content(content, fallback: str = ""):
|
||||||
soup = BeautifulSoup(content, 'html.parser')
|
soup = BeautifulSoup(content, 'html.parser')
|
||||||
for level in range(1, 7):
|
for level in range(1, 7):
|
||||||
header = soup.find(f'h{level}')
|
header = soup.find(f'h{level}')
|
||||||
if header:
|
if header:
|
||||||
return header.get_text(strip=True)
|
return header.get_text(strip=True)
|
||||||
|
|
||||||
return self.language_code.native_name
|
return fallback
|
||||||
|
|
||||||
|
|
||||||
class Template:
|
def shorten_text_and_clean(html_string, max_length=config.formatting.article_preview_length):
|
||||||
|
soup = BeautifulSoup(html_string, 'html.parser')
|
||||||
|
|
||||||
|
# Keep track of total characters added
|
||||||
|
total_chars = 0
|
||||||
|
finished = False
|
||||||
|
|
||||||
|
# Function to recursively trim and clean text
|
||||||
|
def process_element(element):
|
||||||
|
nonlocal total_chars, finished
|
||||||
|
|
||||||
|
for child in list(element.children):
|
||||||
|
if finished:
|
||||||
|
child.extract()
|
||||||
|
continue
|
||||||
|
|
||||||
|
if isinstance(child, str):
|
||||||
|
remaining = max_length - total_chars
|
||||||
|
if remaining <= 0:
|
||||||
|
child.extract()
|
||||||
|
finished = True
|
||||||
|
elif len(child) > remaining:
|
||||||
|
child.replace_with(child[:remaining] + '...')
|
||||||
|
total_chars = max_length
|
||||||
|
finished = True
|
||||||
|
else:
|
||||||
|
total_chars += len(child)
|
||||||
|
elif hasattr(child, 'children'):
|
||||||
|
process_element(child)
|
||||||
|
# Remove empty tags
|
||||||
|
if not child.text.strip():
|
||||||
|
child.decompose()
|
||||||
|
|
||||||
|
process_element(soup)
|
||||||
|
|
||||||
|
return str(soup)
|
||||||
|
|
||||||
|
|
||||||
|
def stem_to_language_code(stem: str) -> str:
|
||||||
|
language_code = stem.lower().replace("-", "_")
|
||||||
|
|
||||||
|
if language_code in config.languages:
|
||||||
|
return language_code
|
||||||
|
|
||||||
|
language_code = language_code.split("_")[0]
|
||||||
|
if language_code in config.languages:
|
||||||
|
return language_code
|
||||||
|
|
||||||
|
logger.error("Didn't recognize %s as a valid language code, add it to the config, or fix your structure.", stem)
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class TemplateDict(dict):
|
||||||
def __init__(self, folder: Path):
|
def __init__(self, folder: Path):
|
||||||
self.folder = folder
|
self.folder = folder
|
||||||
|
super().__init__()
|
||||||
|
|
||||||
self.article: str = (self.folder / "article.html").read_text()
|
def __missing__(self, name: str) -> jinja2.Template:
|
||||||
self.overview: str = (self.folder / "overview.html").read_text()
|
f = self.folder / (name + ".html")
|
||||||
self.overview_card: str = (self.folder / "overview_card.html").read_text()
|
if not f.exists():
|
||||||
|
logger.error("no template with the name %s exists", name)
|
||||||
|
exit(1)
|
||||||
|
|
||||||
TEMPLATE = Template(Path(SOURCE_DIRECTORY, "templates"))
|
t = jinja2.Template(f.read_text())
|
||||||
|
self[name] = t
|
||||||
|
return t
|
||||||
|
|
||||||
|
TEMPLATE: Dict[str, jinja2.Template] = TemplateDict(Path(config.setup.source_directory, "templates"))
|
||||||
|
|
||||||
|
class LanguageDict(dict):
|
||||||
|
def __missing__(self, key: str):
|
||||||
|
if key not in config.languages:
|
||||||
|
raise KeyError(key)
|
||||||
|
|
||||||
|
lang_dict = config.languages[key]
|
||||||
|
lang_dict["priority"] = lang_dict.get("priority", 0)
|
||||||
|
|
||||||
|
elements = key.split("_")
|
||||||
|
if len(elements) > 1:
|
||||||
|
elements[-1] = elements[-1].upper()
|
||||||
|
lang_dict["code"] = "-".join(elements)
|
||||||
|
|
||||||
|
return lang_dict
|
||||||
|
|
||||||
|
LANGUAGES = LanguageDict()
|
||||||
|
|
||||||
|
|
||||||
class ArticleTranslation:
|
class ArticleTranslation:
|
||||||
def __init__(self, file: Path, article_overview: ArticleOverview):
|
def __init__(self, file: Path, article: Article):
|
||||||
self.file = file
|
self.file = file
|
||||||
self.article_overview = article_overview
|
self.article = article
|
||||||
self.language_code = self.file.stem
|
|
||||||
|
|
||||||
|
self.context: Dict[str, Any] = {}
|
||||||
|
|
||||||
|
# initializing the location of the article translation
|
||||||
|
self.language_code = stem_to_language_code(self.file.stem)
|
||||||
|
self.location_in_tree = [self.language_code, *self.article.location_in_tree]
|
||||||
|
self.url = "/" + "/".join(self.location_in_tree)
|
||||||
|
self.dist_path = Path(config.setup.dist_directory, *self.location_in_tree)
|
||||||
|
|
||||||
|
self.priority = LANGUAGES[self.language_code]["priority"]
|
||||||
|
self.real_language_code = LANGUAGES[self.language_code]["code"]
|
||||||
|
|
||||||
|
# TODO remove
|
||||||
self.article_content = self.file.read_text()
|
self.article_content = self.file.read_text()
|
||||||
|
self.article_preview = self.article_content[:config.formatting.article_preview_length] + "..."
|
||||||
if self.file.suffix == ".md":
|
if self.file.suffix == ".md":
|
||||||
self.article_content = markdown.markdown(self.article_content)
|
self.article_content = markdown.markdown(self.article_content)
|
||||||
|
self.article_preview = markdown.markdown(self.article_preview)
|
||||||
|
|
||||||
self.url = "/" + self.language_code + self.article_overview.url
|
self.title = get_first_header_content(self.article_content, fallback="")
|
||||||
self.dist_path = Path(DIST_DIRECTORY, self.url.strip("/"))
|
|
||||||
|
|
||||||
_language_info = DEFAULT_LANGUAGE
|
def __init_context__(self):
|
||||||
parsed_language_code = self.language_code.lower().replace("-", "_")
|
self.context["meta"] = self.article.context_shared
|
||||||
if parsed_language_code in LANGUAGE_INFORMATION:
|
self.context["url"] = self.url
|
||||||
_language_info = LANGUAGE_INFORMATION[parsed_language_code]
|
self.context["language"] = LANGUAGES[self.language_code]
|
||||||
elif parsed_language_code.split("_")[0] in LANGUAGE_INFORMATION:
|
self.context["article_url"] = self.article.url
|
||||||
_language_info = LANGUAGE_INFORMATION[parsed_language_code.split("_")[0]]
|
|
||||||
|
|
||||||
self.language_name: str = _language_info["native_name"]
|
html_content = self.file.read_text()
|
||||||
self.language_flag: str = _language_info["flag"]
|
if self.file.suffix == ".md":
|
||||||
self.priority: int = _language_info.get("priority", 0)
|
html_content = markdown.markdown(html_content)
|
||||||
|
|
||||||
|
self.context["title"] = get_first_header_content(html_content, fallback=LANGUAGES[self.language_code]["native_name"])
|
||||||
|
self.context["content"] = html_content
|
||||||
|
self.context["preview"] = shorten_text_and_clean(html_string=html_content)
|
||||||
|
|
||||||
def build(self):
|
def build(self):
|
||||||
self.dist_path.mkdir(parents=True, exist_ok=True)
|
self.dist_path.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
with Path(self.dist_path, "index.html").open("w") as f:
|
with Path(self.dist_path, "index.html").open("w") as f:
|
||||||
f.write(self.get_article())
|
f.write(TEMPLATE["article_translation"].render(self.context))
|
||||||
|
|
||||||
def _get_values(self) -> Dict[str, str]:
|
|
||||||
return {
|
|
||||||
"article_content": self.article_content,
|
|
||||||
"article_preview": self.article_content[:ARTICLE_PREVIEW_LENGTH] + "...",
|
|
||||||
"article_overview_url": self.article_overview.url,
|
|
||||||
"article_href": self.url,
|
|
||||||
"article_title": get_first_header_content(self.article_content),
|
|
||||||
"article_language_name": self.language_name,
|
|
||||||
"article_language_code": self.language_code,
|
|
||||||
"article_language_flag": self.language_flag,
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_article(self) -> str:
|
|
||||||
global TEMPLATE
|
|
||||||
return replace_values(TEMPLATE.article, self._get_values())
|
|
||||||
|
|
||||||
def get_overview_card(self) -> str:
|
|
||||||
global TEMPLATE
|
|
||||||
return replace_values(TEMPLATE.overview_card, self._get_values())
|
|
||||||
|
|
||||||
|
|
||||||
class ArticleOverview:
|
class Article:
|
||||||
def __init__(self, directory: Path, url: str = ""):
|
def __init__(self, directory: Path, location_in_tree: Optional[List[str]] = None, is_root: bool = False, parent: Optional[Article] = None):
|
||||||
self.directory = directory
|
self.directory = directory
|
||||||
self.name = self.directory.name
|
|
||||||
|
|
||||||
self.url = url + "/" + self.name
|
self.context: Dict[str, Any] = {}
|
||||||
self.dist_path = Path(DIST_DIRECTORY, self.url.strip("/"))
|
self.context_shared: Dict[str, Any] = {}
|
||||||
|
if parent is not None:
|
||||||
|
self.context["parent"] = parent.context_shared
|
||||||
|
|
||||||
self.child_articles: List[ArticleOverview] = []
|
# initializing the config values of the article
|
||||||
self.article_translations: List[ArticleTranslation] = []
|
config_file = self.directory / "index.toml"
|
||||||
|
self.config = toml.load(config_file) if config_file.exists() else {}
|
||||||
|
|
||||||
|
# initializing the location and slug of the article
|
||||||
|
self.slug = self.config.get("name", self.directory.name)
|
||||||
|
if self.slug in ARTICLE_LAKE:
|
||||||
|
logger.error("two articles have the same name at %s and %r", ARTICLE_LAKE[self.slug].directory, self.directory)
|
||||||
|
exit(1)
|
||||||
|
ARTICLE_LAKE[self.slug] = self
|
||||||
|
|
||||||
|
self.location_in_tree: List[str] = location_in_tree or []
|
||||||
|
if not is_root:
|
||||||
|
self.location_in_tree.append(self.slug)
|
||||||
|
self.url = "/" + "/".join(self.location_in_tree)
|
||||||
|
self.dist_path = Path(config.setup.dist_directory, *self.location_in_tree)
|
||||||
|
|
||||||
|
# build the tree
|
||||||
|
self.child_articles: List[Article] = []
|
||||||
|
self.article_translations_list: List[ArticleTranslation] = []
|
||||||
|
self.article_translations_map: Dict[str, ArticleTranslation] = {}
|
||||||
|
|
||||||
for c in self.directory.iterdir():
|
for c in self.directory.iterdir():
|
||||||
if c.is_file():
|
if c.name == "index.toml":
|
||||||
self.article_translations.append(ArticleTranslation(c, self))
|
|
||||||
elif c.is_dir():
|
|
||||||
self.child_articles.append(ArticleOverview(c, self.url))
|
|
||||||
|
|
||||||
# the tree is built
|
|
||||||
self.article_translations.sort(key=lambda a: a.priority, reverse=True)
|
|
||||||
self.overview_cards = "\n".join(a.get_overview_card() for a in self.article_translations)
|
|
||||||
|
|
||||||
def build(self):
|
|
||||||
# builds the tree structure to the dist directory
|
|
||||||
self.dist_path.mkdir(parents=True, exist_ok=True)
|
|
||||||
with Path(self.dist_path, "index.html").open("w") as f:
|
|
||||||
f.write(self.get_overview())
|
|
||||||
|
|
||||||
for at in self.article_translations:
|
|
||||||
at.build()
|
|
||||||
|
|
||||||
for ca in self.child_articles:
|
|
||||||
ca.build()
|
|
||||||
|
|
||||||
|
|
||||||
def _get_values(self) -> Dict[str, str]:
|
|
||||||
return {
|
|
||||||
"overview_cards": self.overview_cards,
|
|
||||||
"overview_slug": self.name,
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_overview(self) -> str:
|
|
||||||
global TEMPLATE
|
|
||||||
return replace_values(TEMPLATE.overview, self._get_values())
|
|
||||||
|
|
||||||
|
|
||||||
def copy_static():
|
|
||||||
src = str(Path(SOURCE_DIRECTORY, "static"))
|
|
||||||
dst = str(Path(DIST_DIRECTORY, "static"))
|
|
||||||
|
|
||||||
|
|
||||||
if not os.path.exists(src):
|
|
||||||
logger.warn("The static folder '%s' wasn't defined.", src)
|
|
||||||
return
|
|
||||||
|
|
||||||
logger.info("copying static files from '%s' to '%r'", src, dst)
|
|
||||||
|
|
||||||
os.makedirs(dst, exist_ok=True)
|
|
||||||
|
|
||||||
for root, dirs, files in os.walk(src):
|
|
||||||
if any(p.startswith(".") for p in Path(root).parts):
|
|
||||||
continue
|
|
||||||
|
|
||||||
# Compute relative path from the source root
|
|
||||||
rel_path = os.path.relpath(root, src)
|
|
||||||
dest_dir = os.path.join(dst, rel_path)
|
|
||||||
|
|
||||||
os.makedirs(dest_dir, exist_ok=True)
|
|
||||||
|
|
||||||
|
|
||||||
for file in files:
|
|
||||||
if file.startswith("."):
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
src_file = os.path.join(root, file)
|
if c.is_file():
|
||||||
dest_file = os.path.join(dest_dir, file)
|
at = ArticleTranslation(c, self)
|
||||||
shutil.copy2(src_file, dest_file)
|
self.article_translations_list.append(at)
|
||||||
|
self.article_translations_map[at.language_code] = at
|
||||||
|
elif c.is_dir():
|
||||||
|
self.child_articles.append(Article(
|
||||||
|
directory=c,
|
||||||
|
location_in_tree=self.location_in_tree.copy(),
|
||||||
|
parent=self,
|
||||||
|
))
|
||||||
|
|
||||||
|
self.article_translations_list.sort(key=lambda a: a.priority, reverse=True)
|
||||||
|
|
||||||
|
logger.info("found %s at %s with the translations %s", self.slug, ".".join(list(self.location_in_tree)), ",".join(self.article_translations_map.keys()))
|
||||||
|
|
||||||
|
def __init_context__(self):
|
||||||
|
self.context_shared["url"] = self.url
|
||||||
|
self.context_shared["slug"] = self.slug
|
||||||
|
|
||||||
|
modified_at = datetime.fromisoformat(self.config["datetime"]) if "datetime" in self.config else datetime.fromtimestamp(self.directory.stat().st_mtime)
|
||||||
|
self.context_shared["date"] = modified_at.strftime(config.formatting.datetime_format)
|
||||||
|
self.context_shared["iso_date"] = modified_at.isoformat()
|
||||||
|
|
||||||
|
self.context.update(self.context_shared)
|
||||||
|
|
||||||
|
# recursive context structures
|
||||||
|
translation_list = self.context["translations"] = []
|
||||||
|
child_article_list = self.context["children"] = []
|
||||||
|
|
||||||
|
for article_translation in self.article_translations_list:
|
||||||
|
self.context[article_translation.real_language_code] = article_translation.context
|
||||||
|
translation_list.append(article_translation.context)
|
||||||
|
|
||||||
|
for child_article in self.child_articles:
|
||||||
|
child_article_list.append(child_article.context)
|
||||||
|
|
||||||
|
# recursively build context
|
||||||
|
for at in self.article_translations_list:
|
||||||
|
at.__init_context__()
|
||||||
|
for a in self.child_articles:
|
||||||
|
a.__init_context__()
|
||||||
|
|
||||||
|
def build(self):
|
||||||
|
self.dist_path.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
|
with Path(self.dist_path, "index.html").open("w") as f:
|
||||||
|
f.write(TEMPLATE["article"].render(self.context))
|
||||||
|
|
||||||
|
for at in self.article_translations_list:
|
||||||
|
at.build()
|
||||||
|
|
||||||
|
for ac in self.child_articles:
|
||||||
|
ac.build()
|
||||||
|
|
||||||
|
|
||||||
|
# GLOBALS
|
||||||
|
logger = logging.getLogger("stsg.build")
|
||||||
|
ARTICLE_LAKE: Dict[str, Article] = {}
|
||||||
|
ARTICLE_REFERENCE_VALUES: DefaultDict[str, Dict[str, str]] = defaultdict(dict)
|
||||||
|
|
||||||
def build():
|
def build():
|
||||||
logger.info("building static page")
|
logger.info("starting build process...")
|
||||||
|
|
||||||
copy_static()
|
logger.info("copying static folder...")
|
||||||
tree = ArticleOverview(directory=Path(SOURCE_DIRECTORY, "pages"))
|
shutil.copytree(Path(config.setup.source_directory, "static"), Path(config.setup.dist_directory, "static"), dirs_exist_ok=True)
|
||||||
|
|
||||||
|
logger.info("building page tree...")
|
||||||
|
tree = Article(directory=Path(config.setup.source_directory, "articles"), is_root=True)
|
||||||
|
|
||||||
|
logger.info("compiling tree context...")
|
||||||
|
tree.__init_context__()
|
||||||
|
|
||||||
|
import json
|
||||||
|
with Path("context.json").open("w") as f:
|
||||||
|
json.dump(tree.context, f, indent=4)
|
||||||
|
|
||||||
|
logger.info("dumping page tree...")
|
||||||
tree.build()
|
tree.build()
|
||||||
769
stsg/config.py
769
stsg/config.py
@@ -1,769 +0,0 @@
|
|||||||
import typing
|
|
||||||
import typing_extensions
|
|
||||||
|
|
||||||
SOURCE_DIRECTORY = "src"
|
|
||||||
DIST_DIRECTORY = "dist"
|
|
||||||
|
|
||||||
# config template stuff
|
|
||||||
ARTICLE_PREVIEW_LENGTH = 200
|
|
||||||
|
|
||||||
# FOR DEVELOPMENT
|
|
||||||
CODE_DIRECTORY = "stsg"
|
|
||||||
|
|
||||||
# LANGUAGE INFORMATION
|
|
||||||
LANGUAGE_INFORMATION: typing.Dict[str, {
|
|
||||||
"flag": str,
|
|
||||||
"name": str,
|
|
||||||
"native_name": str,
|
|
||||||
"priority": typing_extensions.NotRequired[int],
|
|
||||||
}] = {
|
|
||||||
"af": {
|
|
||||||
"flag": "🇿🇦",
|
|
||||||
"name": "Afrikaans",
|
|
||||||
"native_name": "Afrikaans"
|
|
||||||
},
|
|
||||||
"am": {
|
|
||||||
"flag": "🇪🇹",
|
|
||||||
"name": "Amharic",
|
|
||||||
"native_name": "አማርኛ"
|
|
||||||
},
|
|
||||||
"an": {
|
|
||||||
"flag": "🇪🇸",
|
|
||||||
"name": "Aragonese",
|
|
||||||
"native_name": "aragonés"
|
|
||||||
},
|
|
||||||
"ar": {
|
|
||||||
"flag": "🇸🇦",
|
|
||||||
"name": "Arabic",
|
|
||||||
"native_name": "العربية"
|
|
||||||
},
|
|
||||||
"ar_ae": {
|
|
||||||
"flag": "🇦🇪",
|
|
||||||
"name": "Arabic (UAE)",
|
|
||||||
"native_name": "العربية (الإمارات)"
|
|
||||||
},
|
|
||||||
"ar_bh": {
|
|
||||||
"flag": "🇧🇭",
|
|
||||||
"name": "Arabic (Bahrain)",
|
|
||||||
"native_name": "العربية (البحرين)"
|
|
||||||
},
|
|
||||||
"ar_dz": {
|
|
||||||
"flag": "🇩🇿",
|
|
||||||
"name": "Arabic (Algeria)",
|
|
||||||
"native_name": "العربية (الجزائر)"
|
|
||||||
},
|
|
||||||
"ar_eg": {
|
|
||||||
"flag": "🇪🇬",
|
|
||||||
"name": "Arabic (Egypt)",
|
|
||||||
"native_name": "العربية (مصر)"
|
|
||||||
},
|
|
||||||
"ar_iq": {
|
|
||||||
"flag": "🇮🇶",
|
|
||||||
"name": "Arabic (Iraq)",
|
|
||||||
"native_name": "العربية (العراق)"
|
|
||||||
},
|
|
||||||
"ar_jo": {
|
|
||||||
"flag": "🇯🇴",
|
|
||||||
"name": "Arabic (Jordan)",
|
|
||||||
"native_name": "العربية (الأردن)"
|
|
||||||
},
|
|
||||||
"ar_kw": {
|
|
||||||
"flag": "🇰🇼",
|
|
||||||
"name": "Arabic (Kuwait)",
|
|
||||||
"native_name": "العربية (الكويت)"
|
|
||||||
},
|
|
||||||
"ar_lb": {
|
|
||||||
"flag": "🇱🇧",
|
|
||||||
"name": "Arabic (Lebanon)",
|
|
||||||
"native_name": "العربية (لبنان)"
|
|
||||||
},
|
|
||||||
"ar_ly": {
|
|
||||||
"flag": "🇱🇾",
|
|
||||||
"name": "Arabic (Libya)",
|
|
||||||
"native_name": "العربية (ليبيا)"
|
|
||||||
},
|
|
||||||
"ar_ma": {
|
|
||||||
"flag": "🇲🇦",
|
|
||||||
"name": "Arabic (Morocco)",
|
|
||||||
"native_name": "العربية (المغرب)"
|
|
||||||
},
|
|
||||||
"ar_om": {
|
|
||||||
"flag": "🇴🇲",
|
|
||||||
"name": "Arabic (Oman)",
|
|
||||||
"native_name": "العربية (عُمان)"
|
|
||||||
},
|
|
||||||
"ar_qa": {
|
|
||||||
"flag": "🇶🇦",
|
|
||||||
"name": "Arabic (Qatar)",
|
|
||||||
"native_name": "العربية (قطر)"
|
|
||||||
},
|
|
||||||
"ar_sa": {
|
|
||||||
"flag": "🇸🇦",
|
|
||||||
"name": "Arabic (Saudi Arabia)",
|
|
||||||
"native_name": "العربية (السعودية)"
|
|
||||||
},
|
|
||||||
"ar_sd": {
|
|
||||||
"flag": "🇸🇩",
|
|
||||||
"name": "Arabic (Sudan)",
|
|
||||||
"native_name": "العربية (السودان)"
|
|
||||||
},
|
|
||||||
"ar_sy": {
|
|
||||||
"flag": "🇸🇾",
|
|
||||||
"name": "Arabic (Syria)",
|
|
||||||
"native_name": "العربية (سوريا)",
|
|
||||||
"priority": 50,
|
|
||||||
},
|
|
||||||
"ar_tn": {
|
|
||||||
"flag": "🇹🇳",
|
|
||||||
"name": "Arabic (Tunisia)",
|
|
||||||
"native_name": "العربية (تونس)"
|
|
||||||
},
|
|
||||||
"ar_ye": {
|
|
||||||
"flag": "🇾🇪",
|
|
||||||
"name": "Arabic (Yemen)",
|
|
||||||
"native_name": "العربية (اليمن)"
|
|
||||||
},
|
|
||||||
"ars_ae": {
|
|
||||||
"flag": "🇦🇪",
|
|
||||||
"name": "Najdi Arabic (UAE)",
|
|
||||||
"native_name": "نَجْدِيّ"
|
|
||||||
},
|
|
||||||
"ars_arab_sa": {
|
|
||||||
"flag": "🇸🇦",
|
|
||||||
"name": "Najdi Arabic (Saudi Arabia, Arabic Script)",
|
|
||||||
"native_name": "نَجْدِيّ"
|
|
||||||
},
|
|
||||||
"ars_sa": {
|
|
||||||
"flag": "🇸🇦",
|
|
||||||
"name": "Najdi Arabic (Saudi Arabia)",
|
|
||||||
"native_name": "نَجْدِيّ"
|
|
||||||
},
|
|
||||||
"as": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "Assamese",
|
|
||||||
"native_name": "অসমীয়া"
|
|
||||||
},
|
|
||||||
"az": {
|
|
||||||
"flag": "🇦🇿",
|
|
||||||
"name": "Azerbaijani",
|
|
||||||
"native_name": "Azərbaycan"
|
|
||||||
},
|
|
||||||
"be": {
|
|
||||||
"flag": "🇧🇾",
|
|
||||||
"name": "Belarusian",
|
|
||||||
"native_name": "Беларуская"
|
|
||||||
},
|
|
||||||
"bg": {
|
|
||||||
"flag": "🇧🇬",
|
|
||||||
"name": "Bulgarian",
|
|
||||||
"native_name": "Български"
|
|
||||||
},
|
|
||||||
"bm": {
|
|
||||||
"flag": "🇲🇱",
|
|
||||||
"name": "Bambara",
|
|
||||||
"native_name": "bamanankan"
|
|
||||||
},
|
|
||||||
"bn": {
|
|
||||||
"flag": "🇧🇩",
|
|
||||||
"name": "Bengali",
|
|
||||||
"native_name": "বাংলা"
|
|
||||||
},
|
|
||||||
"bn_in": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "Bengali (India)",
|
|
||||||
"native_name": "বাংলা (ভারত)"
|
|
||||||
},
|
|
||||||
"br": {
|
|
||||||
"flag": "🏴",
|
|
||||||
"name": "Breton",
|
|
||||||
"native_name": "brezhoneg"
|
|
||||||
},
|
|
||||||
"bs": {
|
|
||||||
"flag": "🇧🇦",
|
|
||||||
"name": "Bosnian",
|
|
||||||
"native_name": "Bosanski"
|
|
||||||
},
|
|
||||||
"ca": {
|
|
||||||
"flag": "🇪🇸",
|
|
||||||
"name": "Catalan",
|
|
||||||
"native_name": "Català"
|
|
||||||
},
|
|
||||||
"crh": {
|
|
||||||
"flag": "🇺🇦",
|
|
||||||
"name": "Crimean Tatar",
|
|
||||||
"native_name": "qırımtatarca"
|
|
||||||
},
|
|
||||||
"cs": {
|
|
||||||
"flag": "🇨🇿",
|
|
||||||
"name": "Czech",
|
|
||||||
"native_name": "Čeština"
|
|
||||||
},
|
|
||||||
"cv": {
|
|
||||||
"flag": "🇷🇺",
|
|
||||||
"name": "Chuvash",
|
|
||||||
"native_name": "чӑваш чӗлхи"
|
|
||||||
},
|
|
||||||
"cy": {
|
|
||||||
"flag": "🏴",
|
|
||||||
"name": "Welsh",
|
|
||||||
"native_name": "Cymraeg"
|
|
||||||
},
|
|
||||||
"da": {
|
|
||||||
"flag": "🇩🇰",
|
|
||||||
"name": "Danish",
|
|
||||||
"native_name": "Dansk"
|
|
||||||
},
|
|
||||||
"de": {
|
|
||||||
"flag": "🇩🇪",
|
|
||||||
"name": "German",
|
|
||||||
"native_name": "Deutsch",
|
|
||||||
"priority": 100,
|
|
||||||
},
|
|
||||||
"de_at": {
|
|
||||||
"flag": "🇦🇹",
|
|
||||||
"name": "German (Austria)",
|
|
||||||
"native_name": "Deutsch (Österreich)"
|
|
||||||
},
|
|
||||||
"de_be": {
|
|
||||||
"flag": "🇧🇪",
|
|
||||||
"name": "German (Belgium)",
|
|
||||||
"native_name": "Deutsch (Belgien)"
|
|
||||||
},
|
|
||||||
"de_ch": {
|
|
||||||
"flag": "🇨🇭",
|
|
||||||
"name": "German (Switzerland)",
|
|
||||||
"native_name": "Deutsch (Schweiz)"
|
|
||||||
},
|
|
||||||
"dv": {
|
|
||||||
"flag": "🇲🇻",
|
|
||||||
"name": "Dhivehi",
|
|
||||||
"native_name": "ދިވެހި"
|
|
||||||
},
|
|
||||||
"dz": {
|
|
||||||
"flag": "🇧🇹",
|
|
||||||
"name": "Dzongkha",
|
|
||||||
"native_name": "རྫོང་ཁ"
|
|
||||||
},
|
|
||||||
"el": {
|
|
||||||
"flag": "🇬🇷",
|
|
||||||
"name": "Greek",
|
|
||||||
"native_name": "Ελληνικά"
|
|
||||||
},
|
|
||||||
"en": {
|
|
||||||
"flag": "🇺🇸",
|
|
||||||
"name": "English",
|
|
||||||
"native_name": "English",
|
|
||||||
"priority": 80,
|
|
||||||
},
|
|
||||||
"en_au": {
|
|
||||||
"flag": "🇦🇺",
|
|
||||||
"name": "English (Australia)",
|
|
||||||
"native_name": "English (Australia)"
|
|
||||||
},
|
|
||||||
"en_ca": {
|
|
||||||
"flag": "🇨🇦",
|
|
||||||
"name": "English (Canada)",
|
|
||||||
"native_name": "English (Canada)"
|
|
||||||
},
|
|
||||||
"en_gb": {
|
|
||||||
"flag": "🇬🇧",
|
|
||||||
"name": "English (UK)",
|
|
||||||
"native_name": "English (UK)",
|
|
||||||
"priority": 80,
|
|
||||||
},
|
|
||||||
"en_ie": {
|
|
||||||
"flag": "🇮🇪",
|
|
||||||
"name": "English (Ireland)",
|
|
||||||
"native_name": "English (Ireland)"
|
|
||||||
},
|
|
||||||
"en_in": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "English (India)",
|
|
||||||
"native_name": "English (India)"
|
|
||||||
},
|
|
||||||
"en_nz": {
|
|
||||||
"flag": "🇳🇿",
|
|
||||||
"name": "English (New Zealand)",
|
|
||||||
"native_name": "English (New Zealand)"
|
|
||||||
},
|
|
||||||
"en_us": {
|
|
||||||
"flag": "🇺🇸",
|
|
||||||
"name": "English (US)",
|
|
||||||
"native_name": "English (US)",
|
|
||||||
},
|
|
||||||
"es": {
|
|
||||||
"flag": "🇪🇸",
|
|
||||||
"name": "Spanish",
|
|
||||||
"native_name": "Español"
|
|
||||||
},
|
|
||||||
"es_ar": {
|
|
||||||
"flag": "🇦🇷",
|
|
||||||
"name": "Spanish (Argentina)",
|
|
||||||
"native_name": "Español (Argentina)"
|
|
||||||
},
|
|
||||||
"es_mx": {
|
|
||||||
"flag": "🇲🇽",
|
|
||||||
"name": "Spanish (Mexico)",
|
|
||||||
"native_name": "Español (México)"
|
|
||||||
},
|
|
||||||
"et": {
|
|
||||||
"flag": "🇪🇪",
|
|
||||||
"name": "Estonian",
|
|
||||||
"native_name": "Eesti"
|
|
||||||
},
|
|
||||||
"fa": {
|
|
||||||
"flag": "🇮🇷",
|
|
||||||
"name": "Persian",
|
|
||||||
"native_name": "فارسی"
|
|
||||||
},
|
|
||||||
"ff": {
|
|
||||||
"flag": "🌍",
|
|
||||||
"name": "Fula",
|
|
||||||
"native_name": "Fulfulde"
|
|
||||||
},
|
|
||||||
"fi": {
|
|
||||||
"flag": "🇫🇮",
|
|
||||||
"name": "Finnish",
|
|
||||||
"native_name": "Suomi"
|
|
||||||
},
|
|
||||||
"fo": {
|
|
||||||
"flag": "🇫🇴",
|
|
||||||
"name": "Faroese",
|
|
||||||
"native_name": "føroyskt"
|
|
||||||
},
|
|
||||||
"fr": {
|
|
||||||
"flag": "🇫🇷",
|
|
||||||
"name": "French",
|
|
||||||
"native_name": "Français"
|
|
||||||
},
|
|
||||||
"fr_ca": {
|
|
||||||
"flag": "🇨🇦",
|
|
||||||
"name": "French (Canada)",
|
|
||||||
"native_name": "Français (Canada)"
|
|
||||||
},
|
|
||||||
"fr_ch": {
|
|
||||||
"flag": "🇨🇭",
|
|
||||||
"name": "French (Switzerland)",
|
|
||||||
"native_name": "Français (Suisse)"
|
|
||||||
},
|
|
||||||
"ga": {
|
|
||||||
"flag": "🇮🇪",
|
|
||||||
"name": "Irish",
|
|
||||||
"native_name": "Gaeilge"
|
|
||||||
},
|
|
||||||
"gl": {
|
|
||||||
"flag": "🇪🇸",
|
|
||||||
"name": "Galician",
|
|
||||||
"native_name": "Galego"
|
|
||||||
},
|
|
||||||
"gn": {
|
|
||||||
"flag": "🇵🇾",
|
|
||||||
"name": "Guarani",
|
|
||||||
"native_name": "Avañe'ẽ"
|
|
||||||
},
|
|
||||||
"gu": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "Gujarati",
|
|
||||||
"native_name": "ગુજરાતી"
|
|
||||||
},
|
|
||||||
"ha": {
|
|
||||||
"flag": "🇳🇬",
|
|
||||||
"name": "Hausa",
|
|
||||||
"native_name": "هَوُسَ"
|
|
||||||
},
|
|
||||||
"he": {
|
|
||||||
"flag": "🇮🇱",
|
|
||||||
"name": "Hebrew",
|
|
||||||
"native_name": "עברית"
|
|
||||||
},
|
|
||||||
"hi": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "Hindi",
|
|
||||||
"native_name": "हिन्दी"
|
|
||||||
},
|
|
||||||
"hr": {
|
|
||||||
"flag": "🇭🇷",
|
|
||||||
"name": "Croatian",
|
|
||||||
"native_name": "Hrvatski"
|
|
||||||
},
|
|
||||||
"ht": {
|
|
||||||
"flag": "🇭🇹",
|
|
||||||
"name": "Haitian Creole",
|
|
||||||
"native_name": "Kreyòl ayisyen"
|
|
||||||
},
|
|
||||||
"hu": {
|
|
||||||
"flag": "🇭🇺",
|
|
||||||
"name": "Hungarian",
|
|
||||||
"native_name": "Magyar"
|
|
||||||
},
|
|
||||||
"id": {
|
|
||||||
"flag": "🇮🇩",
|
|
||||||
"name": "Indonesian",
|
|
||||||
"native_name": "Bahasa Indonesia"
|
|
||||||
},
|
|
||||||
"io": {
|
|
||||||
"flag": "🌍",
|
|
||||||
"name": "Ido",
|
|
||||||
"native_name": "Ido"
|
|
||||||
},
|
|
||||||
"is": {
|
|
||||||
"flag": "🇮🇸",
|
|
||||||
"name": "Icelandic",
|
|
||||||
"native_name": "Íslenska"
|
|
||||||
},
|
|
||||||
"it": {
|
|
||||||
"flag": "🇮🇹",
|
|
||||||
"name": "Italian",
|
|
||||||
"native_name": "Italiano"
|
|
||||||
},
|
|
||||||
"ja": {
|
|
||||||
"flag": "🇯🇵",
|
|
||||||
"name": "Japanese",
|
|
||||||
"native_name": "日本語"
|
|
||||||
},
|
|
||||||
"jv_id": {
|
|
||||||
"flag": "🇮🇩",
|
|
||||||
"name": "Javanese (Indonesia)",
|
|
||||||
"native_name": "basa jawa"
|
|
||||||
},
|
|
||||||
"ka": {
|
|
||||||
"flag": "🇬🇪",
|
|
||||||
"name": "Georgian",
|
|
||||||
"native_name": "ქართული"
|
|
||||||
},
|
|
||||||
"kg": {
|
|
||||||
"flag": "🇨🇬",
|
|
||||||
"name": "Kongo",
|
|
||||||
"native_name": "KiKongo"
|
|
||||||
},
|
|
||||||
"kj": {
|
|
||||||
"flag": "🇳🇦",
|
|
||||||
"name": "Kuanyama",
|
|
||||||
"native_name": "Oshikwanyama"
|
|
||||||
},
|
|
||||||
"kk": {
|
|
||||||
"flag": "🇰🇿",
|
|
||||||
"name": "Kazakh",
|
|
||||||
"native_name": "Қазақ"
|
|
||||||
},
|
|
||||||
"kl": {
|
|
||||||
"flag": "🇬🇱",
|
|
||||||
"name": "Kalaallisut",
|
|
||||||
"native_name": "kalaallisut"
|
|
||||||
},
|
|
||||||
"km": {
|
|
||||||
"flag": "🇰🇭",
|
|
||||||
"name": "Khmer",
|
|
||||||
"native_name": "ខ្មែរ"
|
|
||||||
},
|
|
||||||
"ko": {
|
|
||||||
"flag": "🇰🇷",
|
|
||||||
"name": "Korean",
|
|
||||||
"native_name": "한국어"
|
|
||||||
},
|
|
||||||
"ks": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "Kashmiri",
|
|
||||||
"native_name": "کٲشُر"
|
|
||||||
},
|
|
||||||
"ku": {
|
|
||||||
"flag": "🇮🇶",
|
|
||||||
"name": "Kurdish",
|
|
||||||
"native_name": "Kurdî",
|
|
||||||
"priority": 49,
|
|
||||||
},
|
|
||||||
"lo": {
|
|
||||||
"flag": "🇱🇦",
|
|
||||||
"name": "Lao",
|
|
||||||
"native_name": "ລາວ"
|
|
||||||
},
|
|
||||||
"lt": {
|
|
||||||
"flag": "🇱🇹",
|
|
||||||
"name": "Lithuanian",
|
|
||||||
"native_name": "Lietuvių"
|
|
||||||
},
|
|
||||||
"lv": {
|
|
||||||
"flag": "🇱🇻",
|
|
||||||
"name": "Latvian",
|
|
||||||
"native_name": "Latviešu"
|
|
||||||
},
|
|
||||||
"mg": {
|
|
||||||
"flag": "🇲🇬",
|
|
||||||
"name": "Malagasy",
|
|
||||||
"native_name": "Malagasy"
|
|
||||||
},
|
|
||||||
"mg_mg": {
|
|
||||||
"flag": "🇲🇬",
|
|
||||||
"name": "Malagasy (Madagascar)",
|
|
||||||
"native_name": "malagasy"
|
|
||||||
},
|
|
||||||
"mh": {
|
|
||||||
"flag": "🇲🇭",
|
|
||||||
"name": "Marshallese",
|
|
||||||
"native_name": "Kajin M̧ajeļ"
|
|
||||||
},
|
|
||||||
"mk": {
|
|
||||||
"flag": "🇲🇰",
|
|
||||||
"name": "Macedonian",
|
|
||||||
"native_name": "Македонски"
|
|
||||||
},
|
|
||||||
"mn_mn": {
|
|
||||||
"flag": "🇲🇳",
|
|
||||||
"name": "Mongolian (Mongolia)",
|
|
||||||
"native_name": "Монгол хэл"
|
|
||||||
},
|
|
||||||
"mr_in": {
|
|
||||||
"flag": "🇮🇳",
|
|
||||||
"name": "Marathi (India)",
|
|
||||||
"native_name": "मराठी"
|
|
||||||
},
|
|
||||||
"ms": {
|
|
||||||
"flag": "🇲🇾",
|
|
||||||
"name": "Malay",
|
|
||||||
"native_name": "Bahasa Melayu"
|
|
||||||
},
|
|
||||||
"my": {
|
|
||||||
"flag": "🇲🇲",
|
|
||||||
"name": "Burmese",
|
|
||||||
"native_name": "မြန်မာဘာသာ"
|
|
||||||
},
|
|
||||||
"na": {
|
|
||||||
"flag": "🇳🇷",
|
|
||||||
"name": "Nauruan",
|
|
||||||
"native_name": "Dorerin Naoero"
|
|
||||||
},
|
|
||||||
"nb": {
|
|
||||||
"flag": "🇳🇴",
|
|
||||||
"name": "Norwegian Bokmål",
|
|
||||||
"native_name": "Norsk Bokmål"
|
|
||||||
},
|
|
||||||
"ng": {
|
|
||||||
"flag": "🇳🇦",
|
|
||||||
"name": "Ndonga",
|
|
||||||
"native_name": "Oshindonga"
|
|
||||||
},
|
|
||||||
"nl": {
|
|
||||||
"flag": "🇳🇱",
|
|
||||||
"name": "Dutch",
|
|
||||||
"native_name": "Nederlands"
|
|
||||||
},
|
|
||||||
"om": {
|
|
||||||
"flag": "🇪🇹",
|
|
||||||
"name": "Oromo",
|
|
||||||
"native_name": "Afaan Oromoo"
|
|
||||||
},
|
|
||||||
"os": {
|
|
||||||
"flag": "🇷🇺",
|
|
||||||
"name": "Ossetian",
|
|
||||||
"native_name": "ирон æвзаг"
|
|
||||||
},
|
|
||||||
"pl": {
|
|
||||||
"flag": "🇵🇱",
|
|
||||||
"name": "Polish",
|
|
||||||
"native_name": "Polski"
|
|
||||||
},
|
|
||||||
"pt": {
|
|
||||||
"flag": "🇵🇹",
|
|
||||||
"name": "Portuguese",
|
|
||||||
"native_name": "Português"
|
|
||||||
},
|
|
||||||
"pt_br": {
|
|
||||||
"flag": "🇧🇷",
|
|
||||||
"name": "Portuguese (Brazil)",
|
|
||||||
"native_name": "Português (Brasil)"
|
|
||||||
},
|
|
||||||
"qu": {
|
|
||||||
"flag": "🇵🇪",
|
|
||||||
"name": "Quechua",
|
|
||||||
"native_name": "Runa Simi"
|
|
||||||
},
|
|
||||||
"ro": {
|
|
||||||
"flag": "🇷🇴",
|
|
||||||
"name": "Romanian",
|
|
||||||
"native_name": "Română"
|
|
||||||
},
|
|
||||||
"ru": {
|
|
||||||
"flag": "🇷🇺",
|
|
||||||
"name": "Russian",
|
|
||||||
"native_name": "Русский"
|
|
||||||
},
|
|
||||||
"rw": {
|
|
||||||
"flag": "🇷🇼",
|
|
||||||
"name": "Kinyarwanda",
|
|
||||||
"native_name": "Ikinyarwanda"
|
|
||||||
},
|
|
||||||
"sc": {
|
|
||||||
"flag": "🇮🇹",
|
|
||||||
"name": "Sardinian",
|
|
||||||
"native_name": "sardu"
|
|
||||||
},
|
|
||||||
"sg": {
|
|
||||||
"flag": "🇨🇫",
|
|
||||||
"name": "Sango",
|
|
||||||
"native_name": "yângâ tî sängö"
|
|
||||||
},
|
|
||||||
"sk": {
|
|
||||||
"flag": "🇸🇰",
|
|
||||||
"name": "Slovak",
|
|
||||||
"native_name": "Slovenčina"
|
|
||||||
},
|
|
||||||
"sl": {
|
|
||||||
"flag": "🇸🇮",
|
|
||||||
"name": "Slovenian",
|
|
||||||
"native_name": "Slovenščina"
|
|
||||||
},
|
|
||||||
"sm": {
|
|
||||||
"flag": "🇼🇸",
|
|
||||||
"name": "Samoan",
|
|
||||||
"native_name": "Gagana Samoa"
|
|
||||||
},
|
|
||||||
"sn": {
|
|
||||||
"flag": "🇿🇼",
|
|
||||||
"name": "Shona",
|
|
||||||
"native_name": "chiShona"
|
|
||||||
},
|
|
||||||
"so": {
|
|
||||||
"flag": "🇸🇴",
|
|
||||||
"name": "Somali",
|
|
||||||
"native_name": "Soomaaliga"
|
|
||||||
},
|
|
||||||
"sr": {
|
|
||||||
"flag": "🇷🇸",
|
|
||||||
"name": "Serbian",
|
|
||||||
"native_name": "Српски"
|
|
||||||
},
|
|
||||||
"ss": {
|
|
||||||
"flag": "🇸🇿",
|
|
||||||
"name": "Swati",
|
|
||||||
"native_name": "SiSwati"
|
|
||||||
},
|
|
||||||
"st": {
|
|
||||||
"flag": "🇱🇸",
|
|
||||||
"name": "Southern Sotho",
|
|
||||||
"native_name": "Sesotho"
|
|
||||||
},
|
|
||||||
"su_id": {
|
|
||||||
"flag": "🇮🇩",
|
|
||||||
"name": "Sundanese (Indonesia)",
|
|
||||||
"native_name": "basa sunda"
|
|
||||||
},
|
|
||||||
"sv": {
|
|
||||||
"flag": "🇸🇪",
|
|
||||||
"name": "Swedish",
|
|
||||||
"native_name": "Svenska"
|
|
||||||
},
|
|
||||||
"th": {
|
|
||||||
"flag": "🇹🇭",
|
|
||||||
"name": "Thai",
|
|
||||||
"native_name": "ไทย"
|
|
||||||
},
|
|
||||||
"tk": {
|
|
||||||
"flag": "🇹🇲",
|
|
||||||
"name": "Turkmen",
|
|
||||||
"native_name": "Türkmen"
|
|
||||||
},
|
|
||||||
"tn": {
|
|
||||||
"flag": "🇧🇼",
|
|
||||||
"name": "Tswana",
|
|
||||||
"native_name": "Setswana"
|
|
||||||
},
|
|
||||||
"to": {
|
|
||||||
"flag": "🇹🇴",
|
|
||||||
"name": "Tongan",
|
|
||||||
"native_name": "faka-Tonga"
|
|
||||||
},
|
|
||||||
"tr": {
|
|
||||||
"flag": "🇹🇷",
|
|
||||||
"name": "Turkish",
|
|
||||||
"native_name": "Türkçe",
|
|
||||||
"priority": 60,
|
|
||||||
},
|
|
||||||
"ts": {
|
|
||||||
"flag": "🇿🇦",
|
|
||||||
"name": "Tsonga",
|
|
||||||
"native_name": "Xitsonga"
|
|
||||||
},
|
|
||||||
"ts_zw": {
|
|
||||||
"flag": "🇿🇼",
|
|
||||||
"name": "Tsonga (Zimbabwe)",
|
|
||||||
"native_name": "xitsonga"
|
|
||||||
},
|
|
||||||
"ty": {
|
|
||||||
"flag": "🇵🇫",
|
|
||||||
"name": "Tahitian",
|
|
||||||
"native_name": "Reo Tahiti"
|
|
||||||
},
|
|
||||||
"uk": {
|
|
||||||
"flag": "🇺🇦",
|
|
||||||
"name": "Ukrainian",
|
|
||||||
"native_name": "Українська"
|
|
||||||
},
|
|
||||||
"ur": {
|
|
||||||
"flag": "🇵🇰",
|
|
||||||
"name": "Urdu",
|
|
||||||
"native_name": "اردو"
|
|
||||||
},
|
|
||||||
"uz": {
|
|
||||||
"flag": "🇺🇿",
|
|
||||||
"name": "Uzbek",
|
|
||||||
"native_name": "oʻzbek"
|
|
||||||
},
|
|
||||||
"ve": {
|
|
||||||
"flag": "🇿🇦",
|
|
||||||
"name": "Venda",
|
|
||||||
"native_name": "Tshivenda"
|
|
||||||
},
|
|
||||||
"vi": {
|
|
||||||
"flag": "🇻🇳",
|
|
||||||
"name": "Vietnamese",
|
|
||||||
"native_name": "Tiếng Việt"
|
|
||||||
},
|
|
||||||
"vo": {
|
|
||||||
"flag": "🌍",
|
|
||||||
"name": "Volapük",
|
|
||||||
"native_name": "Volapük"
|
|
||||||
},
|
|
||||||
"wa": {
|
|
||||||
"flag": "🇧🇪",
|
|
||||||
"name": "Walloon",
|
|
||||||
"native_name": "walon"
|
|
||||||
},
|
|
||||||
"xh": {
|
|
||||||
"flag": "🇿🇦",
|
|
||||||
"name": "Xhosa",
|
|
||||||
"native_name": "isiXhosa"
|
|
||||||
},
|
|
||||||
"yi": {
|
|
||||||
"flag": "🌍",
|
|
||||||
"name": "Yiddish",
|
|
||||||
"native_name": "ייִדיש"
|
|
||||||
},
|
|
||||||
"yo": {
|
|
||||||
"flag": "🇳🇬",
|
|
||||||
"name": "Yoruba",
|
|
||||||
"native_name": "Yorùbá"
|
|
||||||
},
|
|
||||||
"zh": {
|
|
||||||
"flag": "🇨🇳",
|
|
||||||
"name": "Chinese",
|
|
||||||
"native_name": "中文"
|
|
||||||
},
|
|
||||||
"zh_hk": {
|
|
||||||
"flag": "🇭🇰",
|
|
||||||
"name": "Chinese (Hong Kong)",
|
|
||||||
"native_name": "中文(香港)"
|
|
||||||
},
|
|
||||||
"zh_tw": {
|
|
||||||
"flag": "🇹🇼",
|
|
||||||
"name": "Chinese (Taiwan)",
|
|
||||||
"native_name": "中文(台灣)"
|
|
||||||
},
|
|
||||||
"zu": {
|
|
||||||
"flag": "🇿🇦",
|
|
||||||
"name": "Zulu",
|
|
||||||
"native_name": "isiZulu"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
DEFAULT_LANGUAGE = LANGUAGE_INFORMATION["de"]
|
|
||||||
Reference in New Issue
Block a user