Squashed commit add i18n:

commit 238db51e1fcd22e0b65d3176356ac328481f33ef
Author: fyears <1142836+fyears@users.noreply.github.com>
Date:   Sun Mar 20 15:54:10 2022 +0800

    add langs sub module

commit 30e8cad844821e62a1d6d1e36161c594ddf38111
Author: fyears <1142836+fyears@users.noreply.github.com>
Date:   Sun Mar 20 15:30:45 2022 +0800

    i18n
This commit is contained in:
fyears 2022-03-20 15:54:57 +08:00
parent 50ae224769
commit dffada894f
9 changed files with 499 additions and 364 deletions

3
.gitmodules vendored Normal file
View File

@ -0,0 +1,3 @@
[submodule "src/langs"]
path = src/langs
url = https://github.com/remotely-save/langs.git

View File

@ -31,6 +31,7 @@
"@types/lodash": "^4.14.178", "@types/lodash": "^4.14.178",
"@types/mime-types": "^2.1.1", "@types/mime-types": "^2.1.1",
"@types/mocha": "^9.0.0", "@types/mocha": "^9.0.0",
"@types/mustache": "^4.1.2",
"@types/node": "^14.14.37", "@types/node": "^14.14.37",
"@types/qrcode": "^1.4.1", "@types/qrcode": "^1.4.1",
"builtin-modules": "^3.2.0", "builtin-modules": "^3.2.0",
@ -74,6 +75,7 @@
"lodash": "^4.17.21", "lodash": "^4.17.21",
"loglevel": "^1.8.0", "loglevel": "^1.8.0",
"mime-types": "^2.1.33", "mime-types": "^2.1.33",
"mustache": "^4.2.0",
"nanoid": "^3.1.30", "nanoid": "^3.1.30",
"obsidian": "^0.13.26", "obsidian": "^0.13.26",
"p-queue": "^7.2.0", "p-queue": "^7.2.0",

View File

@ -3,6 +3,8 @@
* To avoid circular dependency. * To avoid circular dependency.
*/ */
import type { LangType, LangTypeAndAuto } from "./i18n";
export type SUPPORTED_SERVICES_TYPE = "s3" | "webdav" | "dropbox" | "onedrive"; export type SUPPORTED_SERVICES_TYPE = "s3" | "webdav" | "dropbox" | "onedrive";
export interface S3Config { export interface S3Config {
@ -70,6 +72,7 @@ export interface RemotelySavePluginSettings {
concurrency?: number; concurrency?: number;
syncConfigDir?: boolean; syncConfigDir?: boolean;
syncUnderscoreItems?: boolean; syncUnderscoreItems?: boolean;
lang?: LangTypeAndAuto;
} }
export interface RemoteItem { export interface RemoteItem {

44
src/i18n.ts Normal file
View File

@ -0,0 +1,44 @@
import Mustache from "mustache";
import { moment } from "obsidian";
import { LANGS } from "./langs";
export type LangType = keyof typeof LANGS;
export type LangTypeAndAuto = LangType | "auto";
export type TransItemType = keyof typeof LANGS["en"];
export class I18n {
lang: LangTypeAndAuto;
readonly saveSettingFunc: (tolang: LangTypeAndAuto) => Promise<void>;
constructor(
lang: LangTypeAndAuto,
saveSettingFunc: (tolang: LangTypeAndAuto) => Promise<void>
) {
this.lang = lang;
this.saveSettingFunc = saveSettingFunc;
}
async changeTo(anotherLang: LangTypeAndAuto) {
this.lang = anotherLang;
await this.saveSettingFunc(anotherLang);
}
_get(key: TransItemType) {
let realLang = this.lang;
if (this.lang === "auto" && moment.locale().replace("-", "_") in LANGS) {
realLang = moment.locale().replace("-", "_") as LangType;
} else {
realLang = "en";
}
const res: string =
(LANGS[realLang] as typeof LANGS["en"])[key] || LANGS["en"][key] || key;
return res;
}
t(key: TransItemType, vars?: Record<string, string>) {
if (vars === undefined) {
return this._get(key);
}
return Mustache.render(this._get(key), vars);
}
}

1
src/langs Submodule

@ -0,0 +1 @@
Subproject commit 340ac099611e81aea813a38e4fac3bb99e270dfc

View File

@ -38,6 +38,8 @@ import { fetchMetadataFile, parseRemoteItems, SyncStatusType } from "./sync";
import { doActualSync, getSyncPlan, isPasswordOk } from "./sync"; import { doActualSync, getSyncPlan, isPasswordOk } from "./sync";
import { messyConfigToNormal, normalConfigToMessy } from "./configPersist"; import { messyConfigToNormal, normalConfigToMessy } from "./configPersist";
import { ObsConfigDirFileType, listFilesInObsFolder } from "./obsFolderLister"; import { ObsConfigDirFileType, listFilesInObsFolder } from "./obsFolderLister";
import { I18n } from "./i18n";
import type { LangType, LangTypeAndAuto, TransItemType } from "./i18n";
import * as origLog from "loglevel"; import * as origLog from "loglevel";
import { DeletionOnRemote, MetadataOnRemote } from "./metadataOnRemote"; import { DeletionOnRemote, MetadataOnRemote } from "./metadataOnRemote";
@ -59,6 +61,7 @@ const DEFAULT_SETTINGS: RemotelySavePluginSettings = {
concurrency: 5, concurrency: 5,
syncConfigDir: false, syncConfigDir: false,
syncUnderscoreItems: false, syncUnderscoreItems: false,
lang: "auto",
}; };
interface OAuth2Info { interface OAuth2Info {
@ -92,8 +95,13 @@ export default class RemotelySavePlugin extends Plugin {
currSyncMsg?: string; currSyncMsg?: string;
syncRibbon?: HTMLElement; syncRibbon?: HTMLElement;
autoRunIntervalID?: number; autoRunIntervalID?: number;
i18n: I18n;
async syncRun(triggerSource: SyncTriggerSourceType = "manual") { async syncRun(triggerSource: SyncTriggerSourceType = "manual") {
const t = (x: TransItemType, vars?: any) => {
return this.i18n.t(x, vars);
};
const getNotice = (x: string) => { const getNotice = (x: string) => {
// only show notices in manual mode // only show notices in manual mode
// no notice in auto mode // no notice in auto mode
@ -103,7 +111,12 @@ export default class RemotelySavePlugin extends Plugin {
}; };
if (this.syncStatus !== "idle") { if (this.syncStatus !== "idle") {
// here the notice is shown regardless of triggerSource // here the notice is shown regardless of triggerSource
new Notice(`Remotely Save already running in stage ${this.syncStatus}!`); new Notice(
t("syncrun_alreadyrunning", {
pluginName: this.manifest.name,
syncStatus: this.syncStatus,
})
);
if (this.currSyncMsg !== undefined && this.currSyncMsg !== "") { if (this.currSyncMsg !== undefined && this.currSyncMsg !== "") {
new Notice(this.currSyncMsg); new Notice(this.currSyncMsg);
} }
@ -126,7 +139,10 @@ export default class RemotelySavePlugin extends Plugin {
setIcon(this.syncRibbon, iconNameSyncRunning); setIcon(this.syncRibbon, iconNameSyncRunning);
this.syncRibbon.setAttribute( this.syncRibbon.setAttribute(
"aria-label", "aria-label",
`${this.manifest.name}: ${triggerSource} syncing` t("syncrun_syncingribbon", {
pluginName: this.manifest.name,
triggerSource: triggerSource,
})
); );
} }
@ -134,17 +150,26 @@ export default class RemotelySavePlugin extends Plugin {
if (triggerSource === "dry") { if (triggerSource === "dry") {
getNotice( getNotice(
`0/${MAX_STEPS} Remotely Save running in dry mode, not actual file changes would happen.` t("syncrun_step0", {
maxSteps: `${MAX_STEPS}`,
})
); );
} }
//log.info(`huh ${this.settings.password}`) //log.info(`huh ${this.settings.password}`)
getNotice( getNotice(
`1/${MAX_STEPS} Remotely Save Sync Preparing (${this.settings.serviceType})` t("syncrun_step1", {
maxSteps: `${MAX_STEPS}`,
serviceType: this.settings.serviceType,
})
); );
this.syncStatus = "preparing"; this.syncStatus = "preparing";
getNotice(`2/${MAX_STEPS} Starting to fetch remote meta data.`); getNotice(
t("syncrun_step2", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "getting_remote_files_list"; this.syncStatus = "getting_remote_files_list";
const self = this; const self = this;
const client = new RemoteClient( const client = new RemoteClient(
@ -159,18 +184,26 @@ export default class RemotelySavePlugin extends Plugin {
const remoteRsp = await client.listFromRemote(); const remoteRsp = await client.listFromRemote();
log.debug(remoteRsp); log.debug(remoteRsp);
getNotice(`3/${MAX_STEPS} Checking password correct or not.`); getNotice(
t("syncrun_step3", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "checking_password"; this.syncStatus = "checking_password";
const passwordCheckResult = await isPasswordOk( const passwordCheckResult = await isPasswordOk(
remoteRsp.Contents, remoteRsp.Contents,
this.settings.password this.settings.password
); );
if (!passwordCheckResult.ok) { if (!passwordCheckResult.ok) {
getNotice("something goes wrong while checking password"); getNotice(t("syncrun_passworderr"));
throw Error(passwordCheckResult.reason); throw Error(passwordCheckResult.reason);
} }
getNotice(`4/${MAX_STEPS} Trying to fetch extra meta data from remote.`); getNotice(
t("syncrun_step4", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "getting_remote_extra_meta"; this.syncStatus = "getting_remote_extra_meta";
const { remoteStates, metadataFile } = await parseRemoteItems( const { remoteStates, metadataFile } = await parseRemoteItems(
remoteRsp.Contents, remoteRsp.Contents,
@ -186,7 +219,11 @@ export default class RemotelySavePlugin extends Plugin {
this.settings.password this.settings.password
); );
getNotice(`5/${MAX_STEPS} Starting to fetch local meta data.`); getNotice(
t("syncrun_step5", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "getting_local_meta"; this.syncStatus = "getting_local_meta";
const local = this.app.vault.getAllLoadedFiles(); const local = this.app.vault.getAllLoadedFiles();
const localHistory = await loadDeleteRenameHistoryTableByVault( const localHistory = await loadDeleteRenameHistoryTableByVault(
@ -204,7 +241,11 @@ export default class RemotelySavePlugin extends Plugin {
// log.info(local); // log.info(local);
// log.info(localHistory); // log.info(localHistory);
getNotice(`6/${MAX_STEPS} Starting to generate sync plan.`); getNotice(
t("syncrun_step6", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "generating_plan"; this.syncStatus = "generating_plan";
const { plan, sortedKeys, deletions } = await getSyncPlan( const { plan, sortedKeys, deletions } = await getSyncPlan(
remoteStates, remoteStates,
@ -232,7 +273,11 @@ export default class RemotelySavePlugin extends Plugin {
// The operations below begins to write or delete (!!!) something. // The operations below begins to write or delete (!!!) something.
if (triggerSource !== "dry") { if (triggerSource !== "dry") {
getNotice(`7/${MAX_STEPS} Remotely Save Sync data exchanging!`); getNotice(
t("syncrun_step7", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "syncing"; this.syncStatus = "syncing";
await doActualSync( await doActualSync(
@ -254,11 +299,17 @@ export default class RemotelySavePlugin extends Plugin {
} else { } else {
this.syncStatus = "syncing"; this.syncStatus = "syncing";
getNotice( getNotice(
`7/${MAX_STEPS} Remotely Save real sync is skipped in dry run mode.` t("syncrun_step7skip", {
maxSteps: `${MAX_STEPS}`,
})
); );
} }
getNotice(`8/${MAX_STEPS} Remotely Save finish!`); getNotice(
t("syncrun_step8", {
maxSteps: `${MAX_STEPS}`,
})
);
this.syncStatus = "finish"; this.syncStatus = "finish";
this.syncStatus = "idle"; this.syncStatus = "idle";
@ -273,11 +324,12 @@ export default class RemotelySavePlugin extends Plugin {
}-${Date.now()}: finish sync, triggerSource=${triggerSource}` }-${Date.now()}: finish sync, triggerSource=${triggerSource}`
); );
} catch (error) { } catch (error) {
const msg = `${ const msg = t("syncrun_abort", {
this.manifest.id manifestID: this.manifest.id,
}-${Date.now()}: abort sync, triggerSource=${triggerSource}, error while ${ theDate: `${Date.now()}`,
this.syncStatus triggerSource: triggerSource,
}`; syncStatus: this.syncStatus,
});
log.info(msg); log.info(msg);
log.info(error); log.info(error);
getNotice(msg); getNotice(msg);
@ -308,6 +360,15 @@ export default class RemotelySavePlugin extends Plugin {
await this.loadSettings(); await this.loadSettings();
// lang should be load early, but after settings
this.i18n = new I18n(this.settings.lang, async (lang: LangTypeAndAuto) => {
this.settings.lang = lang;
await this.saveSettings();
});
const t = (x: TransItemType, vars?: any) => {
return this.i18n.t(x, vars);
};
if (this.settings.currLogLevel !== undefined) { if (this.settings.currLogLevel !== undefined) {
log.setLevel(this.settings.currLogLevel as any); log.setLevel(this.settings.currLogLevel as any);
} }
@ -353,7 +414,9 @@ export default class RemotelySavePlugin extends Plugin {
this.settings = Object.assign({}, this.settings, copied); this.settings = Object.assign({}, this.settings, copied);
this.saveSettings(); this.saveSettings();
new Notice( new Notice(
`New not-oauth2 settings for ${this.manifest.name} saved. Reopen the plugin Settings to the effect.` t("protocol_saveqr", {
manifestName: this.manifest.name,
})
); );
} }
}); });
@ -362,9 +425,9 @@ export default class RemotelySavePlugin extends Plugin {
COMMAND_CALLBACK, COMMAND_CALLBACK,
async (inputParams) => { async (inputParams) => {
new Notice( new Notice(
`Your uri call a callback that's not supported yet: ${JSON.stringify( t("protocol_callbacknotsupported", {
inputParams params: JSON.stringify(inputParams),
)}` })
); );
} }
); );
@ -375,12 +438,14 @@ export default class RemotelySavePlugin extends Plugin {
if (inputParams.code !== undefined) { if (inputParams.code !== undefined) {
if (this.oauth2Info.helperModal !== undefined) { if (this.oauth2Info.helperModal !== undefined) {
this.oauth2Info.helperModal.contentEl.empty(); this.oauth2Info.helperModal.contentEl.empty();
this.oauth2Info.helperModal.contentEl.createEl("p", {
text: "Connecting to Dropbox...", t("protocol_dropbox_connecting")
}); .split("\n")
this.oauth2Info.helperModal.contentEl.createEl("p", { .forEach((val) => {
text: "Please DO NOT close this modal.", this.oauth2Info.helperModal.contentEl.createEl("p", {
}); text: val,
});
});
} }
let authRes = await sendAuthReqDropbox( let authRes = await sendAuthReqDropbox(
@ -410,7 +475,11 @@ export default class RemotelySavePlugin extends Plugin {
this.settings.dropbox.username = username; this.settings.dropbox.username = username;
await this.saveSettings(); await this.saveSettings();
new Notice(`Good! We've connected to Dropbox as user ${username}!`); new Notice(
t("protocol_dropbox_connect_succ", {
username: username,
})
);
this.oauth2Info.verifier = ""; // reset it this.oauth2Info.verifier = ""; // reset it
this.oauth2Info.helperModal?.close(); // close it this.oauth2Info.helperModal?.close(); // close it
@ -423,7 +492,9 @@ export default class RemotelySavePlugin extends Plugin {
this.oauth2Info.authDiv = undefined; this.oauth2Info.authDiv = undefined;
this.oauth2Info.revokeAuthSetting?.setDesc( this.oauth2Info.revokeAuthSetting?.setDesc(
`You've connected as user ${this.settings.dropbox.username}. If you want to disconnect, click this button.` t("protocol_dropbox_connect_succ_revoke", {
username: this.settings.dropbox.username,
})
); );
this.oauth2Info.revokeAuthSetting = undefined; this.oauth2Info.revokeAuthSetting = undefined;
this.oauth2Info.revokeDiv?.toggleClass( this.oauth2Info.revokeDiv?.toggleClass(
@ -432,13 +503,11 @@ export default class RemotelySavePlugin extends Plugin {
); );
this.oauth2Info.revokeDiv = undefined; this.oauth2Info.revokeDiv = undefined;
} else { } else {
new Notice( new Notice(t("protocol_dropbox_connect_fail"));
"Something went wrong from response from Dropbox. Maybe you rejected the auth?"
);
throw Error( throw Error(
`do not know how to deal with the callback: ${JSON.stringify( t("protocol_dropbox_connect_unknown", {
inputParams params: JSON.stringify(inputParams),
)}` })
); );
} }
} }
@ -450,12 +519,14 @@ export default class RemotelySavePlugin extends Plugin {
if (inputParams.code !== undefined) { if (inputParams.code !== undefined) {
if (this.oauth2Info.helperModal !== undefined) { if (this.oauth2Info.helperModal !== undefined) {
this.oauth2Info.helperModal.contentEl.empty(); this.oauth2Info.helperModal.contentEl.empty();
this.oauth2Info.helperModal.contentEl.createEl("p", {
text: "Connecting to Onedrive...", t("protocol_onedrive_connecting")
}); .split("\n")
this.oauth2Info.helperModal.contentEl.createEl("p", { .forEach((val) => {
text: "Please DO NOT close this modal.", this.oauth2Info.helperModal.contentEl.createEl("p", {
}); text: val,
});
});
} }
let rsp = await sendAuthReqOnedrive( let rsp = await sendAuthReqOnedrive(
@ -499,7 +570,9 @@ export default class RemotelySavePlugin extends Plugin {
this.oauth2Info.authDiv = undefined; this.oauth2Info.authDiv = undefined;
this.oauth2Info.revokeAuthSetting?.setDesc( this.oauth2Info.revokeAuthSetting?.setDesc(
`You've connected as user ${this.settings.onedrive.username}. If you want to disconnect, click this button.` t("protocol_onedrive_connect_succ_revoke", {
username: this.settings.onedrive.username,
})
); );
this.oauth2Info.revokeAuthSetting = undefined; this.oauth2Info.revokeAuthSetting = undefined;
this.oauth2Info.revokeDiv?.toggleClass( this.oauth2Info.revokeDiv?.toggleClass(
@ -508,13 +581,11 @@ export default class RemotelySavePlugin extends Plugin {
); );
this.oauth2Info.revokeDiv = undefined; this.oauth2Info.revokeDiv = undefined;
} else { } else {
new Notice( new Notice(t("protocol_onedrive_connect_fail"));
"Something went wrong from response from OneDrive. Maybe you rejected the auth?"
);
throw Error( throw Error(
`do not know how to deal with the callback: ${JSON.stringify( t("protocol_onedrive_connect_unknown", {
inputParams params: JSON.stringify(inputParams),
)}` })
); );
} }
} }
@ -528,7 +599,7 @@ export default class RemotelySavePlugin extends Plugin {
this.addCommand({ this.addCommand({
id: "start-sync", id: "start-sync",
name: "start sync", name: t("command_startsync"),
icon: iconNameSyncWait, icon: iconNameSyncWait,
callback: async () => { callback: async () => {
this.syncRun("manual"); this.syncRun("manual");
@ -537,7 +608,7 @@ export default class RemotelySavePlugin extends Plugin {
this.addCommand({ this.addCommand({
id: "start-sync-dry-run", id: "start-sync-dry-run",
name: "start sync (dry run only)", name: t("command_drynrun"),
icon: iconNameSyncWait, icon: iconNameSyncWait,
callback: async () => { callback: async () => {
this.syncRun("dry"); this.syncRun("dry");

File diff suppressed because it is too large Load Diff

View File

@ -1,5 +1,6 @@
import { App, Modal, Notice, PluginSettingTab, Setting } from "obsidian"; import { App, Modal, Notice, PluginSettingTab, Setting } from "obsidian";
import type RemotelySavePlugin from "./main"; // unavoidable import type RemotelySavePlugin from "./main"; // unavoidable
import type { TransItemType } from "./i18n";
import * as origLog from "loglevel"; import * as origLog from "loglevel";
const log = origLog.getLogger("rs-default"); const log = origLog.getLogger("rs-default");
@ -13,40 +14,33 @@ export class SyncAlgoV2Modal extends Modal {
} }
onOpen() { onOpen() {
let { contentEl } = this; let { contentEl } = this;
const t = (x: TransItemType, vars?: any) => {
return this.plugin.i18n.t(x, vars);
};
contentEl.createEl("h2", { contentEl.createEl("h2", {
text: "Remotely Save has a better sync algorithm", text: t("syncalgov2_title"),
}); });
const texts = [
"Welcome to use Remotely Save!",
"From version 0.3.0, a new algorithm has been developed, but it needs uploading extra meta data files _remotely-save-metadata-on-remote.{json,bin} to YOUR configured cloud destinations, besides your notes.",
"So that, for example, the second device can know that what files/folders have been deleted on the first device by reading those files.",
'If you agree, plase click the button "agree", and enjoy the plugin! AND PLEASE REMEMBER TO BACKUP YOUR VAULT FIRSTLY!',
'If you do not agree, you should stop using the current and later versions of Remotely Save. You could consider manually install the old version 0.2.14 which uses old algorithm and does not upload any extra meta data files. By clicking the "Do not agree" button, the plugin will unload itself, and you need to manually disable it in Obsidian settings.',
];
const ul = contentEl.createEl("ul"); const ul = contentEl.createEl("ul");
t("syncalgov2_texts")
for (const t of texts) { .split("\n")
ul.createEl("li", { .forEach((val) => {
text: t, ul.createEl("li", {
text: val,
});
}); });
}
new Setting(contentEl) new Setting(contentEl)
.addButton((button) => { .addButton((button) => {
button.setButtonText("Agree"); button.setButtonText(t("syncalgov2_button_agree"));
button.onClick(async () => { button.onClick(async () => {
this.agree = true; this.agree = true;
this.close(); this.close();
}); });
}) })
.addButton((button) => { .addButton((button) => {
button.setButtonText("Do not agree"); button.setButtonText(t("syncalgov2_button_disagree"));
button.onClick(() => { button.onClick(() => {
this.close(); this.close();
}); });

View File

@ -9,6 +9,7 @@
"noImplicitAny": true, "noImplicitAny": true,
"moduleResolution": "node", "moduleResolution": "node",
// "allowSyntheticDefaultImports": true, // "allowSyntheticDefaultImports": true,
"resolveJsonModule": true,
"esModuleInterop": true, "esModuleInterop": true,
"importHelpers": true, "importHelpers": true,
"isolatedModules": true, "isolatedModules": true,