import { useSignal } from '@preact/signals'; import { AddressBook, Contact } from '/lib/models/contacts.ts'; import { RequestBody as GetRequestBody, ResponseBody as GetResponseBody } from '/routes/api/contacts/get.tsx'; import { RequestBody as AddRequestBody, ResponseBody as AddResponseBody } from '/routes/api/contacts/add.tsx'; import { RequestBody as DeleteRequestBody, ResponseBody as DeleteResponseBody } from '/routes/api/contacts/delete.tsx'; import { RequestBody as ImportRequestBody, ResponseBody as ImportResponseBody } from '/routes/api/contacts/import.tsx'; import { RequestBody as AddAddressBookRequestBody, ResponseBody as AddAddressBookResponseBody, } from '/routes/api/contacts/add-addressbook.tsx'; import { RequestBody as DeleteAddressBookRequestBody, ResponseBody as DeleteAddressBookResponseBody, } from '/routes/api/contacts/delete-addressbook.tsx'; interface ContactsProps { initialAddressBookId: string; initialContacts: Contact[]; initialAddressBooks: AddressBook[]; page: number; contactsCount: number; baseUrl: string; search?: string; } const CONTACTS_PER_PAGE_COUNT = 10; // This helps make the UI a bit faster (less stuff to render) export default function Contacts( { initialContacts, initialAddressBooks, page, contactsCount, search, initialAddressBookId, baseUrl }: ContactsProps, ) { const isAdding = useSignal(false); const isDeleting = useSignal(false); const isExporting = useSignal(false); const isImporting = useSignal(false); const contacts = useSignal(initialContacts); const addressBooks = useSignal(initialAddressBooks); const selectedAddressBookId = useSignal(initialAddressBookId); const selectedAddressBookName = useSignal( initialAddressBooks.find((addressBook) => addressBook.uid === initialAddressBookId)?.displayName || 'Address Book', ); const isAddressBooksDropdownOpen = useSignal(false); const isOptionsDropdownOpen = useSignal(false); async function onClickAddContact() { if (isAdding.value) { return; } const firstName = (prompt(`What's the **first name** for the new contact?`) || '').trim(); if (!firstName) { alert('A first name is required for a new contact!'); return; } const lastName = (prompt(`What's the **last name** for the new contact?`) || '').trim(); isAdding.value = true; try { const requestBody: AddRequestBody = { firstName, lastName, addressBookId: selectedAddressBookId.value }; const response = await fetch(`/api/contacts/add`, { method: 'POST', body: JSON.stringify(requestBody), }); if (!response.ok) { throw new Error(`Failed to add contact. ${response.statusText} ${await response.text()}`); } const result = await response.json() as AddResponseBody; if (!result.success) { throw new Error('Failed to add contact!'); } contacts.value = [...result.contacts]; } catch (error) { console.error(error); } isAdding.value = false; } function toggleOptionsDropdown() { isOptionsDropdownOpen.value = !isOptionsDropdownOpen.value; } async function onClickAddAddressBook() { if (isAdding.value) { return; } const name = (prompt(`What's the **name** for the new address book?`) || '').trim(); if (!name) { alert('A name is required for a new address book!'); return; } isAdding.value = true; isAddressBooksDropdownOpen.value = false; try { const requestBody: AddAddressBookRequestBody = { name }; const response = await fetch(`/api/contacts/add-addressbook`, { method: 'POST', body: JSON.stringify(requestBody), }); if (!response.ok) { throw new Error(`Failed to add address book. ${response.statusText} ${await response.text()}`); } const result = await response.json() as AddAddressBookResponseBody; if (!result.success) { throw new Error('Failed to add address book!'); } addressBooks.value = [...result.addressBooks]; } catch (error) { console.error(error); } isAdding.value = false; } function toggleAddressBooksDropdown() { isAddressBooksDropdownOpen.value = !isAddressBooksDropdownOpen.value; } function onClickSelectAddressBook(addressBookId: string) { selectedAddressBookId.value = addressBookId; selectedAddressBookName.value = addressBooks.value.find((addressBook) => addressBook.uid === addressBookId)?.displayName || 'Address Book'; isAddressBooksDropdownOpen.value = false; window.location.href = `/contacts?addressBookId=${addressBookId}`; } async function onClickDeleteAddressBook(addressBookId: string) { if (confirm('Are you sure you want to delete this address book?')) { if (isDeleting.value) { return; } isDeleting.value = true; try { const requestBody: DeleteAddressBookRequestBody = { addressBookId }; const response = await fetch(`/api/contacts/delete-addressbook`, { method: 'POST', body: JSON.stringify(requestBody), }); if (!response.ok) { throw new Error(`Failed to delete address book. ${response.statusText} ${await response.text()}`); } const result = await response.json() as DeleteAddressBookResponseBody; if (!result.success) { throw new Error('Failed to delete address book!'); } addressBooks.value = [...result.addressBooks]; selectedAddressBookId.value = ''; selectedAddressBookName.value = ''; window.location.href = `/contacts`; } catch (error) { console.error(error); } isDeleting.value = false; } } async function onClickDeleteContact(contactId: string) { if (confirm('Are you sure you want to delete this contact?')) { if (isDeleting.value) { return; } isDeleting.value = true; try { const requestBody: DeleteRequestBody = { contactId, addressBookId: selectedAddressBookId.value }; const response = await fetch(`/api/contacts/delete`, { method: 'POST', body: JSON.stringify(requestBody), }); if (!response.ok) { throw new Error(`Failed to delete contact. ${response.statusText} ${await response.text()}`); } const result = await response.json() as DeleteResponseBody; if (!result.success) { throw new Error('Failed to delete contact!'); } contacts.value = [...result.contacts]; } catch (error) { console.error(error); } isDeleting.value = false; } } function onClickImportVCard() { isOptionsDropdownOpen.value = false; if (isImporting.value) { return; } const fileInput = document.createElement('input'); fileInput.type = 'file'; fileInput.click(); fileInput.onchange = (event) => { const files = (event.target as HTMLInputElement)?.files!; const file = files[0]; if (!file) { return; } const reader = new FileReader(); reader.onload = async (fileRead) => { const importFileContents = fileRead.target?.result; if (!importFileContents || isImporting.value) { return; } isImporting.value = true; try { const vCards = importFileContents!.toString(); const requestBody: ImportRequestBody = { addressBookId: selectedAddressBookId.value, vCards }; const response = await fetch(`/api/contacts/import`, { method: 'POST', body: JSON.stringify(requestBody), }); if (!response.ok) { throw new Error(`Failed to import contact. ${response.statusText} ${await response.text()}`); } const result = await response.json() as ImportResponseBody; if (!result.success) { throw new Error('Failed to import contact!'); } contacts.value = [...result.contacts]; } catch (error) { console.error(error); } isImporting.value = false; }; reader.readAsText(file, 'UTF-8'); }; } async function onClickExportVCard() { isOptionsDropdownOpen.value = false; if (isExporting.value) { return; } isExporting.value = true; const fileName = ['contacts-', new Date().toISOString().substring(0, 19).replace(/:/g, '-'), '.vcf'] .join(''); try { const requestBody: GetRequestBody = { addressBookId: selectedAddressBookId.value }; const response = await fetch(`/api/contacts/get`, { method: 'POST', body: JSON.stringify(requestBody), }); if (!response.ok) { throw new Error(`Failed to export contact. ${response.statusText} ${await response.text()}`); } const result = await response.json() as GetResponseBody; if (!result.success) { throw new Error('Failed to get contact!'); } const exportContents = result.contacts.map((contact) => contact.data).join('\n\n'); // Add content-type const vCardContent = ['data:text/vcard; charset=utf-8,', encodeURIComponent(exportContents)].join(''); // Download the file const data = vCardContent; const link = document.createElement('a'); link.setAttribute('href', data); link.setAttribute('download', fileName); link.click(); link.remove(); } catch (error) { console.error(error); } isExporting.value = false; } const pagesCount = Math.ceil(contactsCount / CONTACTS_PER_PAGE_COUNT); const pages = Array.from({ length: pagesCount }).map((_value, index) => index + 1); return ( <>
{contacts.value.map((contact) => ( ))} {contacts.value.length === 0 ? ( ) : null}
First Name Last Name
{contact.firstName} {contact.lastName}
No contacts to show
{isDeleting.value ? ( <> Deleting... ) : null} {isExporting.value ? ( <> Exporting... ) : null} {isImporting.value ? ( <> Importing... ) : null} {!isDeleting.value && !isExporting.value && !isImporting.value ? <>  : null}
{pagesCount > 0 ? (
) : null}
CardDAV URL:{' '} {baseUrl}/carddav
); }