From 4dbf9eb66f8a81c743a2e5cbba4fadda986aaa94 Mon Sep 17 00:00:00 2001 From: Maxim Belov Date: Fri, 22 Mar 2024 23:12:24 +0200 Subject: [PATCH 1/2] feat(contacts): add plugin --- .../plugins/contacts/index.ts | 403 ++++++++++++++++++ 1 file changed, 403 insertions(+) create mode 100644 src/@awesome-cordova-plugins/plugins/contacts/index.ts diff --git a/src/@awesome-cordova-plugins/plugins/contacts/index.ts b/src/@awesome-cordova-plugins/plugins/contacts/index.ts new file mode 100644 index 0000000000..9c3c674423 --- /dev/null +++ b/src/@awesome-cordova-plugins/plugins/contacts/index.ts @@ -0,0 +1,403 @@ +import { + CordovaCheck, + CordovaInstance, + InstanceCheck, + InstanceProperty, + AwesomeCordovaNativePlugin, + Plugin, + checkAvailability, + getPromise, +} from '@awesome-cordova-plugins/core'; + +declare const window: any, navigator: any; + +export type ContactFieldType = + | '*' + | 'addresses' + | 'birthday' + | 'categories' + | 'country' + | 'department' + | 'displayName' + | 'emails' + | 'name.familyName' + | 'name.formatted' + | 'name.givenName' + | 'name.honorificPrefix' + | 'name.honorificSuffix' + | 'id' + | 'ims' + | 'locality' + | 'name.middleName' + | 'name' + | 'nickname' + | 'note' + | 'organizations' + | 'phoneNumbers' + | 'photos' + | 'postalCode' + | 'region' + | 'streetAddress' + | 'title' + | 'urls'; + +export interface IContactProperties { + /** A globally unique identifier. */ + id?: string; + + /** A globally unique identifier on Android. */ + rawId?: string; + + /** The name of this Contact, suitable for display to end users. */ + displayName?: string; + + /** An object containing all components of a persons name. */ + name?: IContactName; + + /** A casual name by which to address the contact. */ + nickname?: string; + + /** An array of all the contact's phone numbers. */ + phoneNumbers?: IContactField[]; + + /** An array of all the contact's email addresses. */ + emails?: IContactField[]; + + /** An array of all the contact's addresses. */ + addresses?: IContactAddress[]; + + /** An array of all the contact's IM addresses. */ + ims?: IContactField[]; + + /** An array of all the contact's organizations. */ + organizations?: IContactOrganization[]; + + /** The birthday of the contact. */ + birthday?: Date; + + /** A note about the contact. */ + note?: string; + + /** An array of the contact's photos. */ + photos?: IContactField[]; + + /** An array of all the user-defined categories associated with the contact. */ + categories?: IContactField[]; + + /** An array of web pages associated with the contact. */ + urls?: IContactField[]; +} + +/** + * @hidden + */ +export class Contact implements IContactProperties { + @InstanceProperty() id: string; + @InstanceProperty() displayName: string; + @InstanceProperty() name: IContactName; + @InstanceProperty() nickname: string; + @InstanceProperty() phoneNumbers: IContactField[]; + @InstanceProperty() emails: IContactField[]; + @InstanceProperty() addresses: IContactAddress[]; + @InstanceProperty() ims: IContactField[]; + @InstanceProperty() organizations: IContactOrganization[]; + @InstanceProperty() birthday: Date; + @InstanceProperty() note: string; + @InstanceProperty() photos: IContactField[]; + @InstanceProperty() categories: IContactField[]; + @InstanceProperty() urls: IContactField[]; + private _objectInstance: any; + + [key: string]: any; + + constructor() { + if (checkAvailability('navigator.contacts', 'create', 'Contacts') === true) { + this._objectInstance = navigator.contacts.create(); + } + } + + @InstanceCheck() + clone(): Contact { + const newContact: any = new Contact(); + for (const prop in this) { + if (prop === 'id') return; + newContact[prop] = this[prop]; + } + return newContact; + } + + @CordovaInstance() + remove(): Promise { + return; + } + + @InstanceCheck() + save(): Promise { + return getPromise((resolve: Function, reject: Function) => { + this._objectInstance.save((contact: any) => { + this._objectInstance = contact; + resolve(this); + }, reject); + }); + } +} + +/** + * @hidden + */ +export interface IContactError { + /** Error code */ + code: number; + /** Error message */ + message: string; +} + +/** + * @hidden + */ +export declare const ContactError: { + new (code: number): IContactError; + UNKNOWN_ERROR: number; + INVALID_ARGUMENT_ERROR: number; + TIMEOUT_ERROR: number; + PENDING_OPERATION_ERROR: number; + IO_ERROR: number; + NOT_SUPPORTED_ERROR: number; + PERMISSION_DENIED_ERROR: number; +}; + +export interface IContactName { + /** The complete name of the contact. */ + formatted?: string; + /** The contact's family name. */ + familyName?: string; + /** The contact's given name. */ + givenName?: string; + /** The contact's middle name. */ + middleName?: string; + /** The contact's prefix (example Mr. or Dr.) */ + honorificPrefix?: string; + /** The contact's suffix (example Esq.). */ + honorificSuffix?: string; +} + +/** + * @hidden + */ +export class ContactName implements IContactName { + constructor( + public formatted?: string, + public familyName?: string, + public givenName?: string, + public middleName?: string, + public honorificPrefix?: string, + public honorificSuffix?: string + ) {} +} + +export interface IContactField { + /** A string that indicates what type of field this is, home for example. */ + type?: string; + /** The value of the field, such as a phone number or email address. */ + value?: string; + /** Set to true if this ContactField contains the user's preferred value. */ + pref?: boolean; +} + +/** + * @hidden + */ +export class ContactField implements IContactField { + constructor(public type?: string, public value?: string, public pref?: boolean) {} +} + +export interface IContactAddress { + /** Set to true if this ContactAddress contains the user's preferred value. */ + pref?: boolean; + /** A string indicating what type of field this is, home for example. */ + type?: string; + /** The full address formatted for display. */ + formatted?: string; + /** The full street address. */ + streetAddress?: string; + /** The city or locality. */ + locality?: string; + /** The state or region. */ + region?: string; + /** The zip code or postal code. */ + postalCode?: string; + /** The country name. */ + country?: string; +} + +/** + * @hidden + */ +export class ContactAddress implements IContactAddress { + constructor( + public pref?: boolean, + public type?: string, + public formatted?: string, + public streetAddress?: string, + public locality?: string, + public region?: string, + public postalCode?: string, + public country?: string + ) {} +} + +export interface IContactOrganization { + /** Set to true if this ContactOrganization contains the user's preferred value. */ + pref?: boolean; + /** A string that indicates what type of field this is, home for example. */ + type?: string; + /** The name of the organization. */ + name?: string; + /** The department the contract works for. */ + department?: string; + /** The contact's title at the organization. */ + title?: string; +} + +/** + * @hidden + */ +export class ContactOrganization implements IContactOrganization { + constructor( + public type?: string, + public name?: string, + public department?: string, + public title?: string, + public pref?: boolean + ) {} +} + +/** Search options to filter navigator.contacts. */ +export interface IContactFindOptions { + /** The search string used to find navigator.contacts. */ + filter?: string; + /** Determines if the find operation returns multiple navigator.contacts. Defaults to false. */ + multiple?: boolean; + /** Contact fields to be returned back. If specified, the resulting Contact object only features values for these fields. */ + desiredFields?: string[]; + /** + * (Android only): Filters the search to only return contacts with a phone number informed. + */ + hasPhoneNumber?: boolean; +} + +/** + * @hidden + */ +export class ContactFindOptions implements IContactFindOptions { + constructor( + public filter?: string, + public multiple?: boolean, + public desiredFields?: string[], + public hasPhoneNumber?: boolean + ) {} +} + +/** + * @name Contacts + * @premier contacts + * @description + * Access and manage Contacts on the device. + * + * @usage + * + * ```typescript + * import { Contacts, Contact, ContactField, ContactName } from '@ionic-native/contacts/ngx'; + * + * constructor(private contacts: Contacts) { } + * + * let contact: Contact = this.contacts.create(); + * + * contact.name = new ContactName(null, 'Smith', 'John'); + * contact.phoneNumbers = [new ContactField('mobile', '6471234567')]; + * contact.save().then( + * () => console.log('Contact saved!', contact), + * (error: any) => console.error('Error saving contact.', error) + * ); + * + * ``` + * @classes + * Contact + * @interfaces + * IContactProperties + * IContactError + * IContactName + * IContactField + * IContactAddress + * IContactOrganization + * IContactFindOptions + */ +@Plugin({ + pluginName: 'Contacts', + plugin: 'cordova-plugin-contacts', + pluginRef: 'navigator.contacts', + repo: 'https://github.com/apache/cordova-plugin-contacts', + platforms: [ + 'Android', + 'BlackBerry 10', + 'Browser', + 'Firefox OS', + 'iOS', + 'Ubuntu', + 'Windows', + 'Windows 8', + 'Windows Phone', + ], +}) +export class Contacts extends AwesomeCordovaNativePlugin { + /** + * Create a single contact. + * @returns {Contact} Returns a Contact object + */ + create(): Contact { + return new Contact(); + } + + /** + * Search for contacts in the Contacts list. + * @param {ContactFieldType[]} fields Contact fields to be used as a search qualifier + * @param {IContactFindOptions} [options] Optional options for the query + * @returns {Promise} Returns a Promise that resolves with the search results (an array of Contact objects) + */ + @CordovaCheck() + find(fields: ContactFieldType[], options?: IContactFindOptions): Promise { + return getPromise((resolve: Function, reject: Function) => { + navigator.contacts.find( + fields, + (contacts: any[]) => { + resolve(contacts.map(processContact)); + }, + reject, + options + ); + }); + } + + /** + * Select a single Contact. + * @returns {Promise} Returns a Promise that resolves with the selected Contact + */ + @CordovaCheck() + pickContact(): Promise { + return getPromise((resolve: Function, reject: Function) => { + navigator.contacts.pickContact((contact: any) => resolve(processContact(contact)), reject); + }); + } +} + +/** + * @hidden + */ +function processContact(contact: any) { + const newContact = new Contact(); + for (const prop in contact) { + if (typeof contact[prop] === 'function') continue; + newContact[prop] = contact[prop]; + } + return newContact; +} From efe6474ae3ff77c525c97c8368d54d32a34474d8 Mon Sep 17 00:00:00 2001 From: Maxim Belov Date: Fri, 22 Mar 2024 23:26:31 +0200 Subject: [PATCH 2/2] update repo --- src/@awesome-cordova-plugins/plugins/contacts/index.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/@awesome-cordova-plugins/plugins/contacts/index.ts b/src/@awesome-cordova-plugins/plugins/contacts/index.ts index 9c3c674423..7dcc2802a4 100644 --- a/src/@awesome-cordova-plugins/plugins/contacts/index.ts +++ b/src/@awesome-cordova-plugins/plugins/contacts/index.ts @@ -336,7 +336,7 @@ export class ContactFindOptions implements IContactFindOptions { pluginName: 'Contacts', plugin: 'cordova-plugin-contacts', pluginRef: 'navigator.contacts', - repo: 'https://github.com/apache/cordova-plugin-contacts', + repo: 'https://github.com/herdwatch-apps/cordova-plugin-contacts', platforms: [ 'Android', 'BlackBerry 10',