snappymail/dev/External/ko.js

280 lines
7.4 KiB
JavaScript
Raw Normal View History

import { i18n, i18nToNodes, trigger } from 'Common/Translator';
2021-02-11 17:08:27 +08:00
import { doc, createElement, dropdownVisibility } from 'Common/Globals';
import { SaveSettingsStep } from 'Common/Enums';
const
isFunction = v => typeof v === 'function',
koValue = value => !ko.isObservable(value) && isFunction(value) ? value() : ko.unwrap(value);
2014-08-20 23:03:12 +08:00
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.tooltip = {
init: (element, fValueAccessor) => {
const sValue = koValue(fValueAccessor());
2016-06-30 08:02:45 +08:00
if ('off' === element.dataset.tooltipI18n) {
element.title = sValue;
} else {
element.title = i18n(sValue);
trigger.subscribe(() =>
element.title = i18n(sValue)
);
dropdownVisibility.subscribe(() =>
element.title = i18n(sValue)
);
2016-06-30 08:02:45 +08:00
}
},
update: (element, fValueAccessor) => {
const sValue = koValue(fValueAccessor());
if (sValue) {
element.title = 'off' === element.dataset.tooltipI18n ? sValue : i18n(sValue);
} else {
element.title = '';
2014-08-20 23:03:12 +08:00
}
2016-06-30 08:02:45 +08:00
}
};
2014-08-20 23:03:12 +08:00
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.tooltipErrorTip = {
init: element => {
2020-08-27 21:45:47 +08:00
doc.addEventListener('click', () => element.removeAttribute('data-rainloopErrorTip'));
2016-06-30 08:02:45 +08:00
},
update: (element, fValueAccessor) => {
const value = koValue(fValueAccessor());
2020-08-13 19:16:08 +08:00
if (value) {
2020-08-14 04:58:41 +08:00
setTimeout(() => element.setAttribute('data-rainloopErrorTip', value), 100);
2020-08-13 19:16:08 +08:00
} else {
element.removeAttribute('data-rainloopErrorTip');
2014-08-20 23:03:12 +08:00
}
2016-06-30 08:02:45 +08:00
}
};
2014-08-20 23:03:12 +08:00
ko.bindingHandlers.registerBootstrapDropdown = {
init: element => {
rl.Dropdowns.register(element);
element.ddBtn = new BSN.Dropdown(element.querySelector('[data-toggle="dropdown"]'));
}
2016-06-30 08:02:45 +08:00
};
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.openDropdownTrigger = {
update: (element, fValueAccessor) => {
2019-07-05 03:19:24 +08:00
if (ko.unwrap(fValueAccessor())) {
const el = element.ddBtn;
el.open || el.toggle();
// el.focus();
2016-06-30 08:02:45 +08:00
2020-08-24 17:14:35 +08:00
rl.Dropdowns.detectVisibility();
2016-06-30 08:02:45 +08:00
fValueAccessor()(false);
2014-08-20 23:03:12 +08:00
}
2016-06-30 08:02:45 +08:00
}
};
2014-08-20 23:03:12 +08:00
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.dropdownCloser = {
init: element => element.closest('.dropdown').addEventListener('click', event =>
event.target.closestWithin('.e-item', element) && element.ddBtn.toggle()
)
2016-06-30 08:02:45 +08:00
};
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.onEnter = {
init: (element, fValueAccessor, fAllBindingsAccessor, viewModel) => {
let fn = event => {
if ('Enter' == event.key) {
element.dispatchEvent(new Event('change'));
fValueAccessor().call(viewModel);
2014-08-20 23:03:12 +08:00
}
};
element.addEventListener('keydown', fn);
ko.utils.domNodeDisposal.addDisposeCallback(element, () => element.removeEventListener('keydown', fn));
2016-06-30 08:02:45 +08:00
}
};
2014-08-20 23:03:12 +08:00
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.onSpace = {
init: (element, fValueAccessor, fAllBindingsAccessor, viewModel) => {
let fn = event => {
if (' ' == event.key) {
fValueAccessor().call(viewModel, event);
2016-06-30 08:02:45 +08:00
}
};
element.addEventListener('keyup', fn);
ko.utils.domNodeDisposal.addDisposeCallback(element, () => element.removeEventListener('keyup', fn));
2016-06-30 08:02:45 +08:00
}
};
2015-02-15 08:30:21 +08:00
2016-06-30 08:02:45 +08:00
ko.bindingHandlers.modal = {
init: (element, fValueAccessor) => {
const close = element.querySelector('.close'),
click = () => fValueAccessor()(false);
close && close.addEventListener('click.koModal', click);
ko.utils.domNodeDisposal.addDisposeCallback(element, () =>
close.removeEventListener('click.koModal', click)
);
2016-06-30 08:02:45 +08:00
}
};
ko.bindingHandlers.i18nInit = {
init: element => i18nToNodes(element)
2016-06-30 08:02:45 +08:00
};
ko.bindingHandlers.i18nUpdate = {
update: (element, fValueAccessor) => {
2016-06-30 08:02:45 +08:00
ko.unwrap(fValueAccessor());
i18nToNodes(element);
2016-06-30 08:02:45 +08:00
}
};
ko.bindingHandlers.title = {
2020-08-27 21:45:47 +08:00
update: (element, fValueAccessor) => element.title = ko.unwrap(fValueAccessor())
2016-06-30 08:02:45 +08:00
};
ko.bindingHandlers.command = {
init: (element, fValueAccessor, fAllBindingsAccessor, viewModel, bindingContext) => {
const command = fValueAccessor();
2019-07-05 03:19:24 +08:00
if (!command || !command.isCommand) {
throw new Error('Value should be a command');
}
2019-07-05 03:19:24 +08:00
if (!command.enabled) {
command.enabled = ko.observable(true);
}
2014-08-20 23:03:12 +08:00
2019-07-05 03:19:24 +08:00
if (!command.canExecute) {
const __realCanExecute = command.__realCanExecute;
if (isFunction(__realCanExecute)) {
command.canExecute = ko.computed(() => command.enabled() && __realCanExecute.call(viewModel, viewModel));
2019-07-05 03:19:24 +08:00
} else {
command.canExecute = ko.computed(() => command.enabled() && !!__realCanExecute);
}
2016-06-30 08:02:45 +08:00
}
2014-08-20 23:03:12 +08:00
element.classList.add('command');
2020-09-15 15:29:25 +08:00
ko.bindingHandlers['FORM'==element.nodeName ? 'submit' : 'click'].init(
2019-07-05 03:19:24 +08:00
element,
fValueAccessor,
fAllBindingsAccessor,
viewModel,
bindingContext
);
2016-06-30 08:02:45 +08:00
},
update: (element, fValueAccessor) => {
const cl = element.classList,
command = fValueAccessor();
2014-08-20 23:03:12 +08:00
2021-02-09 00:37:21 +08:00
let disabled = !command.enabled();
2014-08-20 23:03:12 +08:00
2021-02-09 00:37:21 +08:00
cl.toggle('no-disabled', !disabled);
2014-08-20 23:03:12 +08:00
2021-02-09 00:37:21 +08:00
disabled = disabled || !command.canExecute();
['disable','disabled'].forEach(s => cl.toggle(s, disabled));
2014-08-20 23:03:12 +08:00
2021-01-25 05:58:06 +08:00
if (element.matches('INPUT,TEXTAREA,BUTTON')) {
2021-02-09 00:37:21 +08:00
element.disabled = disabled;
2014-08-20 23:03:12 +08:00
}
2016-06-30 08:02:45 +08:00
}
};
2014-08-20 23:03:12 +08:00
2021-02-11 17:08:27 +08:00
ko.bindingHandlers.saveTrigger = {
init: (element) => {
let icon = element;
if (element.matches('input,select,textarea')) {
element.classList.add('settings-saved-trigger-input');
element.after(element.saveTriggerIcon = icon = createElement('span'));
}
icon.classList.add('settings-save-trigger');
},
update: (element, fValueAccessor) => {
const value = parseInt(ko.unwrap(fValueAccessor()),10);
let cl = (element.saveTriggerIcon || element).classList;
if (element.saveTriggerIcon) {
cl.toggle('saving', value === SaveSettingsStep.Animate);
cl.toggle('success', value === SaveSettingsStep.TrueResult);
cl.toggle('error', value === SaveSettingsStep.FalseResult);
}
cl = element.classList;
cl.toggle('success', value === SaveSettingsStep.TrueResult);
cl.toggle('error', value === SaveSettingsStep.FalseResult);
}
};
2016-06-30 08:02:45 +08:00
// extenders
2014-10-04 19:58:01 +08:00
ko.extenders.limitedList = (target, limitedList) => {
const result = ko
2019-07-05 03:19:24 +08:00
.computed({
read: target,
write: (newValue) => {
const currentValue = ko.unwrap(target),
list = ko.unwrap(limitedList);
2020-09-04 23:07:35 +08:00
if (Array.isNotEmpty(list)) {
if (list.includes(newValue)) {
2019-07-05 03:19:24 +08:00
target(newValue);
} else if (list.includes(currentValue, list)) {
2019-07-05 03:19:24 +08:00
target(currentValue + ' ');
target(currentValue);
} else {
target(list[0] + ' ');
target(list[0]);
}
} else {
target('');
}
}
2019-07-05 03:19:24 +08:00
})
.extend({ notify: 'always' });
result(target());
2019-07-05 03:19:24 +08:00
if (!result.valueHasMutated) {
result.valueHasMutated = () => target.valueHasMutated();
2016-06-30 08:02:45 +08:00
}
2014-08-20 23:03:12 +08:00
return result;
2016-06-30 08:02:45 +08:00
};
2014-08-20 23:03:12 +08:00
ko.extenders.reversible = (target) => {
let value = target();
2014-08-20 23:03:12 +08:00
target.commit = () => value = target();
2014-08-20 23:03:12 +08:00
target.reverse = () => target(value);
target.commitedValue = () => value;
return target;
2016-06-30 08:02:45 +08:00
};
ko.extenders.toggleSubscribeProperty = (target, options) => {
const prop = options[1];
2019-07-05 03:19:24 +08:00
if (prop) {
target.subscribe(
prev => prev && prev[prop] && prev[prop](false),
2019-07-05 03:19:24 +08:00
options[0],
'beforeChange'
);
2014-08-20 23:03:12 +08:00
target.subscribe(next => next && next[prop] && next[prop](true), options[0]);
2016-06-30 08:02:45 +08:00
}
2014-08-20 23:03:12 +08:00
return target;
2016-06-30 08:02:45 +08:00
};
ko.extenders.falseTimeout = (target, option) => {
2021-02-11 17:08:27 +08:00
target.subscribe((() => target(false)).debounce(parseInt(option, 10) || 0));
return target;
2016-06-30 08:02:45 +08:00
};
// functions
2014-10-04 19:58:01 +08:00
ko.observable.fn.deleteAccessHelper = function() {
2020-10-27 18:09:24 +08:00
return this.extend({ falseTimeout: 3000, toggleSubscribeProperty: [this, 'deleteAccess'] });
2016-06-30 08:02:45 +08:00
};
2014-08-25 15:10:51 +08:00
ko.addObservablesTo = (target, observables) => {
Object.entries(observables).forEach(([key, value]) =>
2021-02-11 17:08:27 +08:00
target[key] = /*Array.isArray(value) ? ko.observableArray(value) :*/ ko.observable(value) );
};
ko.addComputablesTo = (target, computables) =>
Object.entries(computables).forEach(([key, fn]) => target[key] = ko.computed(fn));
ko.addSubscribablesTo = (target, subscribables) =>
Object.entries(subscribables).forEach(([key, fn]) => target[key].subscribe(fn));