2020-10-02 15:10:00 +00:00
|
|
|
/**
|
|
|
|
* This module export a component for editing some document settings consisting of the timezone,
|
|
|
|
* (new settings to be added here ...).
|
|
|
|
*/
|
2022-10-28 16:11:08 +00:00
|
|
|
import {makeT} from 'app/client/lib/localization';
|
2021-08-26 16:35:11 +00:00
|
|
|
import {dom, IDisposableOwner, styled} from 'grainjs';
|
|
|
|
import {Computed, Observable} from 'grainjs';
|
2020-10-02 15:10:00 +00:00
|
|
|
|
2021-08-26 16:35:11 +00:00
|
|
|
|
|
|
|
import {ACSelectItem, buildACSelect} from "app/client/lib/ACSelect";
|
|
|
|
import {ACIndexImpl} from "app/client/lib/ACIndex";
|
|
|
|
import {loadMomentTimezone} from 'app/client/lib/imports';
|
|
|
|
import {DocInfoRec} from 'app/client/models/DocModel';
|
|
|
|
import {DocPageModel} from 'app/client/models/DocPageModel';
|
|
|
|
import {testId, vars} from 'app/client/ui2018/cssVars';
|
2021-09-15 23:35:21 +00:00
|
|
|
import {select} from 'app/client/ui2018/menus';
|
2021-08-26 16:35:11 +00:00
|
|
|
import {saveModal} from 'app/client/ui2018/modals';
|
2021-09-23 18:09:58 +00:00
|
|
|
import {buildCurrencyPicker} from 'app/client/widgets/CurrencyPicker';
|
2021-08-26 16:35:11 +00:00
|
|
|
import {buildTZAutocomplete} from 'app/client/widgets/TZAutocomplete';
|
2021-09-15 23:35:21 +00:00
|
|
|
import {EngineCode} from 'app/common/DocumentSettings';
|
|
|
|
import {GristLoadConfig} from 'app/common/gristUrls';
|
2021-09-23 18:09:58 +00:00
|
|
|
import {propertyCompare} from "app/common/gutil";
|
|
|
|
import {getCurrency, locales} from "app/common/Locales";
|
2022-10-28 16:11:08 +00:00
|
|
|
|
|
|
|
const t = makeT('DocumentSettings');
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
/**
|
|
|
|
* Builds a simple saveModal for saving settings.
|
|
|
|
*/
|
|
|
|
export async function showDocSettingsModal(docInfo: DocInfoRec, docPageModel: DocPageModel): Promise<void> {
|
|
|
|
const moment = await loadMomentTimezone();
|
|
|
|
return saveModal((ctl, owner) => {
|
2021-08-26 16:35:11 +00:00
|
|
|
const timezoneObs = Observable.create(owner, docInfo.timezone.peek());
|
|
|
|
|
2021-09-15 23:35:21 +00:00
|
|
|
const docSettings = docInfo.documentSettingsJson.peek();
|
|
|
|
const {locale, currency, engine} = docSettings;
|
2021-08-26 16:35:11 +00:00
|
|
|
const localeObs = Observable.create(owner, locale);
|
|
|
|
const currencyObs = Observable.create(owner, currency);
|
2021-09-15 23:35:21 +00:00
|
|
|
const engineObs = Observable.create(owner, engine);
|
|
|
|
|
|
|
|
// Check if server supports engine choices - if so, we will allow user to pick.
|
|
|
|
const canChangeEngine = getSupportedEngineChoices().length > 0;
|
2021-08-26 16:35:11 +00:00
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
return {
|
2022-10-28 16:11:08 +00:00
|
|
|
title: t('DocumentSettings'),
|
2020-10-02 15:10:00 +00:00
|
|
|
body: [
|
2022-10-28 16:11:08 +00:00
|
|
|
cssDataRow(t('ThisDocumentID')),
|
2020-10-02 15:10:00 +00:00
|
|
|
cssDataRow(dom('tt', docPageModel.currentDocId.get())),
|
2022-10-28 16:11:08 +00:00
|
|
|
cssDataRow(t('TimeZone')),
|
2021-08-26 16:35:11 +00:00
|
|
|
cssDataRow(dom.create(buildTZAutocomplete, moment, timezoneObs, (val) => timezoneObs.set(val))),
|
2022-10-28 16:11:08 +00:00
|
|
|
cssDataRow(t('Locale')),
|
2021-08-26 16:35:11 +00:00
|
|
|
cssDataRow(dom.create(buildLocaleSelect, localeObs)),
|
2022-10-28 16:11:08 +00:00
|
|
|
cssDataRow(t('Currency')),
|
2021-08-26 16:35:11 +00:00
|
|
|
cssDataRow(dom.domComputed(localeObs, (l) =>
|
|
|
|
dom.create(buildCurrencyPicker, currencyObs, (val) => currencyObs.set(val),
|
2022-10-28 16:11:08 +00:00
|
|
|
{defaultCurrencyLabel: t('LocalCurrency', {currency: getCurrency(l)})})
|
2021-08-26 16:35:11 +00:00
|
|
|
)),
|
2021-09-15 23:35:21 +00:00
|
|
|
canChangeEngine ? [
|
2022-03-24 20:27:34 +00:00
|
|
|
// Small easter egg: you can click on the skull-and-crossbones to
|
|
|
|
// force a reload of the document.
|
2022-10-28 16:11:08 +00:00
|
|
|
cssDataRow(t('EngineRisk', {span:
|
|
|
|
dom('span', '☠',
|
|
|
|
dom.style('cursor', 'pointer'),
|
|
|
|
dom.on('click', async () => {
|
|
|
|
await docPageModel.appModel.api.getDocAPI(docPageModel.currentDocId.get()!).forceReload();
|
|
|
|
document.location.reload();
|
|
|
|
}))
|
|
|
|
})),
|
2021-09-15 23:35:21 +00:00
|
|
|
select(engineObs, getSupportedEngineChoices()),
|
|
|
|
] : null,
|
2020-10-02 15:10:00 +00:00
|
|
|
],
|
2021-09-15 23:35:21 +00:00
|
|
|
// Modal label is "Save", unless engine is changed. If engine is changed, the document will
|
|
|
|
// need a reload to switch engines, so we replace the label with "Save and Reload".
|
2022-10-28 16:11:08 +00:00
|
|
|
saveLabel: dom.text((use) => (use(engineObs) === docSettings.engine) ? t('Save') : t('SaveAndReload')),
|
2021-09-15 23:35:21 +00:00
|
|
|
saveFunc: async () => {
|
|
|
|
await docInfo.updateColValues({
|
|
|
|
timezone: timezoneObs.get(),
|
|
|
|
documentSettings: JSON.stringify({
|
|
|
|
...docInfo.documentSettingsJson.peek(),
|
|
|
|
locale: localeObs.get(),
|
|
|
|
currency: currencyObs.get(),
|
|
|
|
engine: engineObs.get()
|
|
|
|
})
|
|
|
|
});
|
|
|
|
// Reload the document if the engine is changed.
|
|
|
|
if (engineObs.get() !== docSettings.engine) {
|
|
|
|
await docPageModel.appModel.api.getDocAPI(docPageModel.currentDocId.get()!).forceReload();
|
|
|
|
}
|
|
|
|
},
|
2021-08-26 16:35:11 +00:00
|
|
|
// If timezone, locale, or currency hasn't changed, disable the Save button.
|
|
|
|
saveDisabled: Computed.create(owner,
|
|
|
|
(use) => {
|
|
|
|
return (
|
|
|
|
use(timezoneObs) === docInfo.timezone.peek() &&
|
|
|
|
use(localeObs) === docSettings.locale &&
|
2021-09-15 23:35:21 +00:00
|
|
|
use(currencyObs) === docSettings.currency &&
|
|
|
|
use(engineObs) === docSettings.engine
|
2021-08-26 16:35:11 +00:00
|
|
|
);
|
|
|
|
})
|
2020-10-02 15:10:00 +00:00
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-08-26 16:35:11 +00:00
|
|
|
|
|
|
|
type LocaleItem = ACSelectItem & {locale?: string};
|
|
|
|
|
|
|
|
function buildLocaleSelect(
|
|
|
|
owner: IDisposableOwner,
|
|
|
|
locale: Observable<string>
|
|
|
|
) {
|
|
|
|
const localeList: LocaleItem[] = locales.map(l => ({
|
|
|
|
value: l.name, // Use name as a value, we will translate the name into the locale on save
|
|
|
|
label: l.name,
|
|
|
|
locale: l.code,
|
|
|
|
cleanText: l.name.trim().toLowerCase(),
|
2021-09-23 18:09:58 +00:00
|
|
|
})).sort(propertyCompare("label"));
|
2021-08-26 16:35:11 +00:00
|
|
|
const acIndex = new ACIndexImpl<LocaleItem>(localeList, 200, true);
|
|
|
|
// AC select will show the value (in this case locale) not a label when something is selected.
|
|
|
|
// To show the label - create another observable that will be in sync with the value, but
|
|
|
|
// will contain text.
|
|
|
|
const localeCode = locale.get();
|
|
|
|
const localeName = locales.find(l => l.code === localeCode)?.name || localeCode;
|
|
|
|
const textObs = Observable.create(owner, localeName);
|
|
|
|
return buildACSelect(owner,
|
|
|
|
{
|
|
|
|
acIndex, valueObs: textObs,
|
|
|
|
save(value, item: LocaleItem | undefined) {
|
|
|
|
if (!item) { throw new Error("Invalid locale"); }
|
|
|
|
textObs.set(value);
|
|
|
|
locale.set(item.locale!);
|
|
|
|
},
|
|
|
|
},
|
|
|
|
testId("locale-autocomplete")
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
// This matches the style used in showProfileModal in app/client/ui/AccountWidget.
|
|
|
|
const cssDataRow = styled('div', `
|
|
|
|
margin: 16px 0px;
|
|
|
|
font-size: ${vars.largeFontSize};
|
|
|
|
`);
|
2021-09-15 23:35:21 +00:00
|
|
|
|
|
|
|
// Check which engines can be selected in the UI, if any.
|
|
|
|
export function getSupportedEngineChoices(): EngineCode[] {
|
|
|
|
const gristConfig: GristLoadConfig = (window as any).gristConfig || {};
|
|
|
|
return gristConfig.supportEngines || [];
|
|
|
|
}
|