2019-07-05 03:19:24 +08:00
|
|
|
import { AbstractModel } from 'Knoin/AbstractModel';
|
2022-10-31 05:19:52 +08:00
|
|
|
import { addObservablesTo, addComputablesTo } from 'External/ko';
|
2016-07-07 05:03:30 +08:00
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
import { JCard } from 'DAV/JCard';
|
|
|
|
//import { VCardProperty } from 'DAV/VCardProperty';
|
|
|
|
|
|
|
|
const nProps = [
|
|
|
|
'surName',
|
|
|
|
'givenName',
|
|
|
|
'middleName',
|
|
|
|
'namePrefix',
|
|
|
|
'nameSuffix'
|
|
|
|
];
|
|
|
|
|
2022-07-26 04:55:48 +08:00
|
|
|
/*
|
|
|
|
const propertyMap = [
|
|
|
|
// vCard 2.1 properties and up
|
|
|
|
'N' => 'Text',
|
|
|
|
'FN' => 'FlatText',
|
|
|
|
'PHOTO' => 'Binary',
|
|
|
|
'BDAY' => 'DateAndOrTime',
|
|
|
|
'ADR' => 'Text',
|
|
|
|
'TEL' => 'FlatText',
|
|
|
|
'EMAIL' => 'FlatText',
|
|
|
|
'GEO' => 'FlatText',
|
|
|
|
'TITLE' => 'FlatText',
|
|
|
|
'ROLE' => 'FlatText',
|
|
|
|
'LOGO' => 'Binary',
|
|
|
|
'ORG' => 'Text',
|
|
|
|
'NOTE' => 'FlatText',
|
|
|
|
'REV' => 'TimeStamp',
|
|
|
|
'SOUND' => 'FlatText',
|
|
|
|
'URL' => 'Uri',
|
|
|
|
'UID' => 'FlatText',
|
|
|
|
'VERSION' => 'FlatText',
|
2022-09-05 17:04:12 +08:00
|
|
|
'KEY' => 'FlatText', // <uri>data:application/pgp-keys;base64,AZaz09==</uri>
|
2022-07-26 04:55:48 +08:00
|
|
|
'TZ' => 'Text',
|
|
|
|
|
|
|
|
// vCard 3.0 properties
|
|
|
|
'CATEGORIES' => 'Text',
|
|
|
|
'SORT-STRING' => 'FlatText',
|
|
|
|
'PRODID' => 'FlatText',
|
|
|
|
'NICKNAME' => 'Text',
|
|
|
|
|
|
|
|
// rfc2739 properties
|
|
|
|
'FBURL' => 'Uri',
|
|
|
|
'CAPURI' => 'Uri',
|
|
|
|
'CALURI' => 'Uri',
|
|
|
|
'CALADRURI' => 'Uri',
|
|
|
|
|
|
|
|
// rfc4770 properties
|
|
|
|
'IMPP' => 'Uri',
|
|
|
|
|
|
|
|
// vCard 4.0 properties
|
|
|
|
'SOURCE' => 'Uri',
|
|
|
|
'XML' => 'FlatText',
|
|
|
|
'ANNIVERSARY' => 'DateAndOrTime',
|
|
|
|
'CLIENTPIDMAP' => 'Text',
|
|
|
|
'LANG' => 'LanguageTag',
|
|
|
|
'GENDER' => 'Text',
|
|
|
|
'KIND' => 'FlatText',
|
|
|
|
'MEMBER' => 'Uri',
|
|
|
|
'RELATED' => 'Uri',
|
|
|
|
|
|
|
|
// rfc6474 properties
|
|
|
|
'BIRTHPLACE' => 'FlatText',
|
|
|
|
'DEATHPLACE' => 'FlatText',
|
|
|
|
'DEATHDATE' => 'DateAndOrTime',
|
|
|
|
|
|
|
|
// rfc6715 properties
|
|
|
|
'EXPERTISE' => 'FlatText',
|
|
|
|
'HOBBY' => 'FlatText',
|
|
|
|
'INTEREST' => 'FlatText',
|
|
|
|
'ORG-DIRECTORY' => 'FlatText
|
|
|
|
];
|
|
|
|
*/
|
|
|
|
|
2021-01-22 23:32:08 +08:00
|
|
|
export class ContactModel extends AbstractModel {
|
2016-07-16 05:29:42 +08:00
|
|
|
constructor() {
|
2020-10-19 01:19:45 +08:00
|
|
|
super();
|
2016-07-07 05:03:30 +08:00
|
|
|
|
2022-09-07 01:22:06 +08:00
|
|
|
this.jCard = ['vcard',[]];
|
2016-07-07 05:03:30 +08:00
|
|
|
|
2022-10-31 05:19:52 +08:00
|
|
|
addObservablesTo(this, {
|
2022-09-09 16:02:40 +08:00
|
|
|
// Also used by Selector
|
2020-10-25 18:46:58 +08:00
|
|
|
focused: false,
|
|
|
|
selected: false,
|
|
|
|
checked: false,
|
2022-09-09 16:02:40 +08:00
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
deleted: false,
|
|
|
|
readOnly: false,
|
|
|
|
|
|
|
|
id: 0,
|
|
|
|
givenName: '', // FirstName
|
|
|
|
surName: '', // LastName
|
|
|
|
middleName: '', // MiddleName
|
|
|
|
namePrefix: '', // NamePrefix
|
2022-07-25 19:25:02 +08:00
|
|
|
nameSuffix: '', // NameSuffix
|
2022-09-06 20:26:07 +08:00
|
|
|
nickname: null,
|
2022-09-07 05:18:14 +08:00
|
|
|
note: null,
|
2022-09-06 20:26:07 +08:00
|
|
|
|
2022-09-07 04:35:04 +08:00
|
|
|
// Business
|
|
|
|
org: '',
|
|
|
|
department: '',
|
|
|
|
title: '',
|
|
|
|
|
|
|
|
// Crypto
|
2022-09-06 20:26:07 +08:00
|
|
|
encryptpref: '',
|
|
|
|
signpref: ''
|
2022-05-29 06:22:50 +08:00
|
|
|
});
|
|
|
|
// this.email = koArrayWithDestroy();
|
|
|
|
this.email = ko.observableArray();
|
|
|
|
this.tel = ko.observableArray();
|
|
|
|
this.url = ko.observableArray();
|
2022-09-06 20:26:07 +08:00
|
|
|
this.adr = ko.observableArray();
|
2022-05-29 06:22:50 +08:00
|
|
|
|
2022-10-31 05:19:52 +08:00
|
|
|
addComputablesTo(this, {
|
2022-09-07 01:22:06 +08:00
|
|
|
fullName: () => [this.namePrefix(), this.givenName(), this.middleName(), this.surName()].join(' ').trim(),
|
2022-05-29 06:22:50 +08:00
|
|
|
|
|
|
|
display: () => {
|
2022-09-07 01:22:06 +08:00
|
|
|
let a = this.fullName(),
|
2022-11-28 20:49:38 +08:00
|
|
|
b = this.email()[0]?.value(),
|
2022-09-07 01:22:06 +08:00
|
|
|
c = this.nickname();
|
|
|
|
return a || b || c;
|
2022-05-29 06:22:50 +08:00
|
|
|
}
|
|
|
|
/*
|
|
|
|
fullName: {
|
|
|
|
read: () => this.givenName() + " " + this.surName(),
|
|
|
|
write: value => {
|
|
|
|
this.jCard.set('fn', value/*, params, group* /)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
2020-10-25 18:46:58 +08:00
|
|
|
});
|
2016-07-07 05:03:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns {Array|null}
|
|
|
|
*/
|
|
|
|
getNameAndEmailHelper() {
|
2022-05-29 06:22:50 +08:00
|
|
|
let name = (this.givenName() + ' ' + this.surName()).trim(),
|
2022-11-28 20:49:38 +08:00
|
|
|
email = this.email()[0]?.value();
|
2022-05-29 06:22:50 +08:00
|
|
|
/*
|
|
|
|
// this.jCard.getOne('fn')?.notEmpty() ||
|
|
|
|
this.jCard.parseFullName({set:true});
|
|
|
|
// let name = this.jCard.getOne('nickname'),
|
|
|
|
let name = this.jCard.getOne('fn'),
|
|
|
|
email = this.jCard.getOne('email');
|
|
|
|
*/
|
2020-07-28 23:20:14 +08:00
|
|
|
return email ? [email, name] : null;
|
2016-07-07 05:03:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-10-19 01:46:47 +08:00
|
|
|
* @static
|
2022-09-06 20:26:07 +08:00
|
|
|
* @param {jCard} json
|
2020-10-19 01:46:47 +08:00
|
|
|
* @returns {?ContactModel}
|
2016-07-07 05:03:30 +08:00
|
|
|
*/
|
2020-10-19 01:46:47 +08:00
|
|
|
static reviveFromJson(json) {
|
|
|
|
const contact = super.reviveFromJson(json);
|
|
|
|
if (contact) {
|
2022-05-29 06:22:50 +08:00
|
|
|
let jCard = new JCard(json.jCard),
|
|
|
|
props = jCard.getOne('n')?.value;
|
|
|
|
props && props.forEach((value, index) =>
|
|
|
|
value && contact[nProps[index]](value)
|
|
|
|
);
|
|
|
|
|
2022-09-07 05:18:14 +08:00
|
|
|
['nickname', 'note', 'title'].forEach(field => {
|
2022-09-07 04:35:04 +08:00
|
|
|
props = jCard.getOne(field);
|
|
|
|
props && contact[field](props.value);
|
|
|
|
});
|
|
|
|
|
|
|
|
if ((props = jCard.getOne('org')?.value)) {
|
|
|
|
contact.org(props[0]);
|
|
|
|
contact.department(props[1] || '');
|
|
|
|
}
|
2022-07-25 19:25:02 +08:00
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
['email', 'tel', 'url'].forEach(field => {
|
|
|
|
props = jCard.get(field);
|
|
|
|
props && props.forEach(prop => {
|
|
|
|
contact[field].push({
|
|
|
|
value: ko.observable(prop.value)
|
|
|
|
// type: prop.params.type
|
|
|
|
});
|
2016-07-07 05:03:30 +08:00
|
|
|
});
|
2022-05-29 06:22:50 +08:00
|
|
|
});
|
|
|
|
|
2022-09-06 20:26:07 +08:00
|
|
|
props = jCard.get('adr');
|
|
|
|
props && props.forEach(prop => {
|
|
|
|
contact.adr.push({
|
|
|
|
street: ko.observable(prop.value[2]),
|
|
|
|
street_ext: ko.observable(prop.value[1]),
|
|
|
|
locality: ko.observable(prop.value[3]),
|
|
|
|
region: ko.observable(prop.value[4]),
|
|
|
|
postcode: ko.observable(prop.value[5]),
|
|
|
|
pobox: ko.observable(prop.value[0]),
|
|
|
|
country: ko.observable(prop.value[6]),
|
|
|
|
preferred: ko.observable(prop.params.pref),
|
|
|
|
type: ko.observable(prop.params.type) // HOME | WORK
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-09-07 01:22:06 +08:00
|
|
|
props = jCard.getOne('x-crypto');
|
2022-09-06 20:26:07 +08:00
|
|
|
contact.signpref(props?.params.signpref || 'Ask');
|
|
|
|
contact.encryptpref(props?.params.encryptpref || 'Ask');
|
|
|
|
// contact.encryptpref(props?.params.allowed || 'PGP/INLINE,PGP/MIME,S/MIME,S/MIMEOpaque');
|
|
|
|
|
2022-09-07 01:22:06 +08:00
|
|
|
contact.jCard = json.jCard;
|
2016-07-07 05:03:30 +08:00
|
|
|
}
|
2020-10-19 01:46:47 +08:00
|
|
|
return contact;
|
2016-07-07 05:03:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
generateUid() {
|
2022-07-23 04:59:50 +08:00
|
|
|
return '' + this.id;
|
2016-07-07 05:03:30 +08:00
|
|
|
}
|
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
addEmail() {
|
|
|
|
// home, work
|
|
|
|
this.email.push({
|
|
|
|
value: ko.observable('')
|
|
|
|
// type: prop.params.type
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
addTel() {
|
|
|
|
// home, work, text, voice, fax, cell, video, pager, textphone, iana-token, x-name
|
|
|
|
this.tel.push({
|
|
|
|
value: ko.observable('')
|
|
|
|
// type: prop.params.type
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
addUrl() {
|
|
|
|
// home, work
|
|
|
|
this.url.push({
|
|
|
|
value: ko.observable('')
|
|
|
|
// type: prop.params.type
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
addNickname() {
|
|
|
|
// home, work
|
|
|
|
this.nickname() || this.nickname('');
|
|
|
|
}
|
|
|
|
|
2022-09-07 05:18:14 +08:00
|
|
|
addNote() {
|
|
|
|
this.note() || this.note('');
|
|
|
|
}
|
|
|
|
|
2022-09-07 01:22:06 +08:00
|
|
|
hasChanges()
|
|
|
|
{
|
2022-09-08 15:37:27 +08:00
|
|
|
return this.email().filter(v => v.length).length && this.toJSON().jCard != JSON.stringify(this.jCard);
|
2022-09-07 01:22:06 +08:00
|
|
|
}
|
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
toJSON()
|
|
|
|
{
|
2022-09-07 01:22:06 +08:00
|
|
|
let jCard = new JCard(this.jCard);
|
2022-05-29 06:22:50 +08:00
|
|
|
jCard.set('n', [
|
|
|
|
this.surName(),
|
|
|
|
this.givenName(),
|
|
|
|
this.middleName(),
|
|
|
|
this.namePrefix(),
|
|
|
|
this.nameSuffix()
|
|
|
|
]/*, params, group*/);
|
|
|
|
// jCard.parseFullName({set:true});
|
|
|
|
|
2022-09-07 05:18:14 +08:00
|
|
|
['nickname', 'note', 'title'].forEach(field =>
|
2022-09-07 04:35:04 +08:00
|
|
|
this[field]() ? jCard.set(field, this[field]()/*, params, group*/) : jCard.remove(field)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (this.org()) {
|
|
|
|
let org = [this.org()];
|
|
|
|
if (this.department()) {
|
|
|
|
org.push(this.department());
|
|
|
|
}
|
2022-09-07 05:18:14 +08:00
|
|
|
let prop = jCard.getOne('org');
|
|
|
|
prop ? prop.value = org : jCard.set('org', org);
|
2022-09-07 04:35:04 +08:00
|
|
|
} else {
|
|
|
|
jCard.remove('');
|
|
|
|
}
|
2022-07-25 19:25:02 +08:00
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
['email', 'tel', 'url'].forEach(field => {
|
|
|
|
let values = this[field].map(item => item.value());
|
|
|
|
jCard.get(field).forEach(prop => {
|
|
|
|
let i = values.indexOf(prop.value);
|
|
|
|
if (0 > i || !prop.value) {
|
|
|
|
jCard.remove(prop);
|
|
|
|
} else {
|
|
|
|
values.splice(i, 1);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
values.forEach(value => value && jCard.add(field, value));
|
|
|
|
});
|
|
|
|
|
2022-09-07 01:22:06 +08:00
|
|
|
jCard.set('x-crypto', '', {
|
2022-09-06 20:26:07 +08:00
|
|
|
allowed: 'PGP/INLINE,PGP/MIME,S/MIME,S/MIMEOpaque',
|
|
|
|
signpref: this.signpref(),
|
|
|
|
encryptpref: this.encryptpref()
|
|
|
|
}, 'x-crypto');
|
|
|
|
|
2022-05-29 06:22:50 +08:00
|
|
|
// Done by server
|
|
|
|
// jCard.set('rev', '2022-05-21T10:59:52Z')
|
|
|
|
|
|
|
|
return {
|
2023-01-25 01:58:25 +08:00
|
|
|
uid: this.id,
|
2022-05-29 06:22:50 +08:00
|
|
|
jCard: JSON.stringify(jCard)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-07-07 05:03:30 +08:00
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
lineAsCss() {
|
2022-09-05 17:04:12 +08:00
|
|
|
return (this.selected() ? 'selected' : '')
|
|
|
|
+ (this.deleted() ? ' deleted' : '')
|
|
|
|
+ (this.checked() ? ' checked' : '')
|
|
|
|
+ (this.focused() ? ' focused' : '');
|
2016-07-07 05:03:30 +08:00
|
|
|
}
|
|
|
|
}
|