mirror of
https://github.com/outline/outline.git
synced 2026-06-27 18:24:23 +03:00
1285efc49a
* feat: i18n
* Changing language single source of truth from TEAM to USER
* Changes according to @tommoor comments on PR
* Changed package.json for build:i18n and translation label
* Finished 1st MVP of i18n for outline
* new translation labels & Portuguese from Portugal translation
* Fixes from PR request
* Described language dropdown as an experimental feature
* Set keySeparator to false in order to cowork with html keys
* Added useTranslation to Breadcrumb
* Repositioned <strong> element
* Removed extra space from TemplatesMenu
* Fortified the test suite for i18n
* Fixed trans component problematic
* Check if selected language is available
* Update yarn.lock
* Removed unused Trans
* Removing debug variable from i18n init
* Removed debug variable
* test: update snapshots
* flow: Remove decorator usage to get proper flow typing
It's a shame, but hopefully we'll move to Typescript in the next 6 months and we can forget this whole Flow mistake ever happened
* translate: Drafts
* More translatable strings
* Mo translation strings
* translation: Search
* async translations loading
* cache translations in client
* Revert "cache translations in client"
This reverts commit 08fb61ce36.
* Revert localStorage cache for cache headers
* Update Crowdin configuration file
* Moved translation files to locales folder and fixed english text
* Added CONTRIBUTING File for CrowdIn
* chore: Move translations again to please CrowdIn
* fix: loading paths
chore: Add strings for editor
* fix: Improve validation on documents.import endpoint
* test: mock bull
* fix: Unknown mimetype should fallback to Markdown parsing if markdown extension (#1678)
* closes #1675
* Update CONTRIBUTING
* chore: Add link to translation portal from app UI
* refactor: Centralize language config
* fix: Ensure creation of i18n directory in build
* feat: Add language prompt
* chore: Improve contributing guidelines, add link from README
* chore: Normalize tab header casing
* chore: More string externalization
* fix: Language prompt in dark mode
Co-authored-by: André Glatzl <andreglatzl@gmail.com>
200 lines
4.4 KiB
JavaScript
200 lines
4.4 KiB
JavaScript
// @flow
|
|
import { orderBy } from "lodash";
|
|
import { observable, action, autorun, computed } from "mobx";
|
|
import { v4 } from "uuid";
|
|
import Collection from "models/Collection";
|
|
import Document from "models/Document";
|
|
import type { Toast } from "types";
|
|
|
|
const UI_STORE = "UI_STORE";
|
|
|
|
class UiStore {
|
|
// has the user seen the prompt to change the UI language and actioned it
|
|
@observable languagePromptDismissed: boolean;
|
|
|
|
// theme represents the users UI preference (defaults to system)
|
|
@observable theme: "light" | "dark" | "system";
|
|
|
|
// systemTheme represents the system UI theme (Settings -> General in macOS)
|
|
@observable systemTheme: "light" | "dark";
|
|
@observable activeDocumentId: ?string;
|
|
@observable activeCollectionId: ?string;
|
|
@observable progressBarVisible: boolean = false;
|
|
@observable editMode: boolean = false;
|
|
@observable tocVisible: boolean = false;
|
|
@observable mobileSidebarVisible: boolean = false;
|
|
@observable toasts: Map<string, Toast> = new Map();
|
|
|
|
constructor() {
|
|
// Rehydrate
|
|
let data = {};
|
|
try {
|
|
data = JSON.parse(localStorage.getItem(UI_STORE) || "{}");
|
|
} catch (_) {
|
|
// no-op Safari private mode
|
|
}
|
|
|
|
// system theme listeners
|
|
if (window.matchMedia) {
|
|
const colorSchemeQueryList = window.matchMedia(
|
|
"(prefers-color-scheme: dark)"
|
|
);
|
|
|
|
const setSystemTheme = (event) => {
|
|
this.systemTheme = event.matches ? "dark" : "light";
|
|
};
|
|
setSystemTheme(colorSchemeQueryList);
|
|
if (colorSchemeQueryList.addListener) {
|
|
colorSchemeQueryList.addListener(setSystemTheme);
|
|
}
|
|
}
|
|
|
|
// persisted keys
|
|
this.languagePromptDismissed = data.languagePromptDismissed;
|
|
this.tocVisible = data.tocVisible;
|
|
this.theme = data.theme || "system";
|
|
|
|
autorun(() => {
|
|
try {
|
|
localStorage.setItem(UI_STORE, this.asJson);
|
|
} catch (_) {
|
|
// no-op Safari private mode
|
|
}
|
|
});
|
|
}
|
|
|
|
@action
|
|
setTheme = (theme: "light" | "dark" | "system") => {
|
|
this.theme = theme;
|
|
|
|
if (window.localStorage) {
|
|
window.localStorage.setItem("theme", this.theme);
|
|
}
|
|
};
|
|
|
|
@action
|
|
setLanguagePromptDismissed = () => {
|
|
this.languagePromptDismissed = true;
|
|
};
|
|
|
|
@action
|
|
setActiveDocument = (document: Document): void => {
|
|
this.activeDocumentId = document.id;
|
|
|
|
if (
|
|
document.publishedAt &&
|
|
!document.isArchived &&
|
|
!document.isDeleted &&
|
|
!document.isTemplate
|
|
) {
|
|
this.activeCollectionId = document.collectionId;
|
|
}
|
|
};
|
|
|
|
@action
|
|
setActiveCollection = (collection: Collection): void => {
|
|
this.activeCollectionId = collection.id;
|
|
};
|
|
|
|
@action
|
|
clearActiveCollection = (): void => {
|
|
this.activeCollectionId = undefined;
|
|
};
|
|
|
|
@action
|
|
clearActiveDocument = (): void => {
|
|
this.activeDocumentId = undefined;
|
|
this.activeCollectionId = undefined;
|
|
};
|
|
|
|
@action
|
|
showTableOfContents = () => {
|
|
this.tocVisible = true;
|
|
};
|
|
|
|
@action
|
|
hideTableOfContents = () => {
|
|
this.tocVisible = false;
|
|
};
|
|
|
|
@action
|
|
enableEditMode = () => {
|
|
this.editMode = true;
|
|
};
|
|
|
|
@action
|
|
disableEditMode = () => {
|
|
this.editMode = false;
|
|
};
|
|
|
|
@action
|
|
enableProgressBar = () => {
|
|
this.progressBarVisible = true;
|
|
};
|
|
|
|
@action
|
|
disableProgressBar = () => {
|
|
this.progressBarVisible = false;
|
|
};
|
|
|
|
@action
|
|
toggleMobileSidebar = () => {
|
|
this.mobileSidebarVisible = !this.mobileSidebarVisible;
|
|
};
|
|
|
|
@action
|
|
hideMobileSidebar = () => {
|
|
this.mobileSidebarVisible = false;
|
|
};
|
|
|
|
@action
|
|
showToast = (
|
|
message: string,
|
|
options?: {
|
|
type?: "warning" | "error" | "info" | "success",
|
|
timeout?: number,
|
|
action?: {
|
|
text: string,
|
|
onClick: () => void,
|
|
},
|
|
}
|
|
) => {
|
|
if (!message) return;
|
|
|
|
const id = v4();
|
|
const createdAt = new Date().toISOString();
|
|
this.toasts.set(id, { message, createdAt, id, ...options });
|
|
return id;
|
|
};
|
|
|
|
@action
|
|
removeToast = (id: string) => {
|
|
this.toasts.delete(id);
|
|
};
|
|
|
|
@computed
|
|
get resolvedTheme(): "dark" | "light" {
|
|
if (this.theme === "system") {
|
|
return this.systemTheme;
|
|
}
|
|
|
|
return this.theme;
|
|
}
|
|
|
|
@computed
|
|
get orderedToasts(): Toast[] {
|
|
return orderBy(Array.from(this.toasts.values()), "createdAt", "desc");
|
|
}
|
|
|
|
@computed
|
|
get asJson(): string {
|
|
return JSON.stringify({
|
|
tocVisible: this.tocVisible,
|
|
languagePromptDismissed: this.languagePromptDismissed,
|
|
theme: this.theme,
|
|
});
|
|
}
|
|
}
|
|
|
|
export default UiStore;
|