mirror of
https://github.com/TriliumNext/Notes.git
synced 2025-09-01 12:27:41 +08:00
168 lines
5.6 KiB
TypeScript
168 lines
5.6 KiB
TypeScript
import { Dropdown } from "bootstrap";
|
|
import NoteContextAwareWidget from "./note_context_aware_widget.js";
|
|
import { getAvailableLocales, getLocaleById } from "../services/i18n.js";
|
|
import { t } from "i18next";
|
|
import type { EventData } from "../components/app_context.js";
|
|
import type FNote from "../entities/fnote.js";
|
|
import attributes from "../services/attributes.js";
|
|
import type { Locale } from "../../../services/i18n.js";
|
|
import options from "../services/options.js";
|
|
import appContext from "../components/app_context.js";
|
|
|
|
const TPL = `\
|
|
<div class="dropdown note-language-widget">
|
|
<button type="button" data-bs-toggle="dropdown" aria-haspopup="true" aria-expanded="false" class="btn btn-sm dropdown-toggle select-button note-language-button">
|
|
<span class="note-language-desc"></span>
|
|
<span class="caret"></span>
|
|
</button>
|
|
<div class="note-language-dropdown dropdown-menu dropdown-menu-left tn-dropdown-list"></div>
|
|
<button class="language-help-button icon-action bx bx-help-circle" type="button" data-in-app-help="B0lcI9xz1r8K" title="${t("open-help-page")}"></button>
|
|
|
|
<style>
|
|
.note-language-widget {
|
|
display: flex;
|
|
align-items: center;
|
|
}
|
|
|
|
.language-help-button {
|
|
margin-left: 4px;
|
|
}
|
|
|
|
.note-language-dropdown [dir=rtl] {
|
|
text-align: right;
|
|
}
|
|
|
|
.dropdown-item.rtl > .check {
|
|
order: 1;
|
|
}
|
|
</style>
|
|
</div>
|
|
`;
|
|
|
|
const DEFAULT_LOCALE: Locale = {
|
|
id: "",
|
|
name: t("note_language.not_set")
|
|
};
|
|
|
|
export default class NoteLanguageWidget extends NoteContextAwareWidget {
|
|
|
|
private dropdown!: Dropdown;
|
|
private $noteLanguageDropdown!: JQuery<HTMLElement>;
|
|
private $noteLanguageDesc!: JQuery<HTMLElement>;
|
|
private locales: (Locale | "---")[];
|
|
private currentLanguageId?: string;
|
|
|
|
constructor() {
|
|
super();
|
|
this.locales = NoteLanguageWidget.#buildLocales();
|
|
}
|
|
|
|
doRender() {
|
|
this.$widget = $(TPL);
|
|
this.dropdown = Dropdown.getOrCreateInstance(this.$widget.find("[data-bs-toggle='dropdown']")[0]);
|
|
this.$widget.on("show.bs.dropdown", () => this.renderDropdown());
|
|
|
|
this.$noteLanguageDropdown = this.$widget.find(".note-language-dropdown")
|
|
this.$noteLanguageDesc = this.$widget.find(".note-language-desc");
|
|
}
|
|
|
|
renderDropdown() {
|
|
this.$noteLanguageDropdown.empty();
|
|
|
|
if (!this.note) {
|
|
return;
|
|
}
|
|
|
|
for (const locale of this.locales) {
|
|
if (typeof locale === "object") {
|
|
const $title = $("<span>").text(locale.name);
|
|
|
|
const $link = $('<a class="dropdown-item">')
|
|
.attr("data-language", locale.id)
|
|
.append('<span class="check">✓</span> ')
|
|
.append($title)
|
|
.on("click", () => {
|
|
const languageId = $link.attr("data-language") ?? "";
|
|
this.save(languageId);
|
|
});
|
|
|
|
if (locale.rtl) {
|
|
$link.attr("dir", "rtl");
|
|
}
|
|
|
|
if (locale.id === this.currentLanguageId) {
|
|
$link.addClass("selected");
|
|
}
|
|
|
|
this.$noteLanguageDropdown.append($link);
|
|
} else {
|
|
this.$noteLanguageDropdown.append('<div class="dropdown-divider"></div>');
|
|
}
|
|
}
|
|
|
|
const $configureLink = $('<a class="dropdown-item">')
|
|
.append(`<span>${t("note_language.configure-languages")}</span>`)
|
|
.on("click", () => appContext.tabManager.openContextWithNote("_optionsLocalization", { activate: true }));
|
|
this.$noteLanguageDropdown.append($configureLink);
|
|
}
|
|
|
|
async save(languageId: string) {
|
|
if (!this.note) {
|
|
return;
|
|
}
|
|
|
|
attributes.setAttribute(this.note, "label", "language", languageId);
|
|
}
|
|
|
|
async refreshWithNote(note: FNote) {
|
|
const currentLanguageId = note.getLabelValue("language") ?? "";
|
|
const language = getLocaleById(currentLanguageId) ?? DEFAULT_LOCALE;
|
|
this.currentLanguageId = currentLanguageId;
|
|
this.$noteLanguageDesc.text(language.name);
|
|
this.dropdown.hide();
|
|
}
|
|
|
|
async entitiesReloadedEvent({ loadResults }: EventData<"entitiesReloaded">) {
|
|
if (loadResults.isOptionReloaded("languages")) {
|
|
this.locales = NoteLanguageWidget.#buildLocales();
|
|
}
|
|
|
|
if (loadResults.getAttributeRows().find((a) => a.noteId === this.noteId && a.name === "language")) {
|
|
this.refresh();
|
|
}
|
|
}
|
|
|
|
static #buildLocales() {
|
|
const enabledLanguages = JSON.parse(options.get("languages") ?? "[]") as string[];
|
|
const filteredLanguages = getAvailableLocales().filter((l) => typeof l !== "object" || enabledLanguages.includes(l.id));
|
|
const leftToRightLanguages = filteredLanguages.filter((l) => !l.rtl);
|
|
const rightToLeftLanguages = filteredLanguages.filter((l) => l.rtl);
|
|
|
|
let locales: ("---" | Locale)[] = [
|
|
DEFAULT_LOCALE
|
|
];
|
|
|
|
if (leftToRightLanguages.length > 0) {
|
|
locales = [
|
|
...locales,
|
|
"---",
|
|
...leftToRightLanguages
|
|
];
|
|
}
|
|
|
|
if (rightToLeftLanguages.length > 0) {
|
|
locales = [
|
|
...locales,
|
|
"---",
|
|
...rightToLeftLanguages
|
|
];
|
|
}
|
|
|
|
// This will separate the list of languages from the "Configure languages" button.
|
|
// If there is at least one language.
|
|
locales.push("---");
|
|
return locales;
|
|
}
|
|
|
|
}
|