// // Copyright © 2020, 2021 Anticrm Platform Contributors. // Copyright © 2021 Hardcore Engineering Inc. // // 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 { Channel, Contact, Employee, getGravatarUrl, getName } from '@hcengineering/contact' import { Class, Client, DocumentQuery, Ref, RelatedDocument, WithLookup } from '@hcengineering/core' import login from '@hcengineering/login' import { IntlString, Resources, getResource } from '@hcengineering/platform' import { MessageBox, ObjectSearchResult, getClient, getFileUrl } from '@hcengineering/presentation' import { AnyComponent, AnySvelteComponent, showPopup } from '@hcengineering/ui' import AccountArrayEditor from './components/AccountArrayEditor.svelte' import AccountBox from './components/AccountBox.svelte' import AssigneeBox from './components/AssigneeBox.svelte' import Avatar from './components/Avatar.svelte' import ChannelFilter from './components/ChannelFilter.svelte' import Channels from './components/Channels.svelte' import ChannelsDropdown from './components/ChannelsDropdown.svelte' import ChannelsEditor from './components/ChannelsEditor.svelte' import ChannelsPresenter from './components/ChannelsPresenter.svelte' import ChannelsView from './components/ChannelsView.svelte' import CombineAvatars from './components/CombineAvatars.svelte' import ContactArrayEditor from './components/ContactArrayEditor.svelte' import ContactPresenter from './components/ContactPresenter.svelte' import ContactRefPresenter from './components/ContactRefPresenter.svelte' import Contacts from './components/Contacts.svelte' import ContactsTabs from './components/ContactsTabs.svelte' import CreateEmployee from './components/CreateEmployee.svelte' import CreateOrganization from './components/CreateOrganization.svelte' import CreatePerson from './components/CreatePerson.svelte' import EditEmployee from './components/EditEmployee.svelte' import EditMember from './components/EditMember.svelte' import EditOrganization from './components/EditOrganization.svelte' import EditPerson from './components/EditPerson.svelte' import EditableAvatar from './components/EditableAvatar.svelte' import EmployeeAccountPresenter from './components/EmployeeAccountPresenter.svelte' import EmployeeAccountRefPresenter from './components/EmployeeAccountRefPresenter.svelte' import EmployeeArrayEditor from './components/EmployeeArrayEditor.svelte' import EmployeeBox from './components/EmployeeBox.svelte' import EmployeeBrowser from './components/EmployeeBrowser.svelte' import EmployeeEditor from './components/EmployeeEditor.svelte' import EmployeePresenter from './components/EmployeePresenter.svelte' import EmployeeRefPresenter from './components/EmployeeRefPresenter.svelte' import MemberPresenter from './components/MemberPresenter.svelte' import Members from './components/Members.svelte' import MembersPresenter from './components/MembersPresenter.svelte' import MergeEmployee from './components/MergeEmployee.svelte' import OrganizationEditor from './components/OrganizationEditor.svelte' import OrganizationPresenter from './components/OrganizationPresenter.svelte' import PersonEditor from './components/PersonEditor.svelte' import PersonPresenter from './components/PersonPresenter.svelte' import PersonRefPresenter from './components/PersonRefPresenter.svelte' import SocialEditor from './components/SocialEditor.svelte' import SpaceMembers from './components/SpaceMembers.svelte' import UserBox from './components/UserBox.svelte' import UserBoxList from './components/UserBoxList.svelte' import UserInfo from './components/UserInfo.svelte' import UsersPopup from './components/UsersPopup.svelte' import ActivityChannelMessage from './components/activity/ActivityChannelMessage.svelte' import ExpandRightDouble from './components/icons/ExpandRightDouble.svelte' import IconMembers from './components/icons/Members.svelte' import ChannelPresenter from './components/ChannelPresenter.svelte' import ChannelPanel from './components/ChannelPanel.svelte' import ActivityChannelPresenter from './components/activity/ActivityChannelPresenter.svelte' import contact from './plugin' import { contactTitleProvider, employeeSort, filterChannelInResult, filterChannelNinResult, getContactFirstName, getContactLastName, getContactLink, getContactName, getCurrentEmployeeEmail, getCurrentEmployeeName, getCurrentEmployeePosition, resolveLocation } from './utils' export { employeeByIdStore, employeesStore } from './utils' export { Channels, ChannelsEditor, ContactPresenter, ChannelsView, ChannelsDropdown, EmployeePresenter, PersonPresenter, OrganizationPresenter, EmployeeBrowser, MemberPresenter, EmployeeEditor, EmployeeAccountRefPresenter, MembersPresenter, EditPerson, EmployeeRefPresenter, AccountArrayEditor, AccountBox, CreateOrganization, ExpandRightDouble, EditableAvatar, UserBox, AssigneeBox, Avatar, UsersPopup, EmployeeBox, UserBoxList, Members, SpaceMembers, CombineAvatars, UserInfo, IconMembers } const toObjectSearchResult = (e: WithLookup): ObjectSearchResult => ({ doc: e, title: getName(e), icon: Avatar, iconProps: { size: 'x-small', avatar: e.avatar }, component: UserInfo, componentProps: { size: 'x-small' } }) async function queryContact ( _class: Ref>, client: Client, search: string, filter?: { in?: RelatedDocument[], nin?: RelatedDocument[] } ): Promise { const q: DocumentQuery = { name: { $like: `%${search}%` } } return await doContactQuery(_class, q, filter, client) } async function queryEmployee ( client: Client, search: string, filter?: { in?: RelatedDocument[], nin?: RelatedDocument[] } ): Promise { const q1 = await doContactQuery( contact.class.Employee, { name: { $like: `%${search}%` }, active: true }, filter, client ) const q2 = await doContactQuery( contact.class.Employee, { displayName: { $like: `%${search}%` }, active: true }, { in: filter?.in, nin: [...(filter?.nin ?? []), ...Array.from(q1.map((it) => ({ _id: it.doc._id, _class: it.doc._class })))] }, client ) return q1.concat(q2) } async function doContactQuery ( _class: Ref>, q: DocumentQuery, filter: { in?: RelatedDocument[] | undefined, nin?: RelatedDocument[] | undefined } | undefined, client: Client ): Promise { if (_class === contact.class.Employee) { q = { ...q, active: true } } if (filter?.in !== undefined || filter?.nin !== undefined) { q._id = {} if (filter.in !== undefined) { q._id.$in = filter.in?.map((it) => it._id as Ref) } if (filter.nin !== undefined) { q._id.$nin = filter.nin?.map((it) => it._id as Ref) } } return (await client.findAll(_class, q, { limit: 200 })).map(toObjectSearchResult) } async function kickEmployee (doc: Employee): Promise { const client = getClient() const email = await client.findOne(contact.class.EmployeeAccount, { employee: doc._id }) if (!doc.active) { showPopup( MessageBox, { label: contact.string.DeleteEmployee, message: contact.string.DeleteEmployee }, undefined, (res?: boolean) => { if (res === true) { void client.remove(doc) } } ) return } if (email === undefined) { await client.update(doc, { active: false }) } else { showPopup( MessageBox, { label: contact.string.KickEmployee, message: contact.string.KickEmployeeDescr }, undefined, (res?: boolean) => { if (res === true) { // eslint-disable-next-line @typescript-eslint/no-floating-promises getResource(login.function.LeaveWorkspace).then(async (f) => { await f(email.email) }) } } ) } } async function openChannelURL (doc: Channel): Promise { if (doc.value.startsWith('http://') || doc.value.startsWith('https://')) { window.open(doc.value) } } export interface PersonLabelTooltip { personLabel?: IntlString placeholderLabel?: IntlString component?: AnySvelteComponent | AnyComponent props?: any } export * from './utils' export default async (): Promise => ({ actionImpl: { KickEmployee: kickEmployee, OpenChannel: openChannelURL }, component: { ContactArrayEditor, PersonEditor, OrganizationEditor, ContactPresenter, ContactRefPresenter, PersonRefPresenter, PersonPresenter, OrganizationPresenter, ChannelsPresenter, CreatePerson, CreateOrganization, EditPerson, EditEmployee, EditOrganization, SocialEditor, Contacts, ContactsTabs, EmployeeAccountPresenter, EmployeePresenter, EmployeeRefPresenter, Members, MemberPresenter, MembersPresenter, EditMember, EmployeeArrayEditor, EmployeeEditor, CreateEmployee, AccountArrayEditor, ChannelFilter, MergeEmployee, Avatar, UserBoxList, ActivityChannelMessage, ChannelPresenter, ChannelPanel, ActivityChannelPresenter, SpaceMembers }, completion: { EmployeeQuery: async ( client: Client, query: string, filter?: { in?: RelatedDocument[], nin?: RelatedDocument[] } ) => await queryEmployee(client, query, filter), PersonQuery: async (client: Client, query: string, filter?: { in?: RelatedDocument[], nin?: RelatedDocument[] }) => await queryContact(contact.class.Person, client, query, filter), OrganizationQuery: async ( client: Client, query: string, filter?: { in?: RelatedDocument[], nin?: RelatedDocument[] } ) => await queryContact(contact.class.Organization, client, query, filter) }, function: { GetFileUrl: getFileUrl, GetGravatarUrl: getGravatarUrl, GetColorUrl: (uri: string) => uri, EmployeeSort: employeeSort, FilterChannelInResult: filterChannelInResult, FilterChannelNinResult: filterChannelNinResult, GetCurrentEmployeeName: getCurrentEmployeeName, GetCurrentEmployeeEmail: getCurrentEmployeeEmail, GetCurrentEmployeePosition: getCurrentEmployeePosition, GetContactName: getContactName, GetContactFirstName: getContactFirstName, GetContactLastName: getContactLastName, GetContactLink: getContactLink, ContactTitleProvider: contactTitleProvider }, resolver: { Location: resolveLocation } })