// // Copyright © 2020, 2021 Anticrm Platform Contributors. // // Licensed under the Eclipse Public License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. You may // obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // // See the License for the specific language governing permissions and // limitations under the License. // import { Account, AttachedData, AttachedDoc, Class, Client, Data, Doc, FindResult, Ref, Space, Timestamp, UXObject } from '@anticrm/core' import type { Asset, Plugin } from '@anticrm/platform' import { IntlString, plugin } from '@anticrm/platform' import type { AnyComponent } from '@anticrm/ui' import { Viewlet } from '@anticrm/view' /** * @public */ export interface Organizations extends Space {} /** * @public */ export interface Persons extends Space {} /** * @public */ export interface ChannelProvider extends Doc, UXObject { placeholder: IntlString presenter?: AnyComponent integrationType?: Ref } /** * @public */ export interface Channel extends AttachedDoc { provider: Ref value: string items?: number lastMessage?: Timestamp } /** * @public */ export interface Contact extends Doc { name: string avatar?: string | null attachments?: number comments?: number channels?: number city: string } /** * @public */ export interface Person extends Contact {} /** * @public */ export interface Member extends AttachedDoc { contact: Ref } /** * @public */ export interface Organization extends Contact { members: number } /** * @public */ export interface Status extends AttachedDoc { attachedTo: Ref attachedToClass: Ref> name: string dueDate: Timestamp } /** * @public */ export interface Employee extends Person { statuses?: number } /** * @public */ export interface EmployeeAccount extends Account { employee: Ref name: string } const SEP = ',' /** * @public */ export function combineName (first: string, last: string): string { return last + SEP + first } /** * @public */ export function getFirstName (name: string): string { return name !== undefined ? name.substring(name.indexOf(SEP) + 1) : '' } /** * @public */ export function getLastName (name: string): string { return name !== undefined ? name.substring(0, name.indexOf(SEP)) : '' } /** * @public */ export function formatName (name: string): string { return getFirstName(name) + ' ' + getLastName(name) } /** * @public */ export const contactId = 'contact' as Plugin /** * @public */ const contactPlugin = plugin(contactId, { class: { ChannelProvider: '' as Ref>, Channel: '' as Ref>, Contact: '' as Ref>, Person: '' as Ref>, Persons: '' as Ref>, Member: '' as Ref>, Organization: '' as Ref>, Organizations: '' as Ref>, Employee: '' as Ref>, EmployeeAccount: '' as Ref>, Status: '' as Ref> }, component: { SocialEditor: '' as AnyComponent, CreateOrganization: '' as AnyComponent, CreatePerson: '' as AnyComponent }, channelProvider: { Email: '' as Ref, Phone: '' as Ref, LinkedIn: '' as Ref, Twitter: '' as Ref, Telegram: '' as Ref, GitHub: '' as Ref, Facebook: '' as Ref, Homepage: '' as Ref }, icon: { Phone: '' as Asset, Email: '' as Asset, Discord: '' as Asset, Facebook: '' as Asset, Instagram: '' as Asset, LinkedIn: '' as Asset, Telegram: '' as Asset, Twitter: '' as Asset, VK: '' as Asset, WhatsApp: '' as Asset, Youtube: '' as Asset, GitHub: '' as Asset, Edit: '' as Asset, Person: '' as Asset, Company: '' as Asset, SocialEdit: '' as Asset, Homepage: '' as Asset }, space: { Employee: '' as Ref, Contacts: '' as Ref }, app: { Contacts: '' as Ref }, string: { PersonAlreadyExists: '' as IntlString, Person: '' as IntlString, CreateOrganization: '' as IntlString }, viewlet: { TableMember: '' as Ref } }) export default contactPlugin /** * @public */ export async function findPerson ( client: Client, person: Data, channels: AttachedData[] ): Promise { if (channels.length === 0 || person.name.length === 0) { return [] } // Take only first part of first name for match. const values = channels.map((it) => it.value) // Same name persons const potentialChannels = await client.findAll(contactPlugin.class.Channel, { value: { $in: values } }) let potentialPersonIds = Array.from(new Set(potentialChannels.map((it) => it.attachedTo as Ref)).values()) if (potentialPersonIds.length === 0) { const firstName = getFirstName(person.name).split(' ').shift() ?? '' const lastName = getLastName(person.name) // try match using just first/last name potentialPersonIds = ( await client.findAll(contactPlugin.class.Person, { name: { $like: `${lastName}%${firstName}%` } }) ).map((it) => it._id) if (potentialPersonIds.length === 0) { return [] } } const potentialPersons: FindResult = await client.findAll( contactPlugin.class.Person, { _id: { $in: potentialPersonIds } }, { lookup: { _id: { channels: contactPlugin.class.Channel } } } ) const result: Person[] = [] for (const c of potentialPersons) { let matches = 0 if (c.name === person.name) { matches++ } if (c.city === person.city) { matches++ } for (const ch of (c.$lookup?.channels as Channel[]) ?? []) { for (const chc of channels) { if (chc.provider === ch.provider && chc.value === ch.value.trim()) { // We have matched value matches += 2 break } } } if (matches >= 2) { result.push(c) } } return result }