feat: add language picker to post form

Co-authored-by: naskya <m@naskya.net>
This commit is contained in:
Essem 2023-12-23 07:25:08 +09:00 committed by naskya
parent 8b6f88f7aa
commit b64d84e7d1
Signed by: naskya
GPG key ID: 712D413B3A9FED5C
22 changed files with 457 additions and 800 deletions

View file

@ -40,6 +40,7 @@
- マージされていない本家版へのプルリクエストを独断でマージ
- RTL Layout Support ([!10452](https://git.joinfirefish.org/firefish/firefish/-/merge_requests/10452))
- Add language picker to post form ([!10616](https://git.joinfirefish.org/firefish/firefish/-/merge_requests/10616))
- `emojis` の API エンドポイントMisskey v13- 互換)を追加([firefish-mkdir](https://git.mkdir.uk/hiira/firefish-mkdir) から取り込み)
- Docker のベースイメージに Node v21 を使用
- HTML のコードに入るコメントアートを削除

View file

@ -1171,6 +1171,8 @@ pullDownToReload: "Pull down to reload"
enableTimelineStreaming: "Update timelines automatically"
useEmojiCdn: "Get Twemoji from CDN"
useEmojiCdnDescription: "Use Twemoji from the JSDelivr CDN instead of the server's assets."
suggested: "Suggested"
noLanguage: "No language"
_sensitiveMediaDetection:
description: "Reduces the effort of server moderation through automatically recognizing

View file

@ -2032,6 +2032,8 @@ _iconSets:
regular: "標準"
fill: "塗りつぶし"
duotone: "2色"
suggested: "候補"
noLanguage: "言語なし"
moreUrls: "固定するページ"
moreUrlsDescription: "左下のヘルプメニューに固定したいページを以下の形式で、改行区切りで入力してください:\n\"表示名\": https://example.com/"
releaseToReload: "離して再読み込み"

7
neko/pnpm-lock.yaml generated
View file

@ -201,6 +201,9 @@ importers:
file-type:
specifier: 18.7.0
version: 18.7.0
firefish-js:
specifier: workspace:*
version: link:../firefish-js
fluent-ffmpeg:
specifier: 2.1.2
version: 2.1.2
@ -393,9 +396,6 @@ importers:
tinycolor2:
specifier: 1.6.0
version: 1.6.0
tinyld:
specifier: 1.3.4
version: 1.3.4
tmp:
specifier: 0.2.1
version: 0.2.1
@ -15760,6 +15760,7 @@ packages:
resolution: {integrity: sha512-u26CNoaInA4XpDU+8s/6Cq8xHc2T5M4fXB3ICfXPokUQoLzmPgSZU02TAkFwFMJCWTjk53gtkS8pETTreZwCqw==}
engines: {node: '>= 12.10.0', npm: '>= 6.12.0', yarn: '>= 1.20.0'}
hasBin: true
dev: true
/titleize@3.0.0:
resolution: {integrity: sha512-KxVu8EYHDPBdUYdKZdKtU2aj2XfEx9AfjXxE/Aj0vT06w2icA09Vus1rh6eSu1y01akYg6BjIK/hxyLJINoMLQ==}

View file

@ -62,6 +62,7 @@
"feed": "4.2.2",
"file-type": "18.7.0",
"fluent-ffmpeg": "2.1.2",
"firefish-js": "workspace:*",
"got": "13.0.0",
"gunzip-maybe": "1.4.2",
"happy-dom": "12.10.3",
@ -125,7 +126,6 @@
"tar-stream": "3.1.6",
"tesseract.js": "5.0.3",
"tinycolor2": "1.6.0",
"tinyld": "1.3.4",
"tmp": "0.2.1",
"typeorm": "0.3.17",
"ulid": "2.3.0",

View file

@ -1,11 +0,0 @@
import { detect } from "tinyld";
import * as mfm from "mfm-js";
export default function detectLanguage(text: string): string {
const nodes = mfm.parse(text);
const filtered = mfm.extract(nodes, (node) => {
return node.type === "text" || node.type === "quote";
});
const purified = mfm.toString(filtered);
return detect(purified);
}

View file

@ -1,217 +1,71 @@
// TODO: sharedに置いてフロントエンドのと統合したい
export const langmap = {
ach: {
nativeName: "Lwo",
},
ady: {
nativeName: "Адыгэбзэ",
},
export const iso639Langs1 = {
af: {
nativeName: "Afrikaans",
},
"af-NA": {
nativeName: "Afrikaans (Namibia)",
},
"af-ZA": {
nativeName: "Afrikaans (South Africa)",
},
ak: {
nativeName: "Tɕɥi",
},
ar: {
nativeName: "العربية",
rtl: true,
},
"ar-AR": {
nativeName: "العربية",
},
"ar-MA": {
nativeName: "العربية",
},
"ar-SA": {
nativeName: "العربية (السعودية)",
},
"ay-BO": {
ay: {
nativeName: "Aymar aru",
},
az: {
nativeName: "Azərbaycan dili",
},
"az-AZ": {
nativeName: "Azərbaycan dili",
},
"be-BY": {
be: {
nativeName: "Беларуская",
},
bg: {
nativeName: "Български",
},
"bg-BG": {
nativeName: "Български",
},
bn: {
nativeName: "বাংলা",
},
"bn-IN": {
nativeName: "বাংলা (ভারত)",
},
"bn-BD": {
nativeName: "বাংলা(বাংলাদেশ)",
},
br: {
nativeName: "Brezhoneg",
},
"bs-BA": {
bs: {
nativeName: "Bosanski",
},
ca: {
nativeName: "Català",
},
"ca-ES": {
nativeName: "Català",
},
cak: {
nativeName: "Maya Kaqchikel",
},
"ck-US": {
nativeName: "ᏣᎳᎩ (tsalagi)",
},
cs: {
nativeName: "Čeština",
},
"cs-CZ": {
nativeName: "Čeština",
},
cy: {
nativeName: "Cymraeg",
},
"cy-GB": {
nativeName: "Cymraeg",
},
da: {
nativeName: "Dansk",
},
"da-DK": {
nativeName: "Dansk",
},
de: {
nativeName: "Deutsch",
},
"de-AT": {
nativeName: "Deutsch (Österreich)",
},
"de-DE": {
nativeName: "Deutsch (Deutschland)",
},
"de-CH": {
nativeName: "Deutsch (Schweiz)",
},
dsb: {
nativeName: "Dolnoserbšćina",
},
el: {
nativeName: "Ελληνικά",
},
"el-GR": {
nativeName: "Ελληνικά",
},
en: {
nativeName: "English",
},
"en-GB": {
nativeName: "English (UK)",
},
"en-AU": {
nativeName: "English (Australia)",
},
"en-CA": {
nativeName: "English (Canada)",
},
"en-IE": {
nativeName: "English (Ireland)",
},
"en-IN": {
nativeName: "English (India)",
},
"en-PI": {
nativeName: "English (Pirate)",
},
"en-SG": {
nativeName: "English (Singapore)",
},
"en-UD": {
nativeName: "English (Upside Down)",
},
"en-US": {
nativeName: "English (US)",
},
"en-ZA": {
nativeName: "English (South Africa)",
},
"en@pirate": {
nativeName: "English (Pirate)",
},
eo: {
nativeName: "Esperanto",
},
"eo-EO": {
nativeName: "Esperanto",
},
es: {
nativeName: "Español",
},
"es-AR": {
nativeName: "Español (Argentine)",
},
"es-419": {
nativeName: "Español (Latinoamérica)",
},
"es-CL": {
nativeName: "Español (Chile)",
},
"es-CO": {
nativeName: "Español (Colombia)",
},
"es-EC": {
nativeName: "Español (Ecuador)",
},
"es-ES": {
nativeName: "Español (España)",
},
"es-LA": {
nativeName: "Español (Latinoamérica)",
},
"es-NI": {
nativeName: "Español (Nicaragua)",
},
"es-MX": {
nativeName: "Español (México)",
},
"es-US": {
nativeName: "Español (Estados Unidos)",
},
"es-VE": {
nativeName: "Español (Venezuela)",
},
et: {
nativeName: "eesti keel",
},
"et-EE": {
nativeName: "Eesti (Estonia)",
},
eu: {
nativeName: "Euskara",
},
"eu-ES": {
nativeName: "Euskara",
},
fa: {
nativeName: "فارسی",
},
"fa-IR": {
nativeName: "فارسی",
},
"fb-LT": {
nativeName: "Leet Speak",
rtl: true,
},
ff: {
nativeName: "Fulah",
@ -219,154 +73,86 @@ export const langmap = {
fi: {
nativeName: "Suomi",
},
"fi-FI": {
nativeName: "Suomi",
},
fo: {
nativeName: "Føroyskt",
},
"fo-FO": {
nativeName: "Føroyskt (Færeyjar)",
},
fr: {
nativeName: "Français",
},
"fr-CA": {
nativeName: "Français (Canada)",
},
"fr-FR": {
nativeName: "Français (France)",
},
"fr-BE": {
nativeName: "Français (Belgique)",
},
"fr-CH": {
nativeName: "Français (Suisse)",
},
"fy-NL": {
fy: {
nativeName: "Frysk",
},
ga: {
nativeName: "Gaeilge",
},
"ga-IE": {
nativeName: "Gaeilge",
},
gd: {
nativeName: "Gàidhlig",
},
gl: {
nativeName: "Galego",
},
"gl-ES": {
nativeName: "Galego",
},
"gn-PY": {
gn: {
nativeName: "Avañe'ẽ",
},
"gu-IN": {
gu: {
nativeName: "ગુજરાતી",
},
gv: {
nativeName: "Gaelg",
},
"gx-GR": {
nativeName: "Ἑλληνική ἀρχαία",
},
he: {
nativeName: "עברית‏",
},
"he-IL": {
nativeName: "עברית‏",
rtl: true,
},
hi: {
nativeName: "हिन्दी",
},
"hi-IN": {
nativeName: "हिन्दी",
},
hr: {
nativeName: "Hrvatski",
},
"hr-HR": {
nativeName: "Hrvatski",
},
hsb: {
nativeName: "Hornjoserbšćina",
},
ht: {
nativeName: "Kreyòl",
},
hu: {
nativeName: "Magyar",
},
"hu-HU": {
nativeName: "Magyar",
},
hy: {
nativeName: "Հայերեն",
},
"hy-AM": {
nativeName: "Հայերեն (Հայաստան)",
},
id: {
nativeName: "Bahasa Indonesia",
},
"id-ID": {
nativeName: "Bahasa Indonesia",
},
is: {
nativeName: "Íslenska",
},
"is-IS": {
nativeName: "Íslenska (Iceland)",
},
it: {
nativeName: "Italiano",
},
"it-IT": {
nativeName: "Italiano",
},
ja: {
nativeName: "日本語",
},
"ja-JP": {
nativeName: "日本語 (日本)",
},
"jv-ID": {
jv: {
nativeName: "Basa Jawa",
},
"ka-GE": {
ka: {
nativeName: "ქართული",
},
"kk-KZ": {
kk: {
nativeName: "Қазақша",
},
km: {
nativeName: "ភាសាខ្មែរ",
},
kl: {
nativeName: "kalaallisut",
},
"km-KH": {
km: {
nativeName: "ភាសាខ្មែរ",
},
kab: {
nativeName: "Taqbaylit",
},
kn: {
nativeName: "ಕನ್ನಡ",
},
"kn-IN": {
nativeName: "ಕನ್ನಡ (India)",
},
ko: {
nativeName: "한국어",
},
"ko-KR": {
nativeName: "한국어 (한국)",
},
"ku-TR": {
ku: {
nativeName: "Kurdî",
},
kw: {
@ -375,66 +161,39 @@ export const langmap = {
la: {
nativeName: "Latin",
},
"la-VA": {
nativeName: "Latin",
},
lb: {
nativeName: "Lëtzebuergesch",
},
"li-NL": {
li: {
nativeName: "Lèmbörgs",
},
lt: {
nativeName: "Lietuvių",
},
"lt-LT": {
nativeName: "Lietuvių",
},
lv: {
nativeName: "Latviešu",
},
"lv-LV": {
nativeName: "Latviešu",
},
mai: {
nativeName: "मैथिली, মৈথিলী",
},
"mg-MG": {
mg: {
nativeName: "Malagasy",
},
mk: {
nativeName: "Македонски",
},
"mk-MK": {
nativeName: "Македонски (Македонски)",
},
ml: {
nativeName: "മലയാളം",
},
"ml-IN": {
nativeName: "മലയാളം",
},
"mn-MN": {
mn: {
nativeName: "Монгол",
},
mr: {
nativeName: "मराठी",
},
"mr-IN": {
nativeName: "मराठी",
},
ms: {
nativeName: "Bahasa Melayu",
},
"ms-MY": {
nativeName: "Bahasa Melayu",
},
mt: {
nativeName: "Malti",
},
"mt-MT": {
nativeName: "Malti",
},
my: {
nativeName: "ဗမာစကာ",
},
@ -444,223 +203,179 @@ export const langmap = {
nb: {
nativeName: "Norsk (bokmål)",
},
"nb-NO": {
nativeName: "Norsk (bokmål)",
},
ne: {
nativeName: "नेपाली",
},
"ne-NP": {
nativeName: "नेपाली",
},
nl: {
nativeName: "Nederlands",
},
"nl-BE": {
nativeName: "Nederlands (België)",
},
"nl-NL": {
nativeName: "Nederlands (Nederland)",
},
"nn-NO": {
nn: {
nativeName: "Norsk (nynorsk)",
},
oc: {
nativeName: "Occitan",
},
"or-IN": {
or: {
nativeName: "ଓଡ଼ିଆ",
},
pa: {
nativeName: "ਪੰਜਾਬੀ",
},
"pa-IN": {
nativeName: "ਪੰਜਾਬੀ (ਭਾਰਤ ਨੂੰ)",
},
pl: {
nativeName: "Polski",
},
"pl-PL": {
nativeName: "Polski",
},
"ps-AF": {
ps: {
nativeName: "پښتو",
rtl: true,
},
pt: {
nativeName: "Português",
},
"pt-BR": {
nativeName: "Português (Brasil)",
},
"pt-PT": {
nativeName: "Português (Portugal)",
},
"qu-PE": {
qu: {
nativeName: "Qhichwa",
},
"rm-CH": {
rm: {
nativeName: "Rumantsch",
},
ro: {
nativeName: "Română",
},
"ro-RO": {
nativeName: "Română",
},
ru: {
nativeName: "Русский",
},
"ru-RU": {
nativeName: "Русский",
},
"sa-IN": {
sa: {
nativeName: "संस्कृतम्",
},
"se-NO": {
se: {
nativeName: "Davvisámegiella",
},
sh: {
nativeName: "српскохрватски",
},
"si-LK": {
si: {
nativeName: "සිංහල",
},
sk: {
nativeName: "Slovenčina",
},
"sk-SK": {
nativeName: "Slovenčina (Slovakia)",
},
sl: {
nativeName: "Slovenščina",
},
"sl-SI": {
nativeName: "Slovenščina",
},
"so-SO": {
so: {
nativeName: "Soomaaliga",
},
sq: {
nativeName: "Shqip",
},
"sq-AL": {
nativeName: "Shqip",
},
sr: {
nativeName: "Српски",
},
"sr-RS": {
nativeName: "Српски (Serbia)",
},
su: {
nativeName: "Basa Sunda",
},
sv: {
nativeName: "Svenska",
},
"sv-SE": {
nativeName: "Svenska",
},
sw: {
nativeName: "Kiswahili",
},
"sw-KE": {
nativeName: "Kiswahili",
},
ta: {
nativeName: "தமிழ்",
},
"ta-IN": {
nativeName: "தமிழ்",
},
te: {
nativeName: "తెలుగు",
},
"te-IN": {
nativeName: "తెలుగు",
},
tg: {
nativeName: "забо́ни тоҷикӣ́",
},
"tg-TJ": {
nativeName: "тоҷикӣ",
},
th: {
nativeName: "ภาษาไทย",
},
"th-TH": {
nativeName: "ภาษาไทย (ประเทศไทย)",
},
fil: {
nativeName: "Filipino",
},
tlh: {
nativeName: "tlhIngan-Hol",
},
tr: {
nativeName: "Türkçe",
},
"tr-TR": {
nativeName: "Türkçe",
},
"tt-RU": {
tt: {
nativeName: "татарча",
},
uk: {
nativeName: "Українська",
},
"uk-UA": {
nativeName: "Українська",
},
ur: {
nativeName: "اردو",
},
"ur-PK": {
nativeName: "اردو",
rtl: true,
},
uz: {
nativeName: "O'zbek",
},
"uz-UZ": {
nativeName: "O'zbek",
},
vi: {
nativeName: "Tiếng Việt",
},
"vi-VN": {
nativeName: "Tiếng Việt",
},
"xh-ZA": {
xh: {
nativeName: "isiXhosa",
},
yi: {
nativeName: "ייִדיש",
},
"yi-DE": {
nativeName: "ייִדיש (German)",
rtl: true,
},
zh: {
nativeName: "中文",
},
"zh-Hans": {
nativeName: "中文简体",
},
"zh-Hant": {
nativeName: "中文繁體",
},
"zh-CN": {
nativeName: "中文(中国大陆)",
},
"zh-HK": {
nativeName: "中文(香港)",
},
"zh-SG": {
nativeName: "中文(新加坡)",
},
"zh-TW": {
nativeName: "中文(台灣)",
},
"zu-ZA": {
zu: {
nativeName: "isiZulu",
},
};
export const iso639Langs3 = {
ach: {
nativeName: "Lwo",
},
ady: {
nativeName: "Адыгэбзэ",
},
cak: {
nativeName: "Maya Kaqchikel",
},
chr: {
nativeName: "ᏣᎳᎩ (tsalagi)",
},
dsb: {
nativeName: "Dolnoserbšćina",
},
fil: {
nativeName: "Filipino",
},
hsb: {
nativeName: "Hornjoserbšćina",
},
kab: {
nativeName: "Taqbaylit",
},
mai: {
nativeName: "मैथिली, মৈথিলী",
},
tlh: {
nativeName: "tlhIngan-Hol",
},
tok: {
nativeName: "Toki Pona",
},
yue: {
nativeName: "粵語",
},
nan: {
nativeName: "閩南語",
},
};
export const langmapNoRegion = Object.assign({}, iso639Langs1, iso639Langs3);
export const iso639Regional = {
"zh-hans": {
nativeName: "中文(简体)",
},
"zh-hant": {
nativeName: "中文(繁體)",
},
};
export const langmap = Object.assign({}, langmapNoRegion, iso639Regional);

View file

@ -1,3 +1,5 @@
import { langmap } from "@/misc/langmap.js";
export const packedNoteSchema = {
type: "object",
properties: {
@ -19,6 +21,11 @@ export const packedNoteSchema = {
optional: false,
nullable: true,
},
lang: {
type: "string",
enum: [...Object.keys(langmap)],
nullable: true,
},
cw: {
type: "string",
optional: true,

View file

@ -309,16 +309,13 @@ export async function createNote(
) {
text = note.source.content;
if (note.contentMap != null) {
const key = Object.keys(note.contentMap)[0];
lang = Object.keys(langmap).includes(key)
? key.trim().split("-")[0].split("@")[0]
: null;
const key = Object.keys(note.contentMap)[0].toLowerCase();
lang = Object.keys(langmap).includes(key) ? key : null;
}
} else if (note.contentMap != null) {
const entry = Object.entries(note.contentMap)[0];
lang = Object.keys(langmap).includes(entry[0])
? entry[0].trim().split("-")[0].split("@")[0]
: null;
const key = entry[0].toLowerCase();
lang = Object.keys(langmap).includes(key) ? key : null;
text = htmlToMfm(entry[1], note.tag);
} else if (typeof note.content === "string") {
text = htmlToMfm(note.content, note.tag);
@ -584,15 +581,12 @@ export async function updateNote(value: string | IObject, resolver?: Resolver) {
text = post.source.content;
if (post.contentMap != null) {
const key = Object.keys(post.contentMap)[0];
lang = Object.keys(langmap).includes(key)
? key.trim().split("-")[0].split("@")[0]
: null;
lang = Object.keys(langmap).includes(key) ? key : null;
}
} else if (post.contentMap != null) {
const entry = Object.entries(post.contentMap)[0];
lang = Object.keys(langmap).includes(entry[0])
? entry[0].trim().split("-")[0].split("@")[0]
: null;
const key = entry[0].toLowerCase();
lang = Object.keys(langmap).includes(key) ? key : null;
text = htmlToMfm(entry[1], post.tag);
} else if (typeof post.content === "string") {
text = htmlToMfm(post.content, post.tag);

View file

@ -6,7 +6,6 @@ import { DriveFiles, Notes, Users, Emojis, Polls } from "@/models/index.js";
import type { Emoji } from "@/models/entities/emoji.js";
import type { Poll } from "@/models/entities/poll.js";
import toHtml from "@/remote/activitypub/misc/get-note-html.js";
import detectLanguage from "@/misc/detect-language.js";
import renderEmoji from "./emoji.js";
import renderMention from "./mention.js";
import renderHashtag from "./hashtag.js";
@ -115,10 +114,9 @@ export default async function renderNote(
}),
);
const lang = note.lang ?? detectLanguage(text);
const contentMap = lang
const contentMap = note.lang
? {
[lang]: content,
[note.lang]: content,
}
: null;

View file

@ -90,7 +90,7 @@ export const paramDef = {
birthday: { ...Users.birthdaySchema, nullable: true },
lang: {
type: "string",
enum: [null, ...Object.keys(langmap)],
enum: Object.keys(langmap),
nullable: true,
},
avatarId: { type: "string", format: "misskey:id", nullable: true },

View file

@ -1,5 +1,6 @@
import { In } from "typeorm";
import create from "@/services/note/create.js";
import { langmap } from "@/misc/langmap.js";
import type { User } from "@/models/entities/user.js";
import {
Users,
@ -108,7 +109,11 @@ export const paramDef = {
},
},
text: { type: "string", maxLength: MAX_NOTE_TEXT_LENGTH, nullable: true },
lang: { type: "string", nullable: true, maxLength: 10 },
lang: {
type: "string",
enum: Object.keys(langmap),
nullable: true,
},
cw: { type: "string", nullable: true, maxLength: 100 },
localOnly: { type: "boolean", default: false },
noExtractMentions: { type: "boolean", default: false },

View file

@ -35,7 +35,6 @@ import renderUpdate from "@/remote/activitypub/renderer/update.js";
import { deliverToRelays } from "@/services/relay.js";
// import { deliverQuestionUpdate } from "@/services/note/polls/update.js";
import { langmap } from "@/misc/langmap.js";
import detectLanguage from "@/misc/detect-language.js";
export const meta = {
tags: ["notes"],
@ -170,7 +169,11 @@ export const paramDef = {
},
},
text: { type: "string", maxLength: MAX_NOTE_TEXT_LENGTH, nullable: true },
lang: { type: "string", nullable: true, maxLength: 10 },
lang: {
type: "string",
enum: Object.keys(langmap),
nullable: true,
},
cw: { type: "string", nullable: true, maxLength: 250 },
localOnly: { type: "boolean", default: false },
noExtractMentions: { type: "boolean", default: false },
@ -378,11 +381,9 @@ export default define(meta, paramDef, async (ps, user) => {
}
if (ps.lang) {
if (!Object.keys(langmap).includes(ps.lang.trim()))
if (!Object.keys(langmap).includes(ps.lang.toLowerCase()))
throw new Error("invalid param");
ps.lang = ps.lang.trim().split("-")[0].split("@")[0];
} else if (ps.text) {
ps.lang = detectLanguage(ps.text);
ps.lang = ps.lang.toLowerCase();
} else {
ps.lang = null;
}

View file

@ -63,7 +63,6 @@ import meilisearch from "@/db/meilisearch.js";
import { redisClient } from "@/db/redis.js";
import { Mutex } from "redis-semaphore";
import { langmap } from "@/misc/langmap.js";
import detectLanguage from "@/misc/detect-language.js";
const mutedWordsCache = new Cache<
{ userId: UserProfile["userId"]; mutedWords: UserProfile["mutedWords"] }[]
@ -276,11 +275,9 @@ export default async (
}
if (data.lang) {
if (!Object.keys(langmap).includes(data.lang.trim()))
if (!Object.keys(langmap).includes(data.lang.toLowerCase()))
throw new Error("invalid param");
data.lang = data.lang.trim().split("-")[0].split("@")[0];
} else if (data.text) {
data.lang = detectLanguage(data.text);
data.lang = data.lang.toLowerCase();
} else {
data.lang = null;
}

View file

@ -54,6 +54,21 @@
><i :class="icon('ph-eye-slash')"></i
></span>
</button>
<button
ref="languageButton"
v-tooltip="i18n.ts.language"
class="_button language"
@click="setLanguage"
>
<i
v-if="language === '' || language == null"
class="_button"
:class="icon('ph-seal-warning')"
></i>
<p v-else class="_button" style="font-weight: bold">
{{ language.split("-")[0] }}
</p>
</button>
<button
v-tooltip="i18n.ts.previewNoteText"
class="_button preview"
@ -293,6 +308,9 @@ import { uploadFile } from "@/scripts/upload";
import { deepClone } from "@/scripts/clone";
import preprocess from "@/scripts/preprocess";
import { vibrate } from "@/scripts/vibrate";
import { langmap } from "@/scripts/langmap";
import { MenuItem } from "@/types/menu";
import detectLanguage from "@/scripts/detect-language";
import icon from "@/scripts/icon";
const modal = inject("modal");
@ -306,6 +324,7 @@ const props = withDefaults(
specified?: firefish.entities.User;
initialText?: string;
initialVisibility?: typeof firefish.noteVisibilities;
initialLanguage?: typeof firefish.languages;
initialFiles?: firefish.entities.DriveFile[];
initialLocalOnly?: boolean;
initialVisibleUsers?: firefish.entities.User[];
@ -333,6 +352,7 @@ const textareaEl = ref<HTMLTextAreaElement | null>(null);
const cwInputEl = ref<HTMLInputElement | null>(null);
const hashtagsInputEl = ref<HTMLInputElement | null>(null);
const visibilityButton = ref<HTMLElement | null>(null);
const languageButton = ref<HTMLElement | undefined>();
const showBigPostButton = defaultStore.state.showBigPostButton;
@ -360,6 +380,7 @@ const visibility = ref(
: defaultStore.state
.defaultNoteVisibility) as (typeof firefish.noteVisibilities)[number]),
);
const visibleUsers = ref([]);
if (props.initialVisibleUsers) {
props.initialVisibleUsers.forEach(pushVisibleUser);
@ -581,6 +602,7 @@ function watchForDraft() {
watch(files, () => saveDraft(), { deep: true });
watch(visibility, () => saveDraft());
watch(localOnly, () => saveDraft());
watch(language, () => saveDraft());
}
function checkMissingMention() {
@ -708,6 +730,105 @@ function setVisibility() {
);
}
const language = ref<string | null>(
props.initialLanguage ??
defaultStore.state.recentlyUsedPostLanguages[0] ??
localStorage.getItem("lang")?.split("-")[0],
);
function filterLangmapByPrefix(
prefix: string,
): { langCode: string; nativeName: string }[] {
let to_return = Object.entries(langmap)
.filter(([langCode, _]) => langCode.startsWith(prefix))
.map(([langCode, v]) => {
return { langCode, nativeName: v.nativeName };
});
if (prefix === "zh")
to_return = to_return.concat([
{ langCode: "yue", nativeName: langmap["yue"].nativeName },
{ langCode: "nan", nativeName: langmap["nan"].nativeName },
]);
return to_return;
}
function setLanguage() {
const actions: Array<MenuItem> = [];
const detectedLanguage: string = detectLanguage(text.value) ?? "";
if (detectedLanguage !== "" && detectedLanguage !== language.value) {
actions.push({
type: "label",
text: i18n.ts.suggested,
});
filterLangmapByPrefix(detectedLanguage).forEach((v) => {
actions.push({
text: v.nativeName,
danger: false,
active: false,
action: () => {
language.value = v.langCode;
},
});
});
actions.push(null);
}
if (language.value != null)
actions.push({
text: langmap[language.value].nativeName,
danger: false,
active: true,
action: () => {},
});
const langs = Object.keys(langmap);
// Show recently used language first
let recentlyUsedLanguagesExist = false;
for (const lang of defaultStore.state.recentlyUsedPostLanguages) {
if (lang === language.value) continue;
if (!langs.includes(lang)) continue;
actions.push({
text: langmap[lang].nativeName,
danger: false,
active: false,
action: () => {
language.value = lang;
},
});
recentlyUsedLanguagesExist = true;
}
if (recentlyUsedLanguagesExist) actions.push(null);
actions.push({
text: i18n.ts.noLanguage,
danger: false,
active: false,
action: () => {
language.value = null;
},
});
for (const lang of langs) {
if (lang === language.value) continue;
if (defaultStore.state.recentlyUsedPostLanguages.includes(lang))
continue;
actions.push({
text: langmap[lang].nativeName,
danger: false,
active: false,
action: () => {
language.value = lang;
},
});
}
os.popupMenu(actions, languageButton.value, {});
}
function pushVisibleUser(user) {
if (
!visibleUsers.value.some(
@ -859,6 +980,7 @@ function saveDraft() {
cw: cw.value,
visibility: visibility.value,
localOnly: localOnly.value,
lang: language.value,
files: files.value,
poll: poll.value,
},
@ -892,6 +1014,7 @@ async function post() {
channelId: props.channel ? props.channel.id : undefined,
poll: poll.value,
cw: useCw.value ? cw.value || "" : undefined,
lang: language.value ? language.value : undefined,
localOnly: localOnly.value,
visibility:
visibility.value === "private" ? "specified" : visibility.value,
@ -961,6 +1084,20 @@ async function post() {
});
});
vibrate([10, 20, 10, 20, 10, 20, 60]);
// update recentlyUsedLanguages
if (language.value != null) {
const languages = Object.keys(langmap);
const maxLength = 6;
defaultStore.state.recentlyUsedPostLanguages = [language.value]
.concat(
defaultStore.state.recentlyUsedPostLanguages.filter((lang) => {
return lang !== language.value && languages.includes(lang);
}),
)
.slice(0, maxLength);
}
}
function cancel() {
@ -1047,6 +1184,7 @@ onMounted(() => {
cw.value = draft.data.cw;
visibility.value = draft.data.visibility;
localOnly.value = draft.data.localOnly;
language.value = draft.data.lang;
files.value = (draft.data.files || []).filter(
(draftFile) => draftFile,
);
@ -1073,6 +1211,7 @@ onMounted(() => {
}
visibility.value = init.visibility;
localOnly.value = init.localOnly;
language.value = init.lang;
quoteId.value = init.renote ? init.renote.id : null;
}
@ -1150,6 +1289,11 @@ onMounted(() => {
opacity: 0.7;
}
> .language {
height: 34px;
width: 34px;
}
> .preview {
display: inline-block;
padding: 0;

View file

@ -33,6 +33,7 @@ const props = defineProps<{
specified?: firefish.entities.User;
initialText?: string;
initialVisibility?: typeof firefish.noteVisibilities;
initialLanguage?: typeof firefish.languages;
initialFiles?: firefish.entities.DriveFile[];
initialLocalOnly?: boolean;
initialVisibleUsers?: firefish.entities.User[];

View file

@ -10,7 +10,7 @@ export const apiUrl = `${url}/api`;
export const wsUrl = `${url
.replace("http://", "ws://")
.replace("https://", "wss://")}/streaming`;
export const lang = localStorage.getItem("lang");
export const lang = localStorage.getItem("lang")?.split("-")[0] ?? "en";
export const langs = _LANGS_;
export const locale = JSON.parse(localStorage.getItem("locale") || "en-US");
export const version = _VERSION_;

View file

@ -124,7 +124,7 @@ function checkForSplash() {
// #region Set lang attr
const html = document.documentElement;
html.setAttribute("lang", lang || "en-US");
html.setAttribute("dir", langmap[lang].rtl ? "rtl" : "ltr");
html.setAttribute("dir", langmap[lang].rtl === true ? "rtl" : "ltr");
//#endregion
// #region loginId

View file

@ -1,20 +1,7 @@
// TODO: sharedに置いてバックエンドのと統合したい
export const langmap = {
ach: {
nativeName: "Lwo",
},
ady: {
nativeName: "Адыгэбзэ",
},
export const iso639Langs1 = {
af: {
nativeName: "Afrikaans",
},
"af-NA": {
nativeName: "Afrikaans (Namibia)",
},
"af-ZA": {
nativeName: "Afrikaans (South Africa)",
},
ak: {
nativeName: "Tɕɥi",
},
@ -22,359 +9,150 @@ export const langmap = {
nativeName: "العربية",
rtl: true,
},
"ar-AR": {
nativeName: "العربية",
rtl: true,
},
"ar-MA": {
nativeName: "العربية",
rtl: true,
},
"ar-SA": {
nativeName: "العربية (السعودية)",
rtl: true,
},
"ay-BO": {
ay: {
nativeName: "Aymar aru",
},
az: {
nativeName: "Azərbaycan dili",
},
"az-AZ": {
nativeName: "Azərbaycan dili",
},
"be-BY": {
be: {
nativeName: "Беларуская",
},
bg: {
nativeName: "Български",
},
"bg-BG": {
nativeName: "Български",
},
bn: {
nativeName: "বাংলা",
},
"bn-IN": {
nativeName: "বাংলা (ভারত)",
},
"bn-BD": {
nativeName: "বাংলা(বাংলাদেশ)",
},
br: {
nativeName: "Brezhoneg",
},
"bs-BA": {
bs: {
nativeName: "Bosanski",
},
ca: {
nativeName: "Català",
},
"ca-ES": {
nativeName: "Català",
},
cak: {
nativeName: "Maya Kaqchikel",
},
"ck-US": {
nativeName: "ᏣᎳᎩ (tsalagi)",
},
cs: {
nativeName: "Čeština",
},
"cs-CZ": {
nativeName: "Čeština",
},
cy: {
nativeName: "Cymraeg",
},
"cy-GB": {
nativeName: "Cymraeg",
},
da: {
nativeName: "Dansk",
},
"da-DK": {
nativeName: "Dansk",
},
de: {
nativeName: "Deutsch",
},
"de-AT": {
nativeName: "Deutsch (Österreich)",
},
"de-DE": {
nativeName: "Deutsch (Deutschland)",
},
"de-CH": {
nativeName: "Deutsch (Schweiz)",
},
dsb: {
nativeName: "Dolnoserbšćina",
},
el: {
nativeName: "Ελληνικά",
},
"el-GR": {
nativeName: "Ελληνικά",
},
en: {
nativeName: "English",
},
"en-GB": {
nativeName: "English (UK)",
},
"en-AU": {
nativeName: "English (Australia)",
},
"en-CA": {
nativeName: "English (Canada)",
},
"en-IE": {
nativeName: "English (Ireland)",
},
"en-IN": {
nativeName: "English (India)",
},
"en-PI": {
nativeName: "English (Pirate)",
},
"en-SG": {
nativeName: "English (Singapore)",
},
"en-UD": {
nativeName: "English (Upside Down)",
},
"en-US": {
nativeName: "English (US)",
},
"en-ZA": {
nativeName: "English (South Africa)",
},
"en@pirate": {
nativeName: "English (Pirate)",
},
eo: {
nativeName: "Esperanto",
},
"eo-EO": {
nativeName: "Esperanto",
},
es: {
nativeName: "Español",
},
"es-AR": {
nativeName: "Español (Argentine)",
},
"es-419": {
nativeName: "Español (Latinoamérica)",
},
"es-CL": {
nativeName: "Español (Chile)",
},
"es-CO": {
nativeName: "Español (Colombia)",
},
"es-EC": {
nativeName: "Español (Ecuador)",
},
"es-ES": {
nativeName: "Español (España)",
},
"es-LA": {
nativeName: "Español (Latinoamérica)",
},
"es-NI": {
nativeName: "Español (Nicaragua)",
},
"es-MX": {
nativeName: "Español (México)",
},
"es-US": {
nativeName: "Español (Estados Unidos)",
},
"es-VE": {
nativeName: "Español (Venezuela)",
},
et: {
nativeName: "eesti keel",
},
"et-EE": {
nativeName: "Eesti (Estonia)",
},
eu: {
nativeName: "Euskara",
},
"eu-ES": {
nativeName: "Euskara",
},
fa: {
nativeName: "فارسی",
rtl: true,
},
"fa-IR": {
nativeName: "فارسی",
rtl: true,
},
"fb-LT": {
nativeName: "Leet Speak",
},
ff: {
nativeName: "Fulah",
},
fi: {
nativeName: "Suomi",
},
"fi-FI": {
nativeName: "Suomi",
},
fo: {
nativeName: "Føroyskt",
},
"fo-FO": {
nativeName: "Føroyskt (Færeyjar)",
},
fr: {
nativeName: "Français",
},
"fr-CA": {
nativeName: "Français (Canada)",
},
"fr-FR": {
nativeName: "Français (France)",
},
"fr-BE": {
nativeName: "Français (Belgique)",
},
"fr-CH": {
nativeName: "Français (Suisse)",
},
"fy-NL": {
fy: {
nativeName: "Frysk",
},
ga: {
nativeName: "Gaeilge",
},
"ga-IE": {
nativeName: "Gaeilge",
},
gd: {
nativeName: "Gàidhlig",
},
gl: {
nativeName: "Galego",
},
"gl-ES": {
nativeName: "Galego",
},
"gn-PY": {
gn: {
nativeName: "Avañe'ẽ",
},
"gu-IN": {
gu: {
nativeName: "ગુજરાતી",
},
gv: {
nativeName: "Gaelg",
},
"gx-GR": {
nativeName: "Ἑλληνική ἀρχαία",
},
he: {
nativeName: "עברית‏",
rtl: true,
},
"he-IL": {
nativeName: "עברית‏",
rtl: true,
},
hi: {
nativeName: "हिन्दी",
},
"hi-IN": {
nativeName: "हिन्दी",
},
hr: {
nativeName: "Hrvatski",
},
"hr-HR": {
nativeName: "Hrvatski",
},
hsb: {
nativeName: "Hornjoserbšćina",
},
ht: {
nativeName: "Kreyòl",
},
hu: {
nativeName: "Magyar",
},
"hu-HU": {
nativeName: "Magyar",
},
hy: {
nativeName: "Հայերեն",
},
"hy-AM": {
nativeName: "Հայերեն (Հայաստան)",
},
id: {
nativeName: "Bahasa Indonesia",
},
"id-ID": {
nativeName: "Bahasa Indonesia",
},
is: {
nativeName: "Íslenska",
},
"is-IS": {
nativeName: "Íslenska (Iceland)",
},
it: {
nativeName: "Italiano",
},
"it-IT": {
nativeName: "Italiano",
},
ja: {
nativeName: "日本語",
},
"ja-JP": {
nativeName: "日本語 (日本)",
},
"jv-ID": {
jv: {
nativeName: "Basa Jawa",
},
"ka-GE": {
ka: {
nativeName: "ქართული",
},
"kk-KZ": {
kk: {
nativeName: "Қазақша",
},
km: {
nativeName: "ភាសាខ្មែរ",
},
kl: {
nativeName: "kalaallisut",
},
"km-KH": {
km: {
nativeName: "ភាសាខ្មែរ",
},
kab: {
nativeName: "Taqbaylit",
},
kn: {
nativeName: "ಕನ್ನಡ",
},
"kn-IN": {
nativeName: "ಕನ್ನಡ (India)",
},
ko: {
nativeName: "한국어",
},
"ko-KR": {
nativeName: "한국어 (한국)",
},
"ku-TR": {
ku: {
nativeName: "Kurdî",
},
kw: {
@ -383,66 +161,39 @@ export const langmap = {
la: {
nativeName: "Latin",
},
"la-VA": {
nativeName: "Latin",
},
lb: {
nativeName: "Lëtzebuergesch",
},
"li-NL": {
li: {
nativeName: "Lèmbörgs",
},
lt: {
nativeName: "Lietuvių",
},
"lt-LT": {
nativeName: "Lietuvių",
},
lv: {
nativeName: "Latviešu",
},
"lv-LV": {
nativeName: "Latviešu",
},
mai: {
nativeName: "मैथिली, মৈথিলী",
},
"mg-MG": {
mg: {
nativeName: "Malagasy",
},
mk: {
nativeName: "Македонски",
},
"mk-MK": {
nativeName: "Македонски (Македонски)",
},
ml: {
nativeName: "മലയാളം",
},
"ml-IN": {
nativeName: "മലയാളം",
},
"mn-MN": {
mn: {
nativeName: "Монгол",
},
mr: {
nativeName: "मराठी",
},
"mr-IN": {
nativeName: "मराठी",
},
ms: {
nativeName: "Bahasa Melayu",
},
"ms-MY": {
nativeName: "Bahasa Melayu",
},
mt: {
nativeName: "Malti",
},
"mt-MT": {
nativeName: "Malti",
},
my: {
nativeName: "ဗမာစကာ",
},
@ -452,228 +203,179 @@ export const langmap = {
nb: {
nativeName: "Norsk (bokmål)",
},
"nb-NO": {
nativeName: "Norsk (bokmål)",
},
ne: {
nativeName: "नेपाली",
},
"ne-NP": {
nativeName: "नेपाली",
},
nl: {
nativeName: "Nederlands",
},
"nl-BE": {
nativeName: "Nederlands (België)",
},
"nl-NL": {
nativeName: "Nederlands (Nederland)",
},
"nn-NO": {
nn: {
nativeName: "Norsk (nynorsk)",
},
oc: {
nativeName: "Occitan",
},
"or-IN": {
or: {
nativeName: "ଓଡ଼ିଆ",
},
pa: {
nativeName: "ਪੰਜਾਬੀ",
},
"pa-IN": {
nativeName: "ਪੰਜਾਬੀ (ਭਾਰਤ ਨੂੰ)",
},
pl: {
nativeName: "Polski",
},
"pl-PL": {
nativeName: "Polski",
},
"ps-AF": {
ps: {
nativeName: "پښتو",
rtl: true,
},
pt: {
nativeName: "Português",
},
"pt-BR": {
nativeName: "Português (Brasil)",
},
"pt-PT": {
nativeName: "Português (Portugal)",
},
"qu-PE": {
qu: {
nativeName: "Qhichwa",
},
"rm-CH": {
rm: {
nativeName: "Rumantsch",
},
ro: {
nativeName: "Română",
},
"ro-RO": {
nativeName: "Română",
},
ru: {
nativeName: "Русский",
},
"ru-RU": {
nativeName: "Русский",
},
"sa-IN": {
sa: {
nativeName: "संस्कृतम्",
},
"se-NO": {
se: {
nativeName: "Davvisámegiella",
},
sh: {
nativeName: "српскохрватски",
},
"si-LK": {
si: {
nativeName: "සිංහල",
},
sk: {
nativeName: "Slovenčina",
},
"sk-SK": {
nativeName: "Slovenčina (Slovakia)",
},
sl: {
nativeName: "Slovenščina",
},
"sl-SI": {
nativeName: "Slovenščina",
},
"so-SO": {
so: {
nativeName: "Soomaaliga",
},
sq: {
nativeName: "Shqip",
},
"sq-AL": {
nativeName: "Shqip",
},
sr: {
nativeName: "Српски",
},
"sr-RS": {
nativeName: "Српски (Serbia)",
},
su: {
nativeName: "Basa Sunda",
},
sv: {
nativeName: "Svenska",
},
"sv-SE": {
nativeName: "Svenska",
},
sw: {
nativeName: "Kiswahili",
},
"sw-KE": {
nativeName: "Kiswahili",
},
ta: {
nativeName: "தமிழ்",
},
"ta-IN": {
nativeName: "தமிழ்",
},
te: {
nativeName: "తెలుగు",
},
"te-IN": {
nativeName: "తెలుగు",
},
tg: {
nativeName: "забо́ни тоҷикӣ́",
},
"tg-TJ": {
nativeName: "тоҷикӣ",
},
th: {
nativeName: "ภาษาไทย",
},
"th-TH": {
nativeName: "ภาษาไทย (ประเทศไทย)",
},
fil: {
nativeName: "Filipino",
},
tlh: {
nativeName: "tlhIngan-Hol",
},
tr: {
nativeName: "Türkçe",
},
"tr-TR": {
nativeName: "Türkçe",
},
"tt-RU": {
tt: {
nativeName: "татарча",
},
uk: {
nativeName: "Українська",
},
"uk-UA": {
nativeName: "Українська",
},
ur: {
nativeName: "اردو",
rtl: true,
},
"ur-PK": {
nativeName: "اردو",
rtl: true,
},
uz: {
nativeName: "O'zbek",
},
"uz-UZ": {
nativeName: "O'zbek",
},
vi: {
nativeName: "Tiếng Việt",
},
"vi-VN": {
nativeName: "Tiếng Việt",
},
"xh-ZA": {
xh: {
nativeName: "isiXhosa",
},
yi: {
nativeName: "ייִדיש",
rtl: true,
},
"yi-DE": {
nativeName: "ייִדיש (German)",
rtl: true,
},
zh: {
nativeName: "中文",
},
"zh-Hans": {
nativeName: "中文简体",
},
"zh-Hant": {
nativeName: "中文繁體",
},
"zh-CN": {
nativeName: "中文(中国大陆)",
},
"zh-HK": {
nativeName: "中文(香港)",
},
"zh-SG": {
nativeName: "中文(新加坡)",
},
"zh-TW": {
nativeName: "中文(台灣)",
},
"zu-ZA": {
zu: {
nativeName: "isiZulu",
},
};
export const iso639Langs3 = {
ach: {
nativeName: "Lwo",
},
ady: {
nativeName: "Адыгэбзэ",
},
cak: {
nativeName: "Maya Kaqchikel",
},
chr: {
nativeName: "ᏣᎳᎩ (tsalagi)",
},
dsb: {
nativeName: "Dolnoserbšćina",
},
fil: {
nativeName: "Filipino",
},
hsb: {
nativeName: "Hornjoserbšćina",
},
kab: {
nativeName: "Taqbaylit",
},
mai: {
nativeName: "मैथिली, মৈথিলী",
},
tlh: {
nativeName: "tlhIngan-Hol",
},
tok: {
nativeName: "Toki Pona",
},
yue: {
nativeName: "粵語",
},
nan: {
nativeName: "閩南語",
},
};
export const langmapNoRegion = Object.assign({}, iso639Langs1, iso639Langs3);
export const iso639Regional = {
"zh-hans": {
nativeName: "中文(简体)",
},
"zh-hant": {
nativeName: "中文(繁體)",
},
};
export const langmap = Object.assign({}, langmapNoRegion, iso639Regional);

View file

@ -430,6 +430,10 @@ export const defaultStore = markRaw(
where: "device",
default: 140,
},
recentlyUsedPostLanguages: {
where: "account",
default: [] as string[],
},
}),
);

View file

@ -58,3 +58,96 @@ export const permissions = [
"read:gallery-likes",
"write:gallery-likes",
];
export const languages = [
"ach",
"ady",
"af",
"ak",
"ar",
"az",
"bg",
"bn",
"br",
"ca",
"cak",
"cs",
"cy",
"da",
"de",
"dsb",
"el",
"en",
"eo",
"es",
"et",
"eu",
"fa",
"ff",
"fi",
"fo",
"fr",
"ga",
"gd",
"gl",
"gv",
"he",
"hi",
"hr",
"hsb",
"ht",
"hu",
"hy",
"id",
"is",
"it",
"ja",
"km",
"kl",
"kab",
"kn",
"ko",
"kw",
"la",
"lb",
"lt",
"lv",
"mai",
"mk",
"ml",
"mr",
"ms",
"mt",
"my",
"no",
"nb",
"ne",
"nl",
"oc",
"pa",
"pl",
"pt",
"ro",
"ru",
"sh",
"sk",
"sl",
"sq",
"sr",
"su",
"sv",
"sw",
"ta",
"te",
"tg",
"th",
"fil",
"tlh",
"tr",
"uk",
"ur",
"uz",
"vi",
"yi",
"zh",
] as const;

View file

@ -10,6 +10,7 @@ export const permissions = consts.permissions;
export const notificationTypes = consts.notificationTypes;
export const noteVisibilities = consts.noteVisibilities;
export const mutedNoteReasons = consts.mutedNoteReasons;
export const languages = consts.languages;
export const ffVisibility = consts.ffVisibility;
// api extractor not supported yet