diff options
Diffstat (limited to 'testing/web-platform/tests/contacts/contacts-select.https.window.js')
-rw-r--r-- | testing/web-platform/tests/contacts/contacts-select.https.window.js | 185 |
1 files changed, 185 insertions, 0 deletions
diff --git a/testing/web-platform/tests/contacts/contacts-select.https.window.js b/testing/web-platform/tests/contacts/contacts-select.https.window.js new file mode 100644 index 0000000000..850217aadf --- /dev/null +++ b/testing/web-platform/tests/contacts/contacts-select.https.window.js @@ -0,0 +1,185 @@ +// META: script=/resources/test-only-api.js +// META: script=/resources/testdriver.js +// META: script=/resources/testdriver-vendor.js +// META: script=resources/helpers.js +'use strict'; + +// Verifies that |func|, when invoked, throws a TypeError exception. +async function expectTypeError(func) { + try { + await func(); + } catch (e) { + assert_equals(e.name, 'TypeError'); + return; + } + + assert_unreached('expected a TypeError, but none was thrown'); +} + +promise_test(async () => { + try { + await navigator.contacts.select(['name']); + assert_unreached('expected a SecurityError, but none was thrown'); + } catch (e) { + assert_equals(e.name, 'SecurityError'); + } +}, 'The Contact API requires a user gesture') + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + // At least one property must be provided. + await expectTypeError(() => navigator.contacts.select()); + await expectTypeError(() => navigator.contacts.select([])); + + // Per WebIDL parsing, no invalid values may be provided. + await expectTypeError(() => + navigator.contacts.select([''])); + await expectTypeError(() => + navigator.contacts.select(['foo'])); + await expectTypeError(() => + navigator.contacts.select(['name', 'photo'])); + +}, 'The Contact API requires valid properties to be provided'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + // Returns a NULL result, indicating that no results are available. + setSelectedContacts(null); + + await expectTypeError(() => navigator.contacts.select(['name'])); + +}, 'The Contact API can fail when the selector cannot be opened'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + setSelectedContacts([]); + + const properties = await navigator.contacts.getProperties(); + assert_true(properties.length > 0); + + // Requesting the available properties should not fail. + await navigator.contacts.select(properties); + +}, 'Supported contact properties are exposed.'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + const dwightAddress = { + country: 'US', + city: 'Scranton', + addressLine: ['Schrute Farms'], + }; + const michaelIcons = [new Blob('image binary data'.split(''), {type: 'image/test'})]; + + // Returns two contacts with all information available. + setSelectedContacts([ + { name: ['Dwight Schrute'], email: ['dwight@schrutefarmsbnb.com'], tel: ['000-0000'], address: [dwightAddress] }, + { name: ['Michael Scott', 'Prison Mike'], email: ['michael@dundermifflin.com'], icon: michaelIcons }, + ]); + + let results = await navigator.contacts.select(['name', 'email', 'icon', 'tel', 'address'], { multiple: true }); + assert_equals(results.length, 2); + results = results.sort((c1, c2) => JSON.stringify(c1) < JSON.stringify(c2) ? -1 : 1); + + { + const michael = results[0]; + + assert_own_property(michael, 'name'); + assert_own_property(michael, 'email'); + assert_own_property(michael, 'tel'); + assert_own_property(michael, 'address'); + assert_own_property(michael, 'icon'); + + assert_array_equals(michael.name, ['Michael Scott', 'Prison Mike']); + assert_array_equals(michael.email, ['michael@dundermifflin.com']); + assert_array_equals(michael.tel, []); + assert_array_equals(michael.address, []); + + assert_equals(michael.icon.length, michaelIcons.length); + assert_equals(michael.icon[0].type, michaelIcons[0].type); + assert_equals(michael.icon[0].size, michaelIcons[0].size); + assert_equals(await michael.icon[0].text(), await michaelIcons[0].text()); + } + + { + const dwight = results[1]; + assert_own_property(dwight, 'name'); + assert_own_property(dwight, 'email'); + assert_own_property(dwight, 'tel'); + assert_own_property(dwight, 'address'); + assert_own_property(dwight, 'icon'); + + assert_array_equals(dwight.name, ['Dwight Schrute']); + assert_array_equals(dwight.email, ['dwight@schrutefarmsbnb.com']); + assert_array_equals(dwight.tel, ['000-0000']); + assert_array_equals(dwight.icon, []); + + assert_equals(dwight.address.length, 1); + const selectedAddress = dwight.address[0]; + assert_object_equals({ + country: selectedAddress.country, + city: selectedAddress.city, + addressLine: selectedAddress.addressLine, + }, dwightAddress); + } +}, 'The Contact API correctly returns ContactInfo entries'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + // Returns two contacts with all information available. + setSelectedContacts([ + { name: ['Dwight Schrute'], email: ['dwight@schrutefarmsbnb.com'], tel: ['000-0000'] }, + { name: ['Michael Scott', 'Prison Mike'], email: ['michael@dundermifflin.com'] }, + ]); + + const results = await navigator.contacts.select(['name', 'email', 'tel']); + assert_equals(results.length, 1); + +}, 'Only one contact is returned if `multiple` is not set.'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + // Returns partial information since no e-mail addresses are requested. + setSelectedContacts([{ name: ['Creed'], email: ['creedthoughts@www.creedthoughts.gov.www'] }]); + + const results = await navigator.contacts.select(['name']); + + assert_equals(results.length, 1); + + { + const creed = results[0]; + + assert_array_equals(creed.name, ['Creed']); + assert_equals(creed.email, undefined); + assert_equals(creed.tel, undefined); + } +}, 'The Contact API does not include fields that were not requested'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + // Returns partial information since no e-mail addresses are requested. + setSelectedContacts([{ name: ['Kelly'] }]); + + // First request should work. + const promise1 = new Promise((resolve, reject) => { + navigator.contacts.select(['name']).then(resolve) + .catch(e => reject(e.message)); + }); + + // Second request should fail (since the first one didn't complete yet). + const promise2 = new Promise((resolve, reject) => { + navigator.contacts.select(['name']).then(contacts => reject('This was supposed to fail')) + .catch(e => resolve(e.name)); + }); + + const results = await Promise.all([promise1, promise2]); + const contacts = results[0]; + assert_equals(contacts.length, 1); + const contact = contacts[0]; + assert_equals(contact.name[0], 'Kelly'); + assert_equals(results[1], 'InvalidStateError'); + +}, 'The Contact API cannot be used again until the first operation is complete.'); + +contactsTestWithUserActivation(async (test, setSelectedContacts) => { + const iframe = document.createElement('iframe'); + document.body.appendChild(iframe); + iframe.src = 'resources/non-main-frame-select.html'; + await new Promise(resolve => window.addEventListener('message', event => resolve(event.data))) + .then(data => assert_equals(data.errorMsg, 'InvalidStateError')) + .finally(() => iframe.remove()) + +}, 'Test contacts.select() throws an InvalidStateError in a sub-frame'); |