snappymail/dev/View/Popup/Contacts.js

530 lines
12 KiB
JavaScript
Raw Normal View History

import ko from 'ko';
import {
2019-07-05 03:19:24 +08:00
SaveSettingsStep,
Scope
} from 'Common/Enums';
2021-01-25 05:58:06 +08:00
import { ComposeType } from 'Common/EnumsUser';
import { isNonEmptyArray, pInt } from 'Common/Utils';
import { delegateRunOnDestroy, computedPaginatorHelper, showMessageComposer } from 'Common/UtilsUser';
2019-07-05 03:19:24 +08:00
import { Selector } from 'Common/Selector';
2021-02-04 18:25:00 +08:00
import { serverRequestRaw, serverRequest } from 'Common/Links';
2019-07-05 03:19:24 +08:00
import { i18n, getNotification } from 'Common/Translator';
import { SettingsUserStore } from 'Stores/User/Settings';
import { ContactUserStore } from 'Stores/User/Contact';
import Remote from 'Remote/User/Fetch';
2019-07-05 03:19:24 +08:00
import { EmailModel } from 'Model/Email';
import { ContactModel } from 'Model/Contact';
2021-01-25 05:58:06 +08:00
import { ContactPropertyModel, ContactPropertyType } from 'Model/ContactProperty';
import { decorateKoCommands, hideScreenPopup } from 'Knoin/Knoin';
import { AbstractViewPopup } from 'Knoin/AbstractViews';
const CONTACTS_PER_PAGE = 50,
propertyIsMail = prop => prop.isType(ContactPropertyType.Email),
propertyIsName = prop => prop.isType(ContactPropertyType.FirstName) || prop.isType(ContactPropertyType.LastName);
class ContactsPopupView extends AbstractViewPopup {
constructor() {
super('Contacts');
this.bBackToCompose = false;
this.sLastComposeFocusedField = '';
this.allowContactsSync = ContactUserStore.allowSync;
this.enableContactsSync = ContactUserStore.enableSync;
this.addObservables({
search: '',
contactsCount: 0,
currentContact: null,
2016-06-30 08:02:45 +08:00
importUploaderButton: null,
2016-06-30 08:02:45 +08:00
contactsPage: 1,
2016-06-30 08:02:45 +08:00
emptySelection: true,
viewClearSearch: false,
viewID: '',
viewReadOnly: false,
viewSaveTrigger: SaveSettingsStep.Idle,
viewSaving: false,
watchDirty: false,
watchHash: false
});
this.contacts = ContactUserStore;
this.viewProperties = ko.observableArray();
this.useCheckboxesInList = SettingsUserStore.useCheckboxesInList;
this.selector = new Selector(
ContactUserStore,
2019-07-05 03:19:24 +08:00
this.currentContact,
null,
'.e-contact-item .actionHandle',
'.e-contact-item .checkboxItem',
'.e-contact-item.focused'
);
2015-02-16 05:55:59 +08:00
2020-10-25 18:46:58 +08:00
this.selector.on('onItemSelect', contact => {
this.populateViewContact(contact || null);
2019-07-05 03:19:24 +08:00
if (!contact) {
this.emptySelection(true);
}
});
2020-10-25 18:46:58 +08:00
this.selector.on('onItemGetUid', contact => contact ? contact.generateUid() : '');
2016-09-10 06:38:16 +08:00
this.bDropPageAfterDelete = false;
2015-02-16 05:55:59 +08:00
2021-04-23 16:47:24 +08:00
// this.saveCommandDebounce = this.saveCommand.bind(this).debounce(1000);
const
// propertyFocused = property => !property.isValid() && !property.focused(),
pagecount = () => Math.max(1, Math.ceil(this.contactsCount() / CONTACTS_PER_PAGE));
this.addComputables({
contactsPageCount: pagecount,
contactsPaginator: computedPaginatorHelper(this.contactsPage, pagecount),
viewPropertiesNames: () => this.viewProperties.filter(propertyIsName),
viewPropertiesEmails: () => this.viewProperties.filter(propertyIsMail),
viewPropertiesOther: () => this.viewProperties.filter(property => property.isType(ContactPropertyType.Nick)),
viewPropertiesWeb: () => this.viewProperties.filter(property => property.isType(ContactPropertyType.Web)),
viewPropertiesPhones: () => this.viewProperties.filter(property => property.isType(ContactPropertyType.Phone)),
contactHasValidName: () => !!this.viewProperties.find(prop => propertyIsName(prop) && prop.isValid()),
contactsCheckedOrSelected: () => {
const checked = ContactUserStore.filter(item => item.checked && item.checked()),
selected = this.currentContact();
return selected
? [...checked, selected].unique()
: checked;
},
contactsCheckedOrSelectedUids: () => this.contactsCheckedOrSelected().map(contact => contact.id),
viewHash: () => '' + this.viewProperties.map(property => property.value && property.value()).join('')
});
this.search.subscribe(() => this.reloadContactList());
this.viewHash.subscribe(() => {
if (this.watchHash() && !this.viewReadOnly() && !this.watchDirty()) {
this.watchDirty(true);
}
});
decorateKoCommands(this, {
newCommand: 1,
deleteCommand: self => 0 < self.contactsCheckedOrSelected().length,
newMessageCommand: self => 0 < self.contactsCheckedOrSelected().length,
clearCommand: 1,
saveCommand: self => !self.viewSaving() && !self.viewReadOnly()
&& (self.contactHasValidName() || self.viewProperties.find(prop => propertyIsMail(prop) && prop.isValid())),
syncCommand: self => !self.contacts.syncing() && !self.contacts.importing()
});
2016-09-10 06:38:16 +08:00
}
2016-09-10 06:38:16 +08:00
newCommand() {
this.populateViewContact(null);
this.currentContact(null);
}
2016-09-10 06:38:16 +08:00
deleteCommand() {
this.deleteSelectedContacts();
this.emptySelection(true);
}
2016-09-10 06:38:16 +08:00
newMessageCommand() {
2019-07-05 03:19:24 +08:00
let aE = [],
2016-09-10 06:38:16 +08:00
toEmails = null,
ccEmails = null,
bccEmails = null;
2016-09-10 06:38:16 +08:00
const aC = this.contactsCheckedOrSelected();
if (isNonEmptyArray(aC)) {
aE = aC.map(oItem => {
2019-07-05 03:19:24 +08:00
if (oItem) {
const data = oItem.getNameAndEmailHelper(),
2016-09-10 06:38:16 +08:00
email = data ? new EmailModel(data[0], data[1]) : null;
2019-07-05 03:19:24 +08:00
if (email && email.validate()) {
2016-09-10 06:38:16 +08:00
return email;
}
}
2014-08-21 23:08:34 +08:00
2016-09-10 06:38:16 +08:00
return null;
});
aE = aE.filter(value => !!value);
2016-09-10 06:38:16 +08:00
}
if (isNonEmptyArray(aE)) {
2016-09-10 06:38:16 +08:00
this.bBackToCompose = false;
2017-07-06 07:49:12 +08:00
hideScreenPopup(ContactsPopupView);
2013-12-23 08:06:48 +08:00
2019-07-05 03:19:24 +08:00
switch (this.sLastComposeFocusedField) {
2016-09-10 06:38:16 +08:00
case 'cc':
ccEmails = aE;
break;
case 'bcc':
bccEmails = aE;
break;
case 'to':
default:
toEmails = aE;
break;
}
2016-09-10 06:38:16 +08:00
this.sLastComposeFocusedField = '';
setTimeout(() =>
showMessageComposer([ComposeType.Empty, null, toEmails, ccEmails, bccEmails])
, 200);
2016-09-10 06:38:16 +08:00
}
2016-09-10 06:38:16 +08:00
return true;
}
2016-09-10 06:38:16 +08:00
clearCommand() {
this.search('');
}
2014-08-21 23:08:34 +08:00
2016-09-10 06:38:16 +08:00
saveCommand() {
this.viewSaving(true);
this.viewSaveTrigger(SaveSettingsStep.Animate);
2014-08-21 23:08:34 +08:00
const requestUid = Jua.randomId();
2019-07-05 03:19:24 +08:00
Remote.contactSave(
(iError, oData) => {
2019-07-05 03:19:24 +08:00
let res = false;
this.viewSaving(false);
if (
!iError &&
2019-07-05 03:19:24 +08:00
oData.Result.RequestUid === requestUid &&
0 < pInt(oData.Result.ResultID)
) {
if (!this.viewID()) {
2019-07-05 03:19:24 +08:00
this.viewID(pInt(oData.Result.ResultID));
}
this.reloadContactList(); // TODO: remove when e-contact-foreach is dynamic
2019-07-05 03:19:24 +08:00
res = true;
}
setTimeout(() =>
this.viewSaveTrigger(res ? SaveSettingsStep.TrueResult : SaveSettingsStep.FalseResult)
, 350);
2019-07-05 03:19:24 +08:00
if (res) {
this.watchDirty(false);
setTimeout(() => this.viewSaveTrigger(SaveSettingsStep.Idle), 1000);
2019-07-05 03:19:24 +08:00
}
},
requestUid,
this.viewID(),
this.viewProperties.map(oItem => oItem.toJSON())
2019-07-05 03:19:24 +08:00
);
2016-09-10 06:38:16 +08:00
}
2016-09-10 06:38:16 +08:00
syncCommand() {
rl.app.contactsSync(iError => {
iError && alert(getNotification(iError));
2016-06-30 08:02:45 +08:00
2016-09-10 06:38:16 +08:00
this.reloadContactList(true);
});
2016-06-30 08:02:45 +08:00
}
getPropertyPlaceholder(type) {
let result = '';
2019-07-05 03:19:24 +08:00
switch (type) {
case ContactPropertyType.LastName:
result = 'CONTACTS/PLACEHOLDER_ENTER_LAST_NAME';
break;
case ContactPropertyType.FirstName:
result = 'CONTACTS/PLACEHOLDER_ENTER_FIRST_NAME';
break;
case ContactPropertyType.Nick:
result = 'CONTACTS/PLACEHOLDER_ENTER_NICK_NAME';
break;
// no default
}
2014-08-21 23:08:34 +08:00
return result;
2016-06-30 08:02:45 +08:00
}
addNewProperty(type, typeStr) {
2019-07-05 03:19:24 +08:00
this.viewProperties.push(
new ContactPropertyModel(type, typeStr || '', '', true, this.getPropertyPlaceholder(type))
);
2016-06-30 08:02:45 +08:00
}
addNewOrFocusProperty(type, typeStr) {
const item = this.viewProperties.find(prop => prop.isType(type));
2019-07-05 03:19:24 +08:00
if (item) {
item.focused(true);
2019-07-05 03:19:24 +08:00
} else {
this.addNewProperty(type, typeStr);
}
}
2016-06-30 08:02:45 +08:00
addNewEmail() {
this.addNewProperty(ContactPropertyType.Email, 'Home');
}
2016-06-30 08:02:45 +08:00
addNewPhone() {
this.addNewProperty(ContactPropertyType.Phone, 'Mobile');
}
2016-06-30 08:02:45 +08:00
addNewWeb() {
this.addNewProperty(ContactPropertyType.Web);
}
2016-06-30 08:02:45 +08:00
addNewNickname() {
this.addNewOrFocusProperty(ContactPropertyType.Nick);
}
2016-06-30 08:02:45 +08:00
addNewNotes() {
this.addNewOrFocusProperty(ContactPropertyType.Note);
}
2016-06-30 08:02:45 +08:00
addNewBirthday() {
this.addNewOrFocusProperty(ContactPropertyType.Birthday);
}
2016-06-30 08:02:45 +08:00
exportVcf() {
2021-02-04 18:25:00 +08:00
rl.app.download(serverRequestRaw('ContactsVcf'));
}
2016-06-30 08:02:45 +08:00
exportCsv() {
2021-02-04 18:25:00 +08:00
rl.app.download(serverRequestRaw('ContactsCsv'));
}
initUploader() {
2019-07-05 03:19:24 +08:00
if (this.importUploaderButton()) {
const j = new Jua({
2021-02-04 18:25:00 +08:00
action: serverRequest('UploadContacts'),
name: 'uploader',
queueSize: 1,
multipleSizeLimit: 1,
disableMultiple: true,
disableDocumentDropPrevent: true,
clickElement: this.importUploaderButton()
2019-07-05 03:19:24 +08:00
});
2019-07-05 03:19:24 +08:00
if (j) {
j.on('onStart', () => {
ContactUserStore.importing(true);
2019-07-05 03:19:24 +08:00
}).on('onComplete', (id, result, data) => {
ContactUserStore.importing(false);
2019-07-05 03:19:24 +08:00
this.reloadContactList();
if (!id || !result || !data || !data.Result) {
alert(i18n('CONTACTS/ERROR_IMPORT_FILE'));
2019-07-05 03:19:24 +08:00
}
});
}
2014-08-21 23:08:34 +08:00
}
2016-06-30 08:02:45 +08:00
}
removeCheckedOrSelectedContactsFromList() {
const contacts = this.contactsCheckedOrSelected();
2014-08-21 23:08:34 +08:00
2019-07-05 03:19:24 +08:00
let currentContact = this.currentContact(),
count = ContactUserStore.length;
if (contacts.length) {
contacts.forEach(contact => {
if (currentContact && currentContact.id === contact.id) {
currentContact = null;
this.currentContact(null);
}
contact.deleted(true);
--count;
2016-06-30 08:02:45 +08:00
});
2019-07-05 03:19:24 +08:00
if (0 >= count) {
this.bDropPageAfterDelete = true;
}
setTimeout(() => {
contacts.forEach(contact => {
ContactUserStore.remove(contact);
delegateRunOnDestroy(contact);
});
2020-08-14 04:58:41 +08:00
}, 500);
}
2016-06-30 08:02:45 +08:00
}
2014-08-21 23:08:34 +08:00
deleteSelectedContacts() {
if (this.contactsCheckedOrSelected().length) {
2021-04-23 16:47:24 +08:00
Remote.contactsDelete((iError, oData) => {
if (500 < (!iError && oData && oData.Time ? pInt(oData.Time) : 0)) {
this.reloadContactList(this.bDropPageAfterDelete);
} else {
setTimeout(() => this.reloadContactList(this.bDropPageAfterDelete), 500);
}
}, this.contactsCheckedOrSelectedUids());
2014-08-21 23:08:34 +08:00
this.removeCheckedOrSelectedContactsFromList();
}
2016-06-30 08:02:45 +08:00
}
removeProperty(oProp) {
this.viewProperties.remove(oProp);
delegateRunOnDestroy(oProp);
2016-06-30 08:02:45 +08:00
}
/**
* @param {?ContactModel} contact
*/
populateViewContact(contact) {
let id = '';
this.watchHash(false);
this.emptySelection(false);
this.viewReadOnly(false);
2019-07-05 03:19:24 +08:00
if (contact) {
id = contact.id;
this.viewReadOnly(!!contact.readOnly);
} else {
contact = new ContactModel;
contact.initDefaultProperties();
2014-08-21 23:08:34 +08:00
}
this.viewID(id);
2014-10-04 19:58:01 +08:00
// delegateRunOnDestroy(this.viewProperties());
// this.viewProperties([]);
this.viewProperties(contact.properties);
this.watchDirty(false);
this.watchHash(true);
}
/**
* @param {boolean=} dropPagePosition = false
*/
reloadContactList(dropPagePosition = false) {
let offset = (this.contactsPage() - 1) * CONTACTS_PER_PAGE;
2014-08-21 23:08:34 +08:00
this.bDropPageAfterDelete = false;
2016-06-30 08:02:45 +08:00
2019-07-05 03:19:24 +08:00
if (dropPagePosition) {
this.contactsPage(1);
offset = 0;
}
2014-08-21 23:08:34 +08:00
ContactUserStore.loading(true);
2019-07-05 03:19:24 +08:00
Remote.contacts(
(iError, data) => {
2019-07-05 03:19:24 +08:00
let count = 0,
list = [];
2021-03-18 21:48:21 +08:00
if (!iError && isNonEmptyArray(data.Result.List)) {
data.Result.List.forEach(item => {
item = ContactModel.reviveFromJson(item);
item && list.push(item);
});
2019-07-05 03:19:24 +08:00
2021-03-18 21:48:21 +08:00
count = pInt(data.Result.Count);
count = 0 < count ? count : 0;
}
2019-07-05 03:19:24 +08:00
this.contactsCount(count);
delegateRunOnDestroy(ContactUserStore());
ContactUserStore(list);
ContactUserStore.loading(false);
this.viewClearSearch(!!this.search());
2019-07-05 03:19:24 +08:00
},
offset,
CONTACTS_PER_PAGE,
this.search()
);
}
onBuild(dom) {
this.selector.init(dom.querySelector('.b-list-content'), Scope.Contacts);
shortcuts.add('delete', '', Scope.Contacts, () => {
this.deleteCommand();
return false;
});
2016-06-30 08:02:45 +08:00
shortcuts.add('c,w', '', Scope.Contacts, () => {
this.newMessageCommand();
return false;
});
const self = this;
2014-08-21 23:08:34 +08:00
dom.addEventListener('click', event => {
2020-10-12 21:11:25 +08:00
let el = event.target.closestWithin('.e-paginator .e-page', dom);
if (el && ko.dataFor(el)) {
self.contactsPage(pInt(ko.dataFor(el).value));
2019-07-05 03:19:24 +08:00
self.reloadContactList();
}
});
2014-08-21 23:08:34 +08:00
this.initUploader();
}
onShow(bBackToCompose, sLastComposeFocusedField) {
this.bBackToCompose = undefined === bBackToCompose ? false : !!bBackToCompose;
this.sLastComposeFocusedField = undefined === sLastComposeFocusedField ? '' : sLastComposeFocusedField;
rl.route.off();
this.reloadContactList(true);
}
2014-08-21 23:08:34 +08:00
onHide() {
rl.route.on();
2015-02-16 05:55:59 +08:00
this.currentContact(null);
this.emptySelection(true);
this.search('');
this.contactsCount(0);
2014-10-04 19:58:01 +08:00
delegateRunOnDestroy(ContactUserStore());
ContactUserStore([]);
this.sLastComposeFocusedField = '';
2015-02-16 05:55:59 +08:00
2019-07-05 03:19:24 +08:00
if (this.bBackToCompose) {
this.bBackToCompose = false;
showMessageComposer();
}
2016-06-30 08:02:45 +08:00
}
}
2019-07-05 03:19:24 +08:00
export { ContactsPopupView, ContactsPopupView as default };