aboutsummaryrefslogtreecommitdiffstats
path: root/test/e2e
diff options
context:
space:
mode:
authorWhymarrh Whitby <whymarrh.whitby@gmail.com>2019-07-03 09:49:59 +0800
committerGitHub <noreply@github.com>2019-07-03 09:49:59 +0800
commit5737df249d70062252a65039e7c3f96dd046da90 (patch)
tree4d9f019dd800d9f516fade711ac466ff3db7bc3e /test/e2e
parent8b5ef5b45e78e100a24c7fa5c92630e5182a894b (diff)
downloadtangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.tar
tangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.tar.gz
tangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.tar.bz2
tangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.tar.lz
tangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.tar.xz
tangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.tar.zst
tangerine-wallet-browser-5737df249d70062252a65039e7c3f96dd046da90.zip
Move e2e tests out of beta dir (#6785)
Diffstat (limited to 'test/e2e')
-rw-r--r--test/e2e/contract-test/contract.js (renamed from test/e2e/beta/contract-test/contract.js)0
-rw-r--r--test/e2e/contract-test/index.html (renamed from test/e2e/beta/contract-test/index.html)0
-rw-r--r--test/e2e/drizzle.spec.js (renamed from test/e2e/beta/drizzle.spec.js)2
-rw-r--r--test/e2e/fetch-mocks.js (renamed from test/e2e/beta/fetch-mocks.js)0
-rw-r--r--test/e2e/from-import-ui.spec.js (renamed from test/e2e/beta/from-import-beta-ui.spec.js)2
-rw-r--r--test/e2e/helpers.js (renamed from test/e2e/beta/helpers.js)2
-rw-r--r--test/e2e/metamask-responsive-ui.spec.js (renamed from test/e2e/beta/metamask-beta-responsive-ui.spec.js)2
-rw-r--r--test/e2e/metamask-ui.spec.js (renamed from test/e2e/beta/metamask-beta-ui.spec.js)2
-rw-r--r--test/e2e/metamask.spec.js352
-rwxr-xr-xtest/e2e/run-all.sh (renamed from test/e2e/beta/run-all.sh)6
-rwxr-xr-xtest/e2e/run-drizzle.sh (renamed from test/e2e/beta/run-drizzle.sh)2
-rwxr-xr-xtest/e2e/run-web3.sh (renamed from test/e2e/beta/run-web3.sh)2
-rw-r--r--test/e2e/web3.spec.js (renamed from test/e2e/beta/web3.spec.js)2
13 files changed, 11 insertions, 363 deletions
diff --git a/test/e2e/beta/contract-test/contract.js b/test/e2e/contract-test/contract.js
index 3f22b442a..3f22b442a 100644
--- a/test/e2e/beta/contract-test/contract.js
+++ b/test/e2e/contract-test/contract.js
diff --git a/test/e2e/beta/contract-test/index.html b/test/e2e/contract-test/index.html
index 6e134dc36..6e134dc36 100644
--- a/test/e2e/beta/contract-test/index.html
+++ b/test/e2e/contract-test/index.html
diff --git a/test/e2e/beta/drizzle.spec.js b/test/e2e/drizzle.spec.js
index 309df952c..cda55cf01 100644
--- a/test/e2e/beta/drizzle.spec.js
+++ b/test/e2e/drizzle.spec.js
@@ -9,7 +9,7 @@ const {
installWebExt,
getExtensionIdChrome,
getExtensionIdFirefox,
-} = require('../func')
+} = require('./func')
const {
checkBrowserForConsoleErrors,
closeAllWindowHandlesExcept,
diff --git a/test/e2e/beta/fetch-mocks.js b/test/e2e/fetch-mocks.js
index 6b885cc10..6b885cc10 100644
--- a/test/e2e/beta/fetch-mocks.js
+++ b/test/e2e/fetch-mocks.js
diff --git a/test/e2e/beta/from-import-beta-ui.spec.js b/test/e2e/from-import-ui.spec.js
index 625330dbb..31a858cf1 100644
--- a/test/e2e/beta/from-import-beta-ui.spec.js
+++ b/test/e2e/from-import-ui.spec.js
@@ -9,7 +9,7 @@ const {
installWebExt,
getExtensionIdChrome,
getExtensionIdFirefox,
-} = require('../func')
+} = require('./func')
const {
checkBrowserForConsoleErrors,
closeAllWindowHandlesExcept,
diff --git a/test/e2e/beta/helpers.js b/test/e2e/helpers.js
index b6fc35e08..9b2029dde 100644
--- a/test/e2e/beta/helpers.js
+++ b/test/e2e/helpers.js
@@ -2,7 +2,7 @@ const fs = require('fs')
const mkdirp = require('mkdirp')
const pify = require('pify')
const assert = require('assert')
-const { delay } = require('../func')
+const { delay } = require('./func')
const { until } = require('selenium-webdriver')
module.exports = {
diff --git a/test/e2e/beta/metamask-beta-responsive-ui.spec.js b/test/e2e/metamask-responsive-ui.spec.js
index 9ef560556..007b5cbf6 100644
--- a/test/e2e/beta/metamask-beta-responsive-ui.spec.js
+++ b/test/e2e/metamask-responsive-ui.spec.js
@@ -9,7 +9,7 @@ const {
installWebExt,
getExtensionIdChrome,
getExtensionIdFirefox,
-} = require('../func')
+} = require('./func')
const {
checkBrowserForConsoleErrors,
closeAllWindowHandlesExcept,
diff --git a/test/e2e/beta/metamask-beta-ui.spec.js b/test/e2e/metamask-ui.spec.js
index e61277c18..c98a8a965 100644
--- a/test/e2e/beta/metamask-beta-ui.spec.js
+++ b/test/e2e/metamask-ui.spec.js
@@ -9,7 +9,7 @@ const {
installWebExt,
getExtensionIdChrome,
getExtensionIdFirefox,
-} = require('../func')
+} = require('./func')
const {
assertElementNotPresent,
checkBrowserForConsoleErrors,
diff --git a/test/e2e/metamask.spec.js b/test/e2e/metamask.spec.js
deleted file mode 100644
index 13af6cb22..000000000
--- a/test/e2e/metamask.spec.js
+++ /dev/null
@@ -1,352 +0,0 @@
-const path = require('path')
-const assert = require('assert')
-const { By, Key, until } = require('selenium-webdriver')
-const { delay, createModifiedTestBuild, setupBrowserAndExtension, verboseReportOnFailure } = require('./func')
-
-describe('Metamask popup page', function () {
- const browser = process.env.SELENIUM_BROWSER
- let driver, accountAddress, tokenAddress, extensionUri
-
- this.timeout(0)
-
- before(async function () {
- const srcPath = path.resolve(`dist/${browser}`)
- const { extPath } = await createModifiedTestBuild({ browser, srcPath })
- const installResult = await setupBrowserAndExtension({ browser, extPath })
- driver = installResult.driver
- extensionUri = installResult.extensionUri
-
- await driver.get(extensionUri)
- await delay(300)
- })
-
- afterEach(async function () {
- // logs command not supported in firefox
- // https://github.com/SeleniumHQ/selenium/issues/2910
- if (browser === 'chrome') {
- // check for console errors
- const errors = await checkBrowserForConsoleErrors()
- if (errors.length) {
- const errorReports = errors.map(err => err.message)
- const errorMessage = `Errors found in browser console:\n${errorReports.join('\n')}`
- console.error(new Error(errorMessage))
-
- }
- }
- // gather extra data if test failed
- if (this.currentTest.state === 'failed') {
- await verboseReportOnFailure({ browser, driver, title: this.currentTest.title })
- }
- })
-
- after(async function () {
- await driver.quit()
- })
-
- describe('Setup', function () {
-
- it('switches to Chrome extensions list', async function () {
- const windowHandles = await driver.getAllWindowHandles()
- await driver.switchTo().window(windowHandles[0])
- })
-
- it('does not select the new UI option', async () => {
- await delay(300)
- const button = await driver.findElement(By.xpath("//button[contains(text(), 'No thanks, maybe later')]"))
- await button.click()
- await delay(1000)
- })
-
- it('sets provider type to localhost', async function () {
- await delay(300)
- await setProviderType('localhost')
- })
-
- })
-
- describe('Account Creation', () => {
-
- it('matches MetaMask title', async () => {
- const title = await driver.getTitle()
- assert.equal(title, 'MetaMask', 'title matches MetaMask')
- await delay(300)
- })
-
- it('show terms of use', async () => {
- const terms = await driver.findElement(By.css('.terms-header')).getText()
- assert.equal(terms, 'TERMS OF USE', 'shows terms of use')
- delay(300)
- })
-
- it('checks if the TOU button is disabled', async () => {
- const button = await driver.findElement(By.css('button')).isEnabled()
- assert.equal(button, false, 'disabled continue button')
- const element = await driver.findElement(By.linkText('Attributions'))
- await driver.executeScript('arguments[0].scrollIntoView(true)', element)
- await delay(700)
- })
-
- it('allows the button to be clicked when scrolled to the bottom of TOU', async () => {
- const button = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div.flex-column.flex-center.flex-grow > button'))
- await button.click()
- await delay(300)
- })
-
- it('shows privacy notice', async () => {
- const privacy = await driver.findElement(By.css('.terms-header')).getText()
- assert.equal(privacy, 'PRIVACY NOTICE', 'shows privacy notice')
- await driver.findElement(By.css('button')).click()
- await delay(300)
- })
-
- it('shows phishing notice', async () => {
- const noticeHeader = await driver.findElement(By.css('.terms-header')).getText()
- assert.equal(noticeHeader, 'PHISHING WARNING', 'shows phishing warning')
- const element = await driver.findElement(By.css('.markdown'))
- await driver.executeScript('arguments[0].scrollTop = arguments[0].scrollHeight', element)
- await delay(300)
- await driver.findElement(By.css('button')).click()
- await delay(300)
- })
-
- it('accepts password with length of eight', async () => {
- const passwordBox = await driver.findElement(By.id('password-box'))
- const passwordBoxConfirm = await driver.findElement(By.id('password-box-confirm'))
- const button = await driver.findElements(By.css('button'))
-
- await passwordBox.sendKeys('123456789')
- await passwordBoxConfirm.sendKeys('123456789')
- await button[0].click()
- await delay(500)
- })
-
- it('shows value was created and seed phrase', async () => {
- await delay(300)
- const seedPhrase = await driver.findElement(By.css('.twelve-word-phrase')).getText()
- assert.equal(seedPhrase.split(' ').length, 12)
- const continueAfterSeedPhrase = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > button:nth-child(4)'))
- assert.equal(await continueAfterSeedPhrase.getText(), `I'VE COPIED IT SOMEWHERE SAFE`)
- await continueAfterSeedPhrase.click()
- await delay(300)
- })
-
- it('adds a second account', async function () {
- await driver.findElement(By.css('div.full-width > div > div:nth-child(2) > span > div')).click()
- await delay(300)
- await driver.findElement(By.css('div.full-width > div > div:nth-child(2) > span > div > div > span > div > li:nth-child(3) > span')).click()
- })
-
- it('shows account address', async function () {
- await delay(300)
- accountAddress = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > div > div:nth-child(1) > flex-column > div.flex-row > div')).getText()
- })
-
- it('logs out of the vault', async () => {
- await driver.findElement(By.css('.sandwich-expando')).click()
- await delay(500)
- const logoutButton = await driver.findElement(By.css('.menu-droppo > li:nth-child(3)'))
- assert.equal(await logoutButton.getText(), 'Log Out')
- await logoutButton.click()
- })
-
- it('accepts account password after lock', async () => {
- await delay(500)
- await driver.findElement(By.id('password-box')).sendKeys('123456789')
- await driver.findElement(By.id('password-box')).sendKeys(Key.ENTER)
- await delay(500)
- })
-
- it('shows QR code option', async () => {
- await delay(300)
- await driver.findElement(By.css('.fa-ellipsis-h')).click()
- await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div > div:nth-child(1) > flex-column > div.name-label > div > span > i > div > div > li:nth-child(3)')).click()
- await delay(300)
- })
-
- it('checks QR code address is the same as account details address', async () => {
- const QRaccountAddress = await driver.findElement(By.css('.ellip-address')).getText()
- assert.equal(accountAddress.toLowerCase(), QRaccountAddress)
- await driver.findElement(By.css('.fa-arrow-left')).click()
- await delay(500)
- })
- })
-
- describe('Import Ganache seed phrase', function () {
-
- it('logs out', async function () {
- await driver.findElement(By.css('.sandwich-expando')).click()
- await delay(200)
- const logOut = await driver.findElement(By.css('.menu-droppo > li:nth-child(3)'))
- assert.equal(await logOut.getText(), 'Log Out')
- await logOut.click()
- await delay(300)
- })
-
- it('restores from seed phrase', async function () {
- const restoreSeedLink = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > div.flex-row.flex-center.flex-grow > p'))
- assert.equal(await restoreSeedLink.getText(), 'Restore from seed phrase')
- await restoreSeedLink.click()
- await delay(100)
- })
-
- it('adds seed phrase', async function () {
- const testSeedPhrase = 'phrase upgrade clock rough situate wedding elder clever doctor stamp excess tent'
- const seedTextArea = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > textarea'))
- await seedTextArea.sendKeys(testSeedPhrase)
-
- await driver.findElement(By.id('password-box')).sendKeys('123456789')
- await driver.findElement(By.id('password-box-confirm')).sendKeys('123456789')
- await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > div > button:nth-child(2)')).click()
- await delay(500)
- })
-
- it('balance renders', async function () {
- await delay(500)
- const balance = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div > div.flex-row > div.ether-balance.ether-balance-amount > div > div > div:nth-child(1) > div:nth-child(1)'))
- assert.equal(await balance.getText(), '100.000')
- await delay(200)
- })
-
- it('sends transaction', async function () {
- const sendButton = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div > div.flex-row > button:nth-child(4)'))
- assert.equal(await sendButton.getText(), 'SEND')
- await sendButton.click()
- await delay(200)
- })
-
- it('adds recipient address and amount', async function () {
- const sendTranscationScreen = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > h3:nth-child(2)')).getText()
- assert.equal(sendTranscationScreen, 'SEND TRANSACTION')
- const inputAddress = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section:nth-child(3) > div > input'))
- const inputAmmount = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section:nth-child(4) > input'))
- await inputAddress.sendKeys('0x2f318C334780961FB129D2a6c30D0763d9a5C970')
- await inputAmmount.sendKeys('10')
- await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section:nth-child(4) > button')).click()
- await delay(300)
- })
-
- it('confirms transaction', async function () {
- await delay(300)
- const bySubmitButton = By.css('#pending-tx-form > div.flex-row.flex-space-around.conf-buttons > input')
- const submitButton = await driver.wait(until.elementLocated(bySubmitButton))
-
- submitButton.click()
-
- await delay(1500)
- })
-
- it('finds the transaction in the transactions list', async function () {
- const tranasactionAmount = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > section > section > div > div > div > div.ether-balance.ether-balance-amount > div > div > div > div:nth-child(1)'))
- assert.equal(await tranasactionAmount.getText(), '10.0')
- })
- })
-
- describe('Token Factory', function () {
-
- it('navigates to token factory', async function () {
- await driver.get('http://tokenfactory.surge.sh/')
- })
-
- it('navigates to create token contract link', async function () {
- const createToken = await driver.findElement(By.css('#bs-example-navbar-collapse-1 > ul > li:nth-child(3) > a'))
- await createToken.click()
- })
-
- it('adds input for token', async function () {
- const totalSupply = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(5) > input'))
- const tokenName = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(6) > input'))
- const tokenDecimal = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(7) > input'))
- const tokenSymbol = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > div:nth-child(8) > input'))
- const createToken = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > div > button'))
-
- await totalSupply.sendKeys('100')
- await tokenName.sendKeys('Test')
- await tokenDecimal.sendKeys('0')
- await tokenSymbol.sendKeys('TST')
- await createToken.click()
- await delay(1000)
- })
-
- // There is an issue with blank confirmation window in Firefox, but the button is still there and the driver is able to clicked (?.?)
- it('confirms transaction in MetaMask popup', async function () {
- const windowHandles = await driver.getAllWindowHandles()
- await driver.switchTo().window(windowHandles[windowHandles.length - 1])
- const byMetamaskSubmit = By.css('#pending-tx-form > div.flex-row.flex-space-around.conf-buttons > input')
- const metamaskSubmit = await driver.wait(until.elementLocated(byMetamaskSubmit))
- await metamaskSubmit.click()
- await delay(1000)
- })
-
- it('switches back to Token Factory to grab the token contract address', async function () {
- const windowHandles = await driver.getAllWindowHandles()
- await driver.switchTo().window(windowHandles[0])
- const tokenContactAddress = await driver.findElement(By.css('#main > div > div > div > div:nth-child(2) > span:nth-child(3)'))
- tokenAddress = await tokenContactAddress.getText()
- await delay(500)
- })
-
- it('navigates back to MetaMask popup in the tab', async function () {
- await driver.get(extensionUri)
- await delay(700)
- })
- })
-
- describe('Add Token', function () {
-
- it('switches to the add token screen', async function () {
- const tokensTab = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section > div > div.inactiveForm.pointer'))
- assert.equal(await tokensTab.getText(), 'TOKENS')
- await tokensTab.click()
- await delay(300)
- })
-
- it('navigates to the add token screen', async function () {
- const addTokenButton = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > section > div.full-flex-height > div > button'))
- assert.equal(await addTokenButton.getText(), 'ADD TOKEN')
- await addTokenButton.click()
- })
-
- it('checks add token screen rendered', async function () {
- const addTokenScreen = await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div.section-title.flex-row.flex-center > h2'))
- assert.equal(await addTokenScreen.getText(), 'ADD TOKEN')
- })
-
- it('adds token parameters', async function () {
- const tokenContractAddress = await driver.findElement(By.css('#token-address'))
- await tokenContractAddress.sendKeys(tokenAddress)
- await delay(300)
- await driver.findElement(By.css('#app-content > div > div.app-primary.from-right > div > div.flex-column.flex-justify-center.flex-grow.select-none > div > button')).click()
- await delay(200)
- })
-
- it('checks the token balance', async function () {
- const tokenBalance = await driver.findElement(By.css('#app-content > div > div.app-primary.from-left > div > section > div.full-flex-height > ol > li:nth-child(2) > h3'))
- assert.equal(await tokenBalance.getText(), '100 TST')
- })
- })
-
- async function setProviderType (type) {
- await driver.executeScript('window.metamask.setProviderType(arguments[0])', type)
- }
-
- async function checkBrowserForConsoleErrors () {
- const ignoredLogTypes = ['WARNING']
- const ignoredErrorMessages = [
- // React throws error warnings on "dataset", but still sets the data-* properties correctly
- 'Warning: Unknown prop `dataset` on ',
- // Third-party Favicon 404s show up as errors
- 'favicon.ico - Failed to load resource: the server responded with a status of 404 (Not Found)',
- // React Development build - known issue blocked by test build sys
- 'Warning: It looks like you\'re using a minified copy of the development build of React.',
- // Redux Development build - known issue blocked by test build sys
- 'This means that you are running a slower development build of Redux.',
- ]
- const browserLogs = await driver.manage().logs().get('browser')
- const errorEntries = browserLogs.filter(entry => !ignoredLogTypes.includes(entry.level.toString()))
- const errorObjects = errorEntries.map(entry => entry.toJSON())
- // ignore all errors that contain a message in `ignoredErrorMessages`
- const matchedErrorObjects = errorObjects.filter(entry => !ignoredErrorMessages.some(message => entry.message.includes(message)))
- return matchedErrorObjects
- }
-
-})
diff --git a/test/e2e/beta/run-all.sh b/test/e2e/run-all.sh
index 685feab00..d449221f2 100755
--- a/test/e2e/beta/run-all.sh
+++ b/test/e2e/run-all.sh
@@ -6,7 +6,7 @@ set -o pipefail
export PATH="$PATH:./node_modules/.bin"
-shell-parallel -s 'npm run ganache:start -- -b 2' -x 'sleep 5 && static-server test/e2e/beta/contract-test --port 8080' -x 'sleep 5 && mocha test/e2e/beta/metamask-beta-ui.spec'
-shell-parallel -s 'npm run ganache:start -- -b 2' -x 'sleep 5 && static-server test/e2e/beta/contract-test --port 8080' -x 'sleep 5 && mocha test/e2e/beta/metamask-beta-responsive-ui.spec'
+shell-parallel -s 'npm run ganache:start -- -b 2' -x 'sleep 5 && static-server test/e2e/contract-test --port 8080' -x 'sleep 5 && mocha test/e2e/metamask-ui.spec'
+shell-parallel -s 'npm run ganache:start -- -b 2' -x 'sleep 5 && static-server test/e2e/contract-test --port 8080' -x 'sleep 5 && mocha test/e2e/metamask-responsive-ui.spec'
shell-parallel -s 'npm run ganache:start -- -d -b 2 --account=0x53CB0AB5226EEBF4D872113D98332C1555DC304443BEE1CF759D15798D3C55A9,25000000000000000000' \
- -x 'sleep 5 && mocha test/e2e/beta/from-import-beta-ui.spec'
+ -x 'sleep 5 && mocha test/e2e/from-import-ui.spec'
diff --git a/test/e2e/beta/run-drizzle.sh b/test/e2e/run-drizzle.sh
index 0799b5a65..f3db2638a 100755
--- a/test/e2e/beta/run-drizzle.sh
+++ b/test/e2e/run-drizzle.sh
@@ -20,7 +20,7 @@ BROWSER=none npm start >> /dev/null 2>&1 &
npm_start_pid=$!
popd
-if ! mocha test/e2e/beta/drizzle.spec
+if ! mocha test/e2e/drizzle.spec
then
test_status=1
fi
diff --git a/test/e2e/beta/run-web3.sh b/test/e2e/run-web3.sh
index 9f77060de..60a4a8ea8 100755
--- a/test/e2e/beta/run-web3.sh
+++ b/test/e2e/run-web3.sh
@@ -6,4 +6,4 @@ set -o pipefail
export PATH="$PATH:./node_modules/.bin"
-shell-parallel -s 'static-server test/web3 --port 8080' -x 'sleep 5 && mocha test/e2e/beta/web3.spec' \ No newline at end of file
+shell-parallel -s 'static-server test/web3 --port 8080' -x 'sleep 5 && mocha test/e2e/web3.spec'
diff --git a/test/e2e/beta/web3.spec.js b/test/e2e/web3.spec.js
index b3962c821..e5be24672 100644
--- a/test/e2e/beta/web3.spec.js
+++ b/test/e2e/web3.spec.js
@@ -9,7 +9,7 @@ const {
installWebExt,
getExtensionIdChrome,
getExtensionIdFirefox,
-} = require('../func')
+} = require('./func')
const {
checkBrowserForConsoleErrors,
closeAllWindowHandlesExcept,