appium-uiautomator2-driver 1.62.0 → 1.65.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -134,7 +134,7 @@ appium:unlockSuccessTimeout | Maximum number of milliseconds to wait until the d
134
134
 
135
135
  Capability Name | Description
136
136
  --- | ---
137
- appium:mjpegServerPort | The number of the port UiAutomator2 server starts the MJPEG server on. If not provided then no screenshots broadcast thread is started on the server side.
137
+ appium:mjpegServerPort | The number of the port UiAutomator2 server starts the MJPEG server on. If not provided then the screenshots broadcasting service on the remote device does not get exposed to a local port (e.g. no adb port forwarding is happening)
138
138
  appium:mjpegScreenshotUrl | The URL of a service that provides realtime device screenshots in MJPEG format. If provided then the actual command to retrieve a screenshot will be requesting pictures from this service rather than directly from the server
139
139
 
140
140
  ### Web Context
@@ -314,6 +314,7 @@ UiAutomator2 provides several extensions that allow to automate popular mobile g
314
314
 
315
315
  - mobile: dragGesture
316
316
  - mobile: flingGesture
317
+ - mobile: doubleClickGesture
317
318
  - mobile: longClickGesture
318
319
  - mobile: pinchCloseGesture
319
320
  - mobile: pinchOpenGesture
@@ -365,7 +366,7 @@ Name | Type | Required | Description | Example
365
366
  --- | --- | --- | --- | ---
366
367
  buttonLabel | string | no | The name/text of the alert button to click in order to accept it. If not provided then the driver will try to autodetect it | Accept
367
368
 
368
- ### mobile: acceptAlert
369
+ ### mobile: dismissAlert
369
370
 
370
371
  Tries to dismiss an Android alert. This method might not always be reliable as there is no single standard for how Android alerts should look like within the Accessibility representation.
371
372
 
@@ -639,6 +640,17 @@ Note that `description` in `page` can be an empty string most likely when it com
639
640
  }
640
641
  ```
641
642
 
643
+ ### mobile: installMultipleApks
644
+
645
+ Install applications via `install-multiple` option.
646
+ Please read more details in the corresponding section of the `adb --help` command output.
647
+
648
+ #### Arguments
649
+
650
+ Name | Type | Required | Description | Example
651
+ --- | --- | --- | --- | ---
652
+ apks | array<string> | yes | The path to APKs. Each path should be the full path to the apk to be installed, or an URL to a remote location. | `['/path/to/local.apk', 'https://github.com/appium/ruby_lib_core/blob/master/test/functional/app/api.apk.zip?raw=true']`
653
+ options | object | no | Installation options. If you want enable `-g` option, you could specify that `{grantPermissions: true}`. `allowTestPackages` corresponds `-t`, `useSdcard` corresponds `-s`, `replace` corresponds `-r` (`-r` is enabled by default), `partialInstall` corresponds `-p`. | `{grantPermissions: true, partialInstall: true}`
642
654
 
643
655
  ## Applications Management
644
656
 
@@ -13,10 +13,14 @@ var _lodash = _interopRequireDefault(require("lodash"));
13
13
 
14
14
  var _logger = _interopRequireDefault(require("../logger"));
15
15
 
16
+ var _bluebird = _interopRequireDefault(require("bluebird"));
17
+
16
18
  var _appiumBaseDriver = require("appium-base-driver");
17
19
 
18
20
  var _appiumSupport = require("appium-support");
19
21
 
22
+ var _extensions = require("../extensions");
23
+
20
24
  let extensions = {},
21
25
  commands = {},
22
26
  helpers = {};
@@ -121,6 +125,7 @@ extensions.executeMobile = async function (mobileCommand, opts = {}) {
121
125
  execEmuConsoleCommand: 'mobileExecEmuConsoleCommand',
122
126
  dragGesture: 'mobileDragGesture',
123
127
  flingGesture: 'mobileFlingGesture',
128
+ doubleClickGesture: 'mobileDoubleClickGesture',
124
129
  longClickGesture: 'mobileLongClickGesture',
125
130
  pinchCloseGesture: 'mobilePinchCloseGesture',
126
131
  pinchOpenGesture: 'mobilePinchOpenGesture',
@@ -150,7 +155,8 @@ extensions.executeMobile = async function (mobileCommand, opts = {}) {
150
155
  deleteFile: 'mobileDeleteFile',
151
156
  startService: 'mobileStartService',
152
157
  stopService: 'mobileStopService',
153
- getContexts: 'mobileGetContexts'
158
+ getContexts: 'mobileGetContexts',
159
+ installMultipleApks: 'mobileInstallMultipleApks'
154
160
  };
155
161
 
156
162
  if (!_lodash.default.has(mobileCommandsMapping, mobileCommand)) {
@@ -248,9 +254,18 @@ commands.mobileType = async function mobileType(opts = {}) {
248
254
  return await this.adb.typeUnicode(text);
249
255
  };
250
256
 
257
+ commands.mobileInstallMultipleApks = async function (opts = {}) {
258
+ if (!_lodash.default.isArray(opts.apks) || _lodash.default.isEmpty(opts.apks)) {
259
+ throw new _appiumBaseDriver.errors.InvalidArgumentError('No apks are given to install');
260
+ }
261
+
262
+ const apks = await _bluebird.default.all(opts.apks.map(app => this.helpers.configureApp(app, [_extensions.APK_EXTENSION])));
263
+ await this.adb.installMultipleApks(apks, opts.options);
264
+ };
265
+
251
266
  Object.assign(extensions, commands, helpers);
252
267
  var _default = extensions;
253
268
  exports.default = _default;require('source-map-support').install();
254
269
 
255
270
 
256
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/general.js"],"names":["extensions","commands","helpers","getPageSource","uiautomator2","jwproxy","command","getClipboard","adb","getApiLevel","doSendKeys","params","keyevent","keycode","metastate","log","debug","back","getStrings","language","getDeviceLanguage","info","preprocessStringsMap","mapping","result","key","value","_","toPairs","isString","JSON","stringify","apkStrings","opts","app","appPackage","errorAndThrow","tmpRoot","tempDir","openDir","pullApk","err","message","fs","exists","extractStringsFromApk","rimraf","getWindowSize","getWindowRect","width","height","x","y","executeMobile","mobileCommand","mobileCommandsMapping","shell","execEmuConsoleCommand","dragGesture","flingGesture","longClickGesture","pinchCloseGesture","pinchOpenGesture","swipeGesture","scrollGesture","scrollBackTo","scroll","viewportScreenshot","viewportRect","deepLink","startLogsBroadcast","stopLogsBroadcast","acceptAlert","dismissAlert","batteryInfo","deviceInfo","getDeviceTime","changePermissions","getPermissions","performEditorAction","startScreenStreaming","stopScreenStreaming","getNotifications","listSms","type","sensorSet","deleteFile","startService","stopService","getContexts","has","errors","UnknownCommandError","keys","mobileViewportScreenshot","getViewportScreenshot","mobileViewPortRect","getViewPortRect","setUrl","url","startUri","mobileDeepLink","package","pkg","waitForLaunch","openNotifications","updateSettings","settings","driverOnlySettings","serverSettings","setting","BASEDRIVER_HANDLED_SETTINGS","includes","isEmpty","update","getSettings","wrapBootstrapDisconnect","wrapped","suspendChromedriverProxy","chromedriver","proxyReqRes","bind","jwpProxyActive","mobileGetDeviceInfo","mobileType","text","isUndefined","typeUnicode","Object","assign"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from '../logger';\nimport { errors, BASEDRIVER_HANDLED_SETTINGS } from 'appium-base-driver';\nimport { fs, tempDir } from 'appium-support';\n\nlet extensions = {},\n    commands = {},\n    helpers = {};\n\ncommands.getPageSource = async function () {\n  return await this.uiautomator2.jwproxy.command('/source', 'GET', {});\n};\n\ncommands.getClipboard = async function () {\n  return (await this.adb.getApiLevel() < 29)\n    ? (await this.uiautomator2.jwproxy.command('/appium/device/get_clipboard', 'POST', {}))\n    : (await this.adb.getClipboard());\n};\n\n// Need to override this for correct unicode support\ncommands.doSendKeys = async function (params) {\n  await this.uiautomator2.jwproxy.command('/keys', 'POST', params);\n};\n\n// uiautomator2 doesn't support metastate for keyevents\ncommands.keyevent = async function (keycode, metastate) {\n  log.debug(`Ignoring metastate ${metastate}`);\n  await this.adb.keyevent(keycode);\n};\n\n// Use ADB since we don't have UiAutomator\ncommands.back = async function () {\n  await this.adb.keyevent(4);\n};\n\ncommands.getStrings = async function (language) {\n  if (!language) {\n    language = await this.adb.getDeviceLanguage();\n    log.info(`No language specified, returning strings for: ${language}`);\n  }\n\n  // Clients require the resulting mapping to have both keys\n  // and values of type string\n  const preprocessStringsMap = function (mapping) {\n    const result = {};\n    for (const [key, value] of _.toPairs(mapping)) {\n      result[key] = _.isString(value) ? value : JSON.stringify(value);\n    }\n    return result;\n  };\n\n  if (this.apkStrings[language]) {\n    // Return cached strings\n    return preprocessStringsMap(this.apkStrings[language]);\n  }\n\n  if (!this.opts.app && !this.opts.appPackage) {\n    log.errorAndThrow(\"One of 'app' or 'appPackage' capabilities should must be specified\");\n  }\n\n  let app = this.opts.app;\n  const tmpRoot = await tempDir.openDir();\n  try {\n    if (!app) {\n      try {\n        app = await this.adb.pullApk(this.opts.appPackage, tmpRoot);\n      } catch (err) {\n        log.errorAndThrow(`Failed to pull an apk from '${this.opts.appPackage}'. Original error: ${err.message}`);\n      }\n    }\n\n    if (!await fs.exists(app)) {\n      log.errorAndThrow(`The app at '${app}' does not exist`);\n    }\n\n    try {\n      const {apkStrings} = await this.adb.extractStringsFromApk(app, language, tmpRoot);\n      this.apkStrings[language] = apkStrings;\n      return preprocessStringsMap(apkStrings);\n    } catch (err) {\n      log.errorAndThrow(`Cannot extract strings from '${app}'. Original error: ${err.message}`);\n    }\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n};\n\n// memoized in constructor\ncommands.getWindowSize = async function () {\n  return await this.uiautomator2.jwproxy.command('/window/current/size', 'GET', {});\n};\n\n// For W3C\ncommands.getWindowRect = async function () {\n  const {width, height} = await this.getWindowSize();\n  return {\n    width,\n    height,\n    x: 0,\n    y: 0,\n  };\n};\n\nextensions.executeMobile = async function (mobileCommand, opts = {}) {\n  const mobileCommandsMapping = {\n    shell: 'mobileShell',\n\n    execEmuConsoleCommand: 'mobileExecEmuConsoleCommand',\n\n    dragGesture: 'mobileDragGesture',\n    flingGesture: 'mobileFlingGesture',\n    longClickGesture: 'mobileLongClickGesture',\n    pinchCloseGesture: 'mobilePinchCloseGesture',\n    pinchOpenGesture: 'mobilePinchOpenGesture',\n    swipeGesture: 'mobileSwipeGesture',\n    scrollGesture: 'mobileScrollGesture',\n    scrollBackTo: 'mobileScrollBackTo',\n    scroll: 'mobileScroll',\n    viewportScreenshot: 'mobileViewportScreenshot',\n    viewportRect: 'mobileViewPortRect',\n\n    deepLink: 'mobileDeepLink',\n\n    startLogsBroadcast: 'mobileStartLogsBroadcast',\n    stopLogsBroadcast: 'mobileStopLogsBroadcast',\n\n    acceptAlert: 'mobileAcceptAlert',\n    dismissAlert: 'mobileDismissAlert',\n\n    batteryInfo: 'mobileGetBatteryInfo',\n\n    deviceInfo: 'mobileGetDeviceInfo',\n\n    getDeviceTime: 'mobileGetDeviceTime',\n\n    changePermissions: 'mobileChangePermissions',\n    getPermissions: 'mobileGetPermissions',\n\n    performEditorAction: 'mobilePerformEditorAction',\n\n    startScreenStreaming: 'mobileStartScreenStreaming',\n    stopScreenStreaming: 'mobileStopScreenStreaming',\n\n    getNotifications: 'mobileGetNotifications',\n\n    listSms: 'mobileListSms',\n\n    type: 'mobileType',\n    sensorSet: 'sensorSet',\n\n    deleteFile: 'mobileDeleteFile',\n\n    startService: 'mobileStartService',\n    stopService: 'mobileStopService',\n\n    getContexts: 'mobileGetContexts',\n  };\n\n  if (!_.has(mobileCommandsMapping, mobileCommand)) {\n    throw new errors.UnknownCommandError(`Unknown mobile command \"${mobileCommand}\". ` +\n      `Only ${_.keys(mobileCommandsMapping)} commands are supported.`);\n  }\n  return await this[mobileCommandsMapping[mobileCommand]](opts);\n};\n\ncommands.mobileViewportScreenshot = async function () {\n  return await this.getViewportScreenshot();\n};\n\n/**\n * @typedef {object} Rectangle\n * @property {number} left - The left coordinate of the Rectangle.\n * @property {number} top - The top coordinate of the Rectangle.\n * @property {number} width - The width of Rectangle.\n * @property {number} height - The height of Rectangle.\n */\n\n/**\n * Returns the viewport coordinates.\n * @returns {Rectangle} The viewport coordinates.\n */\ncommands.mobileViewPortRect = async function mobileViewPortRect () {\n  return await this.getViewPortRect();\n};\n\ncommands.setUrl = async function (url) {\n  await this.adb.startUri(url, this.opts.appPackage);\n};\n\n/**\n * @typedef {object} DeepLinkOpts\n * @property {!string} url - The name of URL to start.\n * @property {!string} package - The name of the package to start the URI with.\n * @property {?boolean} waitForLaunch [true] - if `false` then adb won't wait\n * for the started activity to return the control\n */\n\n/**\n * Start URL that take users directly to specific content in the app\n * @param {DeepLinkOpts} opts\n */\ncommands.mobileDeepLink = async function (opts = {}) {\n  const {\n    url,\n    package: pkg,\n    waitForLaunch,\n  } = opts;\n  return await this.adb.startUri(url, pkg, { waitForLaunch });\n};\n\ncommands.openNotifications = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/open_notifications', 'POST', {});\n};\n\ncommands.updateSettings = async function (settings) {\n  // we have some settings that are set on the settings object in the driver\n  // only, for example image finding settings. The uiauto2 server does not know\n  // what to do with them, so just set them on this driver's settings instance,\n  // and don't forward them to the server\n  let driverOnlySettings = {};\n  let serverSettings = {};\n  for (let [setting, value] of _.toPairs(settings)) {\n    if (BASEDRIVER_HANDLED_SETTINGS.includes(setting)) {\n      driverOnlySettings[setting] = value;\n    } else {\n      serverSettings[setting] = value;\n    }\n  }\n  if (!_.isEmpty(driverOnlySettings)) {\n    log.info(`Found some settings designed to be handled by BaseDriver: ` +\n             `${JSON.stringify(_.keys(driverOnlySettings))}. Not ` +\n             `sending these on to the UiAutomator2 server and instead ` +\n             `setting directly on the driver`);\n    await this.settings.update(driverOnlySettings);\n  }\n  if (!_.isEmpty(serverSettings)) {\n    log.info('Forwarding the following settings to the UiAutomator2 server: ' +\n             JSON.stringify(_.keys(serverSettings)));\n    await this.uiautomator2.jwproxy.command('/appium/settings', 'POST',\n      {settings: serverSettings});\n  }\n};\n\ncommands.getSettings = async function () {\n  // as above, we might have some driver-only settings to return as well\n  const driverOnlySettings = this.settings.getSettings();\n  const serverSettings = await this.uiautomator2.jwproxy.command('/appium/settings', 'GET');\n  return {...driverOnlySettings, ...serverSettings};\n};\n\n/**\n * Overriding appium-android-driver's wrapBootstrapDisconnect,\n * unlike in appium-android-driver avoiding adb restarting as it intern\n * kills UiAutomator2 server running in the device.\n **/\nhelpers.wrapBootstrapDisconnect = async function (wrapped) {\n  await wrapped();\n};\n\n// Stop proxying to any Chromedriver and redirect to uiautomator2\nhelpers.suspendChromedriverProxy = function () {\n  this.chromedriver = null;\n  this.proxyReqRes = this.uiautomator2.proxyReqRes.bind(this.uiautomator2);\n  this.jwpProxyActive = true;\n};\n\n/**\n * The list of available info entries can be found at\n * https://github.com/appium/appium-uiautomator2-server/blob/master/app/src/main/java/io/appium/uiautomator2/handler/GetDeviceInfo.java\n */\ncommands.mobileGetDeviceInfo = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/info', 'GET');\n};\n\n/**\n * @typedef {Object} TypingOptions\n * @property {!string|number|boolean} text - The text to type\n */\n\n/**\n * Types the given Unicode string.\n * It is expected that the focus is already put\n * to the destination input field before this method is called.\n *\n * @param {TypingOptions} opts\n * @returns {boolean} `true` if the input text has been successfully sent to adb\n * @throws {Error} if `text` property has not been provided\n */\ncommands.mobileType = async function mobileType (opts = {}) {\n  const {\n    text,\n  } = opts;\n  if (_.isUndefined(text)) {\n    log.errorAndThrow(`The 'text' argument is mandatory`);\n  }\n  return await this.adb.typeUnicode(text);\n};\n\nObject.assign(extensions, commands, helpers);\n\nexport default extensions;\n"],"file":"lib/commands/general.js","sourceRoot":"../../.."}
271
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/general.js"],"names":["extensions","commands","helpers","getPageSource","uiautomator2","jwproxy","command","getClipboard","adb","getApiLevel","doSendKeys","params","keyevent","keycode","metastate","log","debug","back","getStrings","language","getDeviceLanguage","info","preprocessStringsMap","mapping","result","key","value","_","toPairs","isString","JSON","stringify","apkStrings","opts","app","appPackage","errorAndThrow","tmpRoot","tempDir","openDir","pullApk","err","message","fs","exists","extractStringsFromApk","rimraf","getWindowSize","getWindowRect","width","height","x","y","executeMobile","mobileCommand","mobileCommandsMapping","shell","execEmuConsoleCommand","dragGesture","flingGesture","doubleClickGesture","longClickGesture","pinchCloseGesture","pinchOpenGesture","swipeGesture","scrollGesture","scrollBackTo","scroll","viewportScreenshot","viewportRect","deepLink","startLogsBroadcast","stopLogsBroadcast","acceptAlert","dismissAlert","batteryInfo","deviceInfo","getDeviceTime","changePermissions","getPermissions","performEditorAction","startScreenStreaming","stopScreenStreaming","getNotifications","listSms","type","sensorSet","deleteFile","startService","stopService","getContexts","installMultipleApks","has","errors","UnknownCommandError","keys","mobileViewportScreenshot","getViewportScreenshot","mobileViewPortRect","getViewPortRect","setUrl","url","startUri","mobileDeepLink","package","pkg","waitForLaunch","openNotifications","updateSettings","settings","driverOnlySettings","serverSettings","setting","BASEDRIVER_HANDLED_SETTINGS","includes","isEmpty","update","getSettings","wrapBootstrapDisconnect","wrapped","suspendChromedriverProxy","chromedriver","proxyReqRes","bind","jwpProxyActive","mobileGetDeviceInfo","mobileType","text","isUndefined","typeUnicode","mobileInstallMultipleApks","isArray","apks","InvalidArgumentError","B","all","map","configureApp","APK_EXTENSION","options","Object","assign"],"mappings":"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","sourcesContent":["import _ from 'lodash';\nimport log from '../logger';\nimport B from 'bluebird';\nimport { errors, BASEDRIVER_HANDLED_SETTINGS } from 'appium-base-driver';\nimport { fs, tempDir } from 'appium-support';\nimport { APK_EXTENSION } from '../extensions';\n\nlet extensions = {},\n    commands = {},\n    helpers = {};\n\ncommands.getPageSource = async function () {\n  return await this.uiautomator2.jwproxy.command('/source', 'GET', {});\n};\n\ncommands.getClipboard = async function () {\n  return (await this.adb.getApiLevel() < 29)\n    ? (await this.uiautomator2.jwproxy.command('/appium/device/get_clipboard', 'POST', {}))\n    : (await this.adb.getClipboard());\n};\n\n// Need to override this for correct unicode support\ncommands.doSendKeys = async function (params) {\n  await this.uiautomator2.jwproxy.command('/keys', 'POST', params);\n};\n\n// uiautomator2 doesn't support metastate for keyevents\ncommands.keyevent = async function (keycode, metastate) {\n  log.debug(`Ignoring metastate ${metastate}`);\n  await this.adb.keyevent(keycode);\n};\n\n// Use ADB since we don't have UiAutomator\ncommands.back = async function () {\n  await this.adb.keyevent(4);\n};\n\ncommands.getStrings = async function (language) {\n  if (!language) {\n    language = await this.adb.getDeviceLanguage();\n    log.info(`No language specified, returning strings for: ${language}`);\n  }\n\n  // Clients require the resulting mapping to have both keys\n  // and values of type string\n  const preprocessStringsMap = function (mapping) {\n    const result = {};\n    for (const [key, value] of _.toPairs(mapping)) {\n      result[key] = _.isString(value) ? value : JSON.stringify(value);\n    }\n    return result;\n  };\n\n  if (this.apkStrings[language]) {\n    // Return cached strings\n    return preprocessStringsMap(this.apkStrings[language]);\n  }\n\n  if (!this.opts.app && !this.opts.appPackage) {\n    log.errorAndThrow(\"One of 'app' or 'appPackage' capabilities should must be specified\");\n  }\n\n  let app = this.opts.app;\n  const tmpRoot = await tempDir.openDir();\n  try {\n    if (!app) {\n      try {\n        app = await this.adb.pullApk(this.opts.appPackage, tmpRoot);\n      } catch (err) {\n        log.errorAndThrow(`Failed to pull an apk from '${this.opts.appPackage}'. Original error: ${err.message}`);\n      }\n    }\n\n    if (!await fs.exists(app)) {\n      log.errorAndThrow(`The app at '${app}' does not exist`);\n    }\n\n    try {\n      const {apkStrings} = await this.adb.extractStringsFromApk(app, language, tmpRoot);\n      this.apkStrings[language] = apkStrings;\n      return preprocessStringsMap(apkStrings);\n    } catch (err) {\n      log.errorAndThrow(`Cannot extract strings from '${app}'. Original error: ${err.message}`);\n    }\n  } finally {\n    await fs.rimraf(tmpRoot);\n  }\n};\n\n// memoized in constructor\ncommands.getWindowSize = async function () {\n  return await this.uiautomator2.jwproxy.command('/window/current/size', 'GET', {});\n};\n\n// For W3C\ncommands.getWindowRect = async function () {\n  const {width, height} = await this.getWindowSize();\n  return {\n    width,\n    height,\n    x: 0,\n    y: 0,\n  };\n};\n\nextensions.executeMobile = async function (mobileCommand, opts = {}) {\n  const mobileCommandsMapping = {\n    shell: 'mobileShell',\n\n    execEmuConsoleCommand: 'mobileExecEmuConsoleCommand',\n\n    dragGesture: 'mobileDragGesture',\n    flingGesture: 'mobileFlingGesture',\n    doubleClickGesture: 'mobileDoubleClickGesture',\n    longClickGesture: 'mobileLongClickGesture',\n    pinchCloseGesture: 'mobilePinchCloseGesture',\n    pinchOpenGesture: 'mobilePinchOpenGesture',\n    swipeGesture: 'mobileSwipeGesture',\n    scrollGesture: 'mobileScrollGesture',\n    scrollBackTo: 'mobileScrollBackTo',\n    scroll: 'mobileScroll',\n    viewportScreenshot: 'mobileViewportScreenshot',\n    viewportRect: 'mobileViewPortRect',\n\n    deepLink: 'mobileDeepLink',\n\n    startLogsBroadcast: 'mobileStartLogsBroadcast',\n    stopLogsBroadcast: 'mobileStopLogsBroadcast',\n\n    acceptAlert: 'mobileAcceptAlert',\n    dismissAlert: 'mobileDismissAlert',\n\n    batteryInfo: 'mobileGetBatteryInfo',\n\n    deviceInfo: 'mobileGetDeviceInfo',\n\n    getDeviceTime: 'mobileGetDeviceTime',\n\n    changePermissions: 'mobileChangePermissions',\n    getPermissions: 'mobileGetPermissions',\n\n    performEditorAction: 'mobilePerformEditorAction',\n\n    startScreenStreaming: 'mobileStartScreenStreaming',\n    stopScreenStreaming: 'mobileStopScreenStreaming',\n\n    getNotifications: 'mobileGetNotifications',\n\n    listSms: 'mobileListSms',\n\n    type: 'mobileType',\n    sensorSet: 'sensorSet',\n\n    deleteFile: 'mobileDeleteFile',\n\n    startService: 'mobileStartService',\n    stopService: 'mobileStopService',\n\n    getContexts: 'mobileGetContexts',\n\n    installMultipleApks: 'mobileInstallMultipleApks',\n  };\n\n  if (!_.has(mobileCommandsMapping, mobileCommand)) {\n    throw new errors.UnknownCommandError(`Unknown mobile command \"${mobileCommand}\". ` +\n      `Only ${_.keys(mobileCommandsMapping)} commands are supported.`);\n  }\n  return await this[mobileCommandsMapping[mobileCommand]](opts);\n};\n\ncommands.mobileViewportScreenshot = async function () {\n  return await this.getViewportScreenshot();\n};\n\n/**\n * @typedef {object} Rectangle\n * @property {number} left - The left coordinate of the Rectangle.\n * @property {number} top - The top coordinate of the Rectangle.\n * @property {number} width - The width of Rectangle.\n * @property {number} height - The height of Rectangle.\n */\n\n/**\n * Returns the viewport coordinates.\n * @returns {Rectangle} The viewport coordinates.\n */\ncommands.mobileViewPortRect = async function mobileViewPortRect () {\n  return await this.getViewPortRect();\n};\n\ncommands.setUrl = async function (url) {\n  await this.adb.startUri(url, this.opts.appPackage);\n};\n\n/**\n * @typedef {object} DeepLinkOpts\n * @property {!string} url - The name of URL to start.\n * @property {!string} package - The name of the package to start the URI with.\n * @property {?boolean} waitForLaunch [true] - if `false` then adb won't wait\n * for the started activity to return the control\n */\n\n/**\n * Start URL that take users directly to specific content in the app\n * @param {DeepLinkOpts} opts\n */\ncommands.mobileDeepLink = async function (opts = {}) {\n  const {\n    url,\n    package: pkg,\n    waitForLaunch,\n  } = opts;\n  return await this.adb.startUri(url, pkg, { waitForLaunch });\n};\n\ncommands.openNotifications = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/open_notifications', 'POST', {});\n};\n\ncommands.updateSettings = async function (settings) {\n  // we have some settings that are set on the settings object in the driver\n  // only, for example image finding settings. The uiauto2 server does not know\n  // what to do with them, so just set them on this driver's settings instance,\n  // and don't forward them to the server\n  let driverOnlySettings = {};\n  let serverSettings = {};\n  for (let [setting, value] of _.toPairs(settings)) {\n    if (BASEDRIVER_HANDLED_SETTINGS.includes(setting)) {\n      driverOnlySettings[setting] = value;\n    } else {\n      serverSettings[setting] = value;\n    }\n  }\n  if (!_.isEmpty(driverOnlySettings)) {\n    log.info(`Found some settings designed to be handled by BaseDriver: ` +\n             `${JSON.stringify(_.keys(driverOnlySettings))}. Not ` +\n             `sending these on to the UiAutomator2 server and instead ` +\n             `setting directly on the driver`);\n    await this.settings.update(driverOnlySettings);\n  }\n  if (!_.isEmpty(serverSettings)) {\n    log.info('Forwarding the following settings to the UiAutomator2 server: ' +\n             JSON.stringify(_.keys(serverSettings)));\n    await this.uiautomator2.jwproxy.command('/appium/settings', 'POST',\n      {settings: serverSettings});\n  }\n};\n\ncommands.getSettings = async function () {\n  // as above, we might have some driver-only settings to return as well\n  const driverOnlySettings = this.settings.getSettings();\n  const serverSettings = await this.uiautomator2.jwproxy.command('/appium/settings', 'GET');\n  return {...driverOnlySettings, ...serverSettings};\n};\n\n/**\n * Overriding appium-android-driver's wrapBootstrapDisconnect,\n * unlike in appium-android-driver avoiding adb restarting as it intern\n * kills UiAutomator2 server running in the device.\n **/\nhelpers.wrapBootstrapDisconnect = async function (wrapped) {\n  await wrapped();\n};\n\n// Stop proxying to any Chromedriver and redirect to uiautomator2\nhelpers.suspendChromedriverProxy = function () {\n  this.chromedriver = null;\n  this.proxyReqRes = this.uiautomator2.proxyReqRes.bind(this.uiautomator2);\n  this.jwpProxyActive = true;\n};\n\n/**\n * The list of available info entries can be found at\n * https://github.com/appium/appium-uiautomator2-server/blob/master/app/src/main/java/io/appium/uiautomator2/handler/GetDeviceInfo.java\n */\ncommands.mobileGetDeviceInfo = async function () {\n  return await this.uiautomator2.jwproxy.command('/appium/device/info', 'GET');\n};\n\n/**\n * @typedef {Object} TypingOptions\n * @property {!string|number|boolean} text - The text to type\n */\n\n/**\n * Types the given Unicode string.\n * It is expected that the focus is already put\n * to the destination input field before this method is called.\n *\n * @param {TypingOptions} opts\n * @returns {boolean} `true` if the input text has been successfully sent to adb\n * @throws {Error} if `text` property has not been provided\n */\ncommands.mobileType = async function mobileType (opts = {}) {\n  const {\n    text,\n  } = opts;\n  if (_.isUndefined(text)) {\n    log.errorAndThrow(`The 'text' argument is mandatory`);\n  }\n  return await this.adb.typeUnicode(text);\n};\n\n\n/**\n * @typedef {Object} InstallOptions\n * @property {boolean} allowTestPackages [false] - Set to true in order to allow test\n *                                                 packages installation.\n * @property {boolean} useSdcard [false] - Set to true to install the app on sdcard\n *                                         instead of the device memory.\n * @property {boolean} grantPermissions [false] - Set to true in order to grant all the\n *                                                permissions requested in the application's manifest\n *                                                automatically after the installation is completed\n *                                                under Android 6+.\n * @property {boolean} replace [true] - Set it to false if you don't want\n *                                      the application to be upgraded/reinstalled\n *                                      if it is already present on the device.\n * @property {boolean} partialInstall [false] - Install apks partially. It is used for 'install-multiple'.\n *                                             https://android.stackexchange.com/questions/111064/what-is-a-partial-application-install-via-adb\n */\n\n/**\n * @typedef {Object} InstallMultipleApksOptions\n * @property {Array<string>} apks - The list of APKs to install. Each APK should be a path to a apk\n *                                  or downloadable URL as HTTP/HTTPS.\n * @property {InstallOptions} options\n */\n\n/**\n * Install multiple APKs with `install-multiple` option.\n *\n * @param {InstallMultipleApksOptions} opts\n * @throws {Error} if an error occured while installing the given APKs.\n */\ncommands.mobileInstallMultipleApks = async function (opts = {}) {\n  if (!_.isArray(opts.apks) || _.isEmpty(opts.apks)) {\n    throw new errors.InvalidArgumentError('No apks are given to install');\n  }\n  const apks = await B.all(opts.apks\n    .map((app) => this.helpers.configureApp(app, [APK_EXTENSION])));\n  await this.adb.installMultipleApks(apks, opts.options);\n};\n\nObject.assign(extensions, commands, helpers);\n\nexport default extensions;\n"],"file":"lib/commands/general.js","sourceRoot":"../../.."}
@@ -55,6 +55,18 @@ commands.mobileLongClickGesture = async function mobileLongClickGesture(opts = {
55
55
  });
56
56
  };
57
57
 
58
+ commands.mobileDoubleClickGesture = async function mobileDoubleClickGesture(opts = {}) {
59
+ const {
60
+ elementId,
61
+ x,
62
+ y
63
+ } = opts;
64
+ return await this.uiautomator2.jwproxy.command('/appium/gestures/double_click', 'POST', {
65
+ origin: toOrigin(elementId),
66
+ offset: toPoint(x, y)
67
+ });
68
+ };
69
+
58
70
  commands.mobileDragGesture = async function mobileDragGesture(opts = {}) {
59
71
  const {
60
72
  elementId,
@@ -205,4 +217,4 @@ var _default = commands;
205
217
  exports.default = _default;require('source-map-support').install();
206
218
 
207
219
 
208
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/gestures.js"],"names":["commands","toOrigin","element","util","wrapElement","unwrapElement","undefined","toPoint","x","y","_","isFinite","toRect","left","top","width","height","some","v","mobileLongClickGesture","opts","elementId","duration","uiautomator2","jwproxy","command","origin","offset","mobileDragGesture","startX","startY","endX","endY","speed","start","end","mobileFlingGesture","direction","area","mobilePinchCloseGesture","percent","mobilePinchOpenGesture","mobileSwipeGesture","mobileScrollGesture","mobileScrollBackTo","elementToId","errors","InvalidArgumentError","mobileScroll","strategy","selector","maxSwipes","params"],"mappings":"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","sourcesContent":["import { util } from 'appium-support';\nimport _ from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst commands = {};\n\n\nfunction toOrigin (element) {\n  return element ? util.wrapElement(util.unwrapElement(element)) : undefined;\n}\n\nfunction toPoint (x, y) {\n  return _.isFinite(x) && _.isFinite(y) ? {x, y} : undefined;\n}\n\nfunction toRect (left, top, width, height) {\n  if ([left, top, width, height].some((v) => !_.isFinite(v))) {\n    return undefined;\n  }\n  return {left, top, width, height};\n}\n\n\n/**\n * @typedef {Object} LongClickOptions\n * @property {?string} elementId - The id of the element to be clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to click on\n * @property {?number} y - The y coordinate to click on\n * @property {?number} duration [500] - Click duration in milliseconds.\n * The value must not be negative\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?LongClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileLongClickGesture = async function mobileLongClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n    duration,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/long_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n    duration,\n  });\n};\n\n/**\n * @typedef {Object} DragOptions\n * @property {?string} elementId - The id of the element to be dragged.\n * If the element id is missing then the start coordinates must be provided.\n * If both the element id and the start coordinates are provided then these\n * coordinates are considered as offsets from the top left element corner.\n * @property {?number} startX - The x coordinate where the dragging starts\n * @property {?number} startY - The y coordinate where the dragging starts\n * @property {!number} endX - The x coordinate where the dragging ends\n * @property {!number} endY - The y coordinate where the dragging ends\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Drags this object to the specified location.\n *\n * @param {?DragOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDragGesture = async function mobileDragGesture (opts = {}) {\n  const {\n    elementId,\n    startX, startY,\n    endX, endY,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/drag', 'POST', {\n    origin: toOrigin(elementId),\n    start: toPoint(startX, startY),\n    end: toPoint(endX, endY),\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} FlingOptions\n * @property {?string} elementId - The id of the element to be flinged.\n * If the element id is missing then fling bounding area must be provided.\n * If both the element id and the fling bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the fling bounding area\n * @property {?number} top - The top coordinate of the fling bounding area\n * @property {?number} width - The width of the fling bounding area\n * @property {?number} height - The height of the fling bounding area\n * @property {!string} direction - Direction of the fling.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {?number} speed [7500 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must be greater than the minimum fling\n * velocity for the given view (50 by default)\n */\n\n/**\n * Drags to the specified location.\n *\n * @param {?FlingOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileFlingGesture = async function mobileFlingGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/fling', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} PinchOptions\n * @property {?string} elementId - The id of the element to be pinched.\n * If the element id is missing then pinch bounding area must be provided.\n * If both the element id and the pinch bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the pinch bounding area\n * @property {?number} top - The top coordinate of the pinch bounding area\n * @property {?number} width - The width of the pinch bounding area\n * @property {?number} height - The height of the pinch bounding area\n * @property {!number} percent - The size of the pinch as a percentage of the pinch area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a pinch close gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchCloseGesture = async function mobilePinchCloseGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_close', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * Performs a pinch open gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchOpenGesture = async function mobilePinchOpenGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_open', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} SwipeOptions\n * @property {?string} elementId - The id of the element to be swiped.\n * If the element id is missing then swipe bounding area must be provided.\n * If both the element id and the swipe bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the swipe bounding area\n * @property {?number} top - The top coordinate of the swipe bounding area\n * @property {?number} width - The width of the swipe bounding area\n * @property {?number} height - The height of the swipe bounding area\n * @property {!string} direction - Direction of the swipe.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the swipe as a percentage of the swipe area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a swipe gesture.\n *\n * @param {?SwipeOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileSwipeGesture = async function mobileSwipeGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/swipe', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollOptions\n * @property {?string} elementId - The id of the element to be scrolled.\n * If the element id is missing then scroll bounding area must be provided.\n * If both the element id and the scroll bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the scroll bounding area\n * @property {?number} top - The top coordinate of the scroll bounding area\n * @property {?number} width - The width of the scroll bounding area\n * @property {?number} height - The height of the scroll bounding area\n * @property {!string} direction - Direction of the scroll.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the scroll as a percentage of the scrolling area size.\n * Valid values must be float numbers greater than zero, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this gesture\n * in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a scroll gesture.\n *\n * @param {?ScrollOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileScrollGesture = async function mobileScrollGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/scroll', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollElementToElementOpts\n * @property {string} elementId The identifier of the scrollable element,\n * which is going to be scrolled. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy.\n * @property {string} elementToId The identifier of the item, which belongs\n * to the scrollable element above, and which should become visible after\n * the scrolling operation is finished. It is required this element\n * was located by `-android uiautomator` strategy.\n */\n\n/**\n * Scrolls the given scrollable element `elementId` until `elementToId`\n * becomes visible. This function returns immediately if the `elementToId`\n * is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollElementToElementOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScrollBackTo = async function (opts = {}) {\n  const {elementId, elementToId} = opts;\n  if (!elementId || !elementToId) {\n    throw new errors.InvalidArgumentError(`Both elementId and elementToId arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command(\n    `/appium/element/${util.unwrapElement(elementId)}/scroll_to/${util.unwrapElement(elementToId)}`, 'POST', {});\n};\n\n/**\n * @typedef {Object} ScrollOpts\n * @property {?string} element The identifier of an element. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy. If this property is not provided then the first currently available scrollable view\n * is selected for the interaction.\n * @property {!string} strategy The following strategies are supported:\n * - `accessibility id` (UiSelector().description)\n * - `class name` (UiSelector().className)\n * - `-android uiautomator` (UiSelector)\n * @property {!string} selector The corresponding lookup value for the given\n * strategy.\n * @property {?number} maxSwipes The maximum number of swipes to perform\n * on the target scrollable view in order to reach the destination element.\n * In case this value is unset then it would be retrieved from the scrollable\n * element itself (vua `getMaxSearchSwipes()` property).\n */\n\n/**\n * Scrolls the given scrollable element until the element identified\n * by `strategy` and `selector` becomes visible. This function returns immediately if the\n * destination element is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScroll = async function (opts = {}) {\n  const {element, strategy, selector, maxSwipes} = opts;\n  if (!strategy || !selector) {\n    throw new errors.InvalidArgumentError(`Both strategy and selector arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command('/touch/scroll', 'POST', {\n    params: {\n      origin: toOrigin(element),\n      strategy, selector, maxSwipes\n    },\n  });\n};\n\nexport default commands;\n"],"file":"lib/commands/gestures.js","sourceRoot":"../../.."}
220
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/commands/gestures.js"],"names":["commands","toOrigin","element","util","wrapElement","unwrapElement","undefined","toPoint","x","y","_","isFinite","toRect","left","top","width","height","some","v","mobileLongClickGesture","opts","elementId","duration","uiautomator2","jwproxy","command","origin","offset","mobileDoubleClickGesture","mobileDragGesture","startX","startY","endX","endY","speed","start","end","mobileFlingGesture","direction","area","mobilePinchCloseGesture","percent","mobilePinchOpenGesture","mobileSwipeGesture","mobileScrollGesture","mobileScrollBackTo","elementToId","errors","InvalidArgumentError","mobileScroll","strategy","selector","maxSwipes","params"],"mappings":"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","sourcesContent":["import { util } from 'appium-support';\nimport _ from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst commands = {};\n\n\nfunction toOrigin (element) {\n  return element ? util.wrapElement(util.unwrapElement(element)) : undefined;\n}\n\nfunction toPoint (x, y) {\n  return _.isFinite(x) && _.isFinite(y) ? {x, y} : undefined;\n}\n\nfunction toRect (left, top, width, height) {\n  if ([left, top, width, height].some((v) => !_.isFinite(v))) {\n    return undefined;\n  }\n  return {left, top, width, height};\n}\n\n\n/**\n * @typedef {Object} LongClickOptions\n * @property {?string} elementId - The id of the element to be clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to click on\n * @property {?number} y - The y coordinate to click on\n * @property {?number} duration [500] - Click duration in milliseconds.\n * The value must not be negative\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?LongClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileLongClickGesture = async function mobileLongClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n    duration,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/long_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n    duration,\n  });\n};\n\n/**\n * @typedef {Object} DoubleClickOptions\n * @property {?string} elementId - The id of the element to be double clicked.\n * If the element is missing then both click offset coordinates must be provided.\n * If both the element id and offset are provided then the coordinates\n * are parsed as relative offsets from the top left corner of the element.\n * @property {?number} x - The x coordinate to double click on\n * @property {?number} y - The y coordinate to double click on\n */\n\n/**\n * Performs a click that lasts for the given duration\n *\n * @param {?DoubleClickOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDoubleClickGesture = async function mobileDoubleClickGesture (opts = {}) {\n  const {\n    elementId,\n    x, y,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/double_click', 'POST', {\n    origin: toOrigin(elementId),\n    offset: toPoint(x, y),\n  });\n};\n\n/**\n * @typedef {Object} DragOptions\n * @property {?string} elementId - The id of the element to be dragged.\n * If the element id is missing then the start coordinates must be provided.\n * If both the element id and the start coordinates are provided then these\n * coordinates are considered as offsets from the top left element corner.\n * @property {?number} startX - The x coordinate where the dragging starts\n * @property {?number} startY - The y coordinate where the dragging starts\n * @property {!number} endX - The x coordinate where the dragging ends\n * @property {!number} endY - The y coordinate where the dragging ends\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Drags this object to the specified location.\n *\n * @param {?DragOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileDragGesture = async function mobileDragGesture (opts = {}) {\n  const {\n    elementId,\n    startX, startY,\n    endX, endY,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/drag', 'POST', {\n    origin: toOrigin(elementId),\n    start: toPoint(startX, startY),\n    end: toPoint(endX, endY),\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} FlingOptions\n * @property {?string} elementId - The id of the element to be flinged.\n * If the element id is missing then fling bounding area must be provided.\n * If both the element id and the fling bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the fling bounding area\n * @property {?number} top - The top coordinate of the fling bounding area\n * @property {?number} width - The width of the fling bounding area\n * @property {?number} height - The height of the fling bounding area\n * @property {!string} direction - Direction of the fling.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {?number} speed [7500 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must be greater than the minimum fling\n * velocity for the given view (50 by default)\n */\n\n/**\n * Drags to the specified location.\n *\n * @param {?FlingOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileFlingGesture = async function mobileFlingGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/fling', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} PinchOptions\n * @property {?string} elementId - The id of the element to be pinched.\n * If the element id is missing then pinch bounding area must be provided.\n * If both the element id and the pinch bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the pinch bounding area\n * @property {?number} top - The top coordinate of the pinch bounding area\n * @property {?number} width - The width of the pinch bounding area\n * @property {?number} height - The height of the pinch bounding area\n * @property {!number} percent - The size of the pinch as a percentage of the pinch area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [2500 * displayDensity] - The speed at which to perform\n * this gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a pinch close gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchCloseGesture = async function mobilePinchCloseGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_close', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * Performs a pinch open gesture.\n *\n * @param {?PinchOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobilePinchOpenGesture = async function mobilePinchOpenGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/pinch_open', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} SwipeOptions\n * @property {?string} elementId - The id of the element to be swiped.\n * If the element id is missing then swipe bounding area must be provided.\n * If both the element id and the swipe bounding area are provided then the\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the swipe bounding area\n * @property {?number} top - The top coordinate of the swipe bounding area\n * @property {?number} width - The width of the swipe bounding area\n * @property {?number} height - The height of the swipe bounding area\n * @property {!string} direction - Direction of the swipe.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the swipe as a percentage of the swipe area size.\n * Valid values must be float numbers in range 0..1, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this\n * gesture in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a swipe gesture.\n *\n * @param {?SwipeOptions} opts\n * @throws {Error} if provided options are not valid\n */\ncommands.mobileSwipeGesture = async function mobileSwipeGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/swipe', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollOptions\n * @property {?string} elementId - The id of the element to be scrolled.\n * If the element id is missing then scroll bounding area must be provided.\n * If both the element id and the scroll bounding area are provided then this\n * area is effectively ignored.\n * @property {?number} left - The left coordinate of the scroll bounding area\n * @property {?number} top - The top coordinate of the scroll bounding area\n * @property {?number} width - The width of the scroll bounding area\n * @property {?number} height - The height of the scroll bounding area\n * @property {!string} direction - Direction of the scroll.\n * Acceptable values are: `up`, `down`, `left` and `right` (case insensitive)\n * @property {!number} percent - The size of the scroll as a percentage of the scrolling area size.\n * Valid values must be float numbers greater than zero, where 1.0 is 100%\n * @property {?number} speed [5000 * displayDensity] - The speed at which to perform this gesture\n * in pixels per second. The value must not be negative\n */\n\n/**\n * Performs a scroll gesture.\n *\n * @param {?ScrollOptions} opts\n * @throws {Error} if provided options are not valid\n * @returns {boolean} True if the object can still scroll in the given direction.\n */\ncommands.mobileScrollGesture = async function mobileScrollGesture (opts = {}) {\n  const {\n    elementId,\n    left, top, width, height,\n    direction,\n    percent,\n    speed,\n  } = opts;\n  return await this.uiautomator2.jwproxy.command('/appium/gestures/scroll', 'POST', {\n    origin: toOrigin(elementId),\n    area: toRect(left, top, width, height),\n    direction,\n    percent,\n    speed,\n  });\n};\n\n/**\n * @typedef {Object} ScrollElementToElementOpts\n * @property {string} elementId The identifier of the scrollable element,\n * which is going to be scrolled. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy.\n * @property {string} elementToId The identifier of the item, which belongs\n * to the scrollable element above, and which should become visible after\n * the scrolling operation is finished. It is required this element\n * was located by `-android uiautomator` strategy.\n */\n\n/**\n * Scrolls the given scrollable element `elementId` until `elementToId`\n * becomes visible. This function returns immediately if the `elementToId`\n * is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollElementToElementOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScrollBackTo = async function (opts = {}) {\n  const {elementId, elementToId} = opts;\n  if (!elementId || !elementToId) {\n    throw new errors.InvalidArgumentError(`Both elementId and elementToId arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command(\n    `/appium/element/${util.unwrapElement(elementId)}/scroll_to/${util.unwrapElement(elementToId)}`, 'POST', {});\n};\n\n/**\n * @typedef {Object} ScrollOpts\n * @property {?string} element The identifier of an element. It is required this element\n * is a valid scrollable container and it was located by `-android uiautomator`\n * strategy. If this property is not provided then the first currently available scrollable view\n * is selected for the interaction.\n * @property {!string} strategy The following strategies are supported:\n * - `accessibility id` (UiSelector().description)\n * - `class name` (UiSelector().className)\n * - `-android uiautomator` (UiSelector)\n * @property {!string} selector The corresponding lookup value for the given\n * strategy.\n * @property {?number} maxSwipes The maximum number of swipes to perform\n * on the target scrollable view in order to reach the destination element.\n * In case this value is unset then it would be retrieved from the scrollable\n * element itself (vua `getMaxSearchSwipes()` property).\n */\n\n/**\n * Scrolls the given scrollable element until the element identified\n * by `strategy` and `selector` becomes visible. This function returns immediately if the\n * destination element is already visible in the view port. Otherwise it would scroll\n * to the very beginning of the scrollable control and tries to reach the destination element\n * by scrolling its parent to the end step by step. The scroll direction (vertical or horizontal)\n * is detected automatically.\n *\n * @param {ScrollOpts} opts\n * @throws {Error} if the scrolling operation cannot be performed\n */\ncommands.mobileScroll = async function (opts = {}) {\n  const {element, strategy, selector, maxSwipes} = opts;\n  if (!strategy || !selector) {\n    throw new errors.InvalidArgumentError(`Both strategy and selector arguments must be provided`);\n  }\n  return await this.uiautomator2.jwproxy.command('/touch/scroll', 'POST', {\n    params: {\n      origin: toOrigin(element),\n      strategy, selector, maxSwipes\n    },\n  });\n};\n\nexport default commands;\n"],"file":"lib/commands/gestures.js","sourceRoot":"../../.."}
@@ -20,6 +20,7 @@ parser.registerNestingOperators('>', '+', '~');
20
20
  parser.registerAttrEqualityMods('^', '$', '*', '~');
21
21
  parser.enableSubstitutes();
22
22
  const RESOURCE_ID = 'resource-id';
23
+ const ID_LOCATOR_PATTERN = /^[a-zA-Z_][a-zA-Z0-9._]*:id\/[\S]+$/;
23
24
  const BOOLEAN_ATTRS = ['checkable', 'checked', 'clickable', 'enabled', 'focusable', 'focused', 'long-clickable', 'scrollable', 'selected'];
24
25
  const NUMERIC_ATTRS = ['index', 'instance'];
25
26
  const STR_ATTRS = ['description', RESOURCE_ID, 'text', 'class-name', 'package-name'];
@@ -68,8 +69,8 @@ function getWordMatcherRegex(word) {
68
69
  return `\\b(\\w*${(0, _lodash.escapeRegExp)(word)}\\w*)\\b`;
69
70
  }
70
71
 
71
- function prependAndroidId(str) {
72
- return str.startsWith('android:id/') ? str : `android:id/${str}`;
72
+ function formatIdLocator(locator) {
73
+ return ID_LOCATOR_PATTERN.test(locator) ? locator : `android:id/${locator}`;
73
74
  }
74
75
 
75
76
  function parseAttr(cssAttr) {
@@ -91,7 +92,7 @@ function parseAttr(cssAttr) {
91
92
  let value = cssAttr.value || '';
92
93
 
93
94
  if (attrName === RESOURCE_ID) {
94
- value = prependAndroidId(value);
95
+ value = formatIdLocator(value);
95
96
  }
96
97
 
97
98
  if (value === '') {
@@ -171,7 +172,7 @@ function parseCssRule(cssRule) {
171
172
  }
172
173
 
173
174
  if (cssRule.id) {
174
- uiAutomatorSelector += `.resourceId("${prependAndroidId(cssRule.id)}")`;
175
+ uiAutomatorSelector += `.resourceId("${formatIdLocator(cssRule.id)}")`;
175
176
  }
176
177
 
177
178
  if (cssRule.attrs) {
@@ -229,4 +230,4 @@ var _default = CssConverter;
229
230
  exports.default = _default;require('source-map-support').install();
230
231
 
231
232
 
232
- //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/css-converter.js"],"names":["CssConverter","parser","CssSelectorParser","registerSelectorPseudos","registerNestingOperators","registerAttrEqualityMods","enableSubstitutes","RESOURCE_ID","BOOLEAN_ATTRS","NUMERIC_ATTRS","STR_ATTRS","ALL_ATTRS","ATTRIBUTE_ALIASES","toSnakeCase","str","tokens","split","map","charAt","toUpperCase","slice","toLowerCase","out","join","assertGetBool","css","val","value","includes","Error","name","assertGetAttrName","attrName","officialAttr","aliasAttrs","getWordMatcherRegex","word","prependAndroidId","startsWith","parseAttr","cssAttr","valueType","methodName","operator","parsePseudo","cssPseudo","pseudoName","parseCssRule","cssRule","nestingOperator","uiAutomatorSelector","tagName","androidClass","classNames","cssClassNames","push","id","attrs","attr","pseudos","pseudo","rule","parseCssObject","type","selectors","selector","toUiAutomatorSelector","cssSelector","cssObj","parse","e","errors","InvalidSelectorError"],"mappings":"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","sourcesContent":["import { CssSelectorParser } from 'css-selector-parser';\nimport { escapeRegExp } from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst CssConverter = {};\n\nconst parser = new CssSelectorParser();\nparser.registerSelectorPseudos('has');\nparser.registerNestingOperators('>', '+', '~');\nparser.registerAttrEqualityMods('^', '$', '*', '~');\nparser.enableSubstitutes();\n\nconst RESOURCE_ID = 'resource-id';\n\nconst BOOLEAN_ATTRS = [\n  'checkable', 'checked', 'clickable', 'enabled', 'focusable',\n  'focused', 'long-clickable', 'scrollable', 'selected',\n];\n\nconst NUMERIC_ATTRS = [\n  'index', 'instance',\n];\n\nconst STR_ATTRS = [\n  'description', RESOURCE_ID, 'text', 'class-name', 'package-name'\n];\n\nconst ALL_ATTRS = [\n  ...BOOLEAN_ATTRS,\n  ...NUMERIC_ATTRS,\n  ...STR_ATTRS,\n];\n\nconst ATTRIBUTE_ALIASES = [\n  [RESOURCE_ID, ['id']],\n  ['description', [\n    'content-description', 'content-desc',\n    'desc', 'accessibility-id',\n  ]],\n  ['index', ['nth-child']],\n];\n\n/**\n * Convert hyphen separated word to snake case\n *\n * @param {string} str\n * @returns {string} The hyphen separated word translated to snake case\n */\nfunction toSnakeCase (str) {\n  if (!str) {\n    return '';\n  }\n  const tokens = str.split('-').map((str) => str.charAt(0).toUpperCase() + str.slice(1).toLowerCase());\n  const out = tokens.join('');\n  return out.charAt(0).toLowerCase() + out.slice(1);\n}\n\n/**\n * @typedef {Object} CssNameValueObject\n * @property {?name} name The name of the CSS object\n * @property {?string} value The value of the CSS object\n */\n\n/**\n * Get the boolean from a CSS object. If empty, return true. If not true/false/empty, throw exception\n *\n * @param {CssNameValueObject} css A CSS object that has 'name' and 'value'\n * @returns {string} Either 'true' or 'false'. If value is empty, return 'true'\n */\nfunction assertGetBool (css) {\n  const val = css.value?.toLowerCase() || 'true'; // an omitted boolean attribute means 'true' (e.g.: input[checked] means checked is true)\n  if (['true', 'false'].includes(val)) {\n    return val;\n  }\n  throw new Error(`'${css.name}' must be true, false or empty. Found '${css.value}'`);\n}\n\n/**\n * Get the canonical form of a CSS attribute name\n *\n * Converts to lowercase and if an attribute name is an alias for something else, return\n * what it is an alias for\n *\n * @param {Object} css CSS object\n * @returns {string} The canonical attribute name\n */\nfunction assertGetAttrName (css) {\n  const attrName = css.name.toLowerCase();\n\n  // Check if it's supported and if it is, return it\n  if (ALL_ATTRS.includes(attrName)) {\n    return attrName.toLowerCase();\n  }\n\n  // If attrName is an alias for something else, return that\n  for (const [officialAttr, aliasAttrs] of ATTRIBUTE_ALIASES) {\n    if (aliasAttrs.includes(attrName)) {\n      return officialAttr;\n    }\n  }\n  throw new Error(`'${attrName}' is not a valid attribute. ` +\n    `Supported attributes are '${ALL_ATTRS.join(', ')}'`);\n}\n\n/**\n * Get a regex that matches a whole word. For the ~= CSS attribute selector.\n *\n * @param {string} word\n * @returns {string} A regex \"word\" matcher\n */\nfunction getWordMatcherRegex (word) {\n  return `\\\\b(\\\\w*${escapeRegExp(word)}\\\\w*)\\\\b`;\n}\n\n/**\n * Add android:id/ to beginning of string if it's not there already\n *\n * @param {string} str\n * @returns {string} String with `android:id/` prepended (if it wasn't already)\n */\nfunction prependAndroidId (str) {\n  return str.startsWith('android:id/') ? str : `android:id/${str}`;\n}\n\n/**\n * @typedef {Object} CssAttr\n * @property {?string} valueType Type of attribute (must be string or empty)\n * @property {?string} value Value of the attribute\n * @property {?string} operator The operator between value and value type (=, *=, , ^=, $=)\n */\n\n/**\n * Convert a CSS attribute into a UiSelector method call\n *\n * @param {CssAttr} cssAttr CSS attribute object\n * @returns {string} CSS attribute parsed as UiSelector\n */\nfunction parseAttr (cssAttr) {\n  if (cssAttr.valueType && cssAttr.valueType !== 'string') {\n    throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` +\n      `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);\n  }\n  const attrName = assertGetAttrName(cssAttr);\n  const methodName = toSnakeCase(attrName);\n\n  // Validate that it's a supported attribute\n  if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {\n    throw new Error(`'${attrName}' is not supported. Supported attributes are ` +\n      `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);\n  }\n\n  // Parse boolean, if it's a boolean attribute\n  if (BOOLEAN_ATTRS.includes(attrName)) {\n    return `.${methodName}(${assertGetBool(cssAttr)})`;\n  }\n\n  // Otherwise parse as string\n  let value = cssAttr.value || '';\n  if (attrName === RESOURCE_ID) {\n    value = prependAndroidId(value);\n  }\n  if (value === '') {\n    return `.${methodName}Matches(\"\")`;\n  }\n\n  switch (cssAttr.operator) {\n    case '=':\n      return `.${methodName}(\"${value}\")`;\n    case '*=':\n      if (['description', 'text'].includes(attrName)) {\n        return `.${methodName}Contains(\"${value}\")`;\n      }\n      return `.${methodName}Matches(\"${escapeRegExp(value)}\")`;\n    case '^=':\n      if (['description', 'text'].includes(attrName)) {\n        return `.${methodName}StartsWith(\"${value}\")`;\n      }\n      return `.${methodName}Matches(\"^${escapeRegExp(value)}\")`;\n    case '$=':\n      return `.${methodName}Matches(\"${escapeRegExp(value)}$\")`;\n    case '~=':\n      return `.${methodName}Matches(\"${getWordMatcherRegex(value)}\")`;\n    default:\n      // Unreachable, but adding error in case a new CSS attribute is added.\n      throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` +\n        ` '=', '*=', '^=', '$=' and '~=' are supported.`);\n  }\n}\n\n/**\n * @typedef {Object} CssPseudo\n * @property {?string} valueType The type of CSS pseudo selector (https://www.npmjs.com/package/css-selector-parser for reference)\n * @property {?string} name The name of the pseudo selector\n * @property {?string} value The value of the pseudo selector\n */\n\n/**\n * Convert a CSS pseudo class to a UiSelector\n *\n * @param {CssPseudo} cssPseudo CSS Pseudo class\n * @returns {string} Pseudo selector parsed as UiSelector\n */\nfunction parsePseudo (cssPseudo) {\n  if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {\n    throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` +\n      `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);\n  }\n\n  const pseudoName = assertGetAttrName(cssPseudo);\n\n  if (BOOLEAN_ATTRS.includes(pseudoName)) {\n    return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;\n  }\n\n  if (NUMERIC_ATTRS.includes(pseudoName)) {\n    return `.${pseudoName}(${cssPseudo.value})`;\n  }\n}\n\n/**\n * @typedef {Object} CssRule\n * @property {?string} nestingOperator The nesting operator (aka: combinator https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors)\n * @property {?string} tagName The tag name (aka: type selector https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors)\n * @property {?string[]} classNames An array of CSS class names\n * @property {?CssAttr[]} attrs An array of CSS attributes\n * @property {?CssPseudo[]} attrs An array of CSS pseudos\n * @property {?string} id CSS identifier\n * @property {?CssRule} rule A descendant of this CSS rule\n */\n\n/**\n * Convert a CSS rule to a UiSelector\n * @param {CssRule} cssRule CSS rule definition\n */\nfunction parseCssRule (cssRule) {\n  const { nestingOperator } = cssRule;\n  if (nestingOperator && nestingOperator !== ' ') {\n    throw new Error(`'${nestingOperator}' is not a supported combinator. ` +\n      `Only child combinator (>) and descendant combinator are supported.`);\n  }\n\n  let uiAutomatorSelector = 'new UiSelector()';\n  if (cssRule.tagName && cssRule.tagName !== '*') {\n    let androidClass = [cssRule.tagName];\n    if (cssRule.classNames) {\n      for (const cssClassNames of cssRule.classNames) {\n        androidClass.push(cssClassNames);\n      }\n      uiAutomatorSelector += `.className(\"${androidClass.join('.')}\")`;\n    } else {\n      uiAutomatorSelector += `.classNameMatches(\"${cssRule.tagName}\")`;\n    }\n  } else if (cssRule.classNames) {\n    uiAutomatorSelector += `.classNameMatches(\"${cssRule.classNames.join('\\\\.')}\")`;\n  }\n  if (cssRule.id) {\n    uiAutomatorSelector += `.resourceId(\"${prependAndroidId(cssRule.id)}\")`;\n  }\n  if (cssRule.attrs) {\n    for (const attr of cssRule.attrs) {\n      uiAutomatorSelector += parseAttr(attr);\n    }\n  }\n  if (cssRule.pseudos) {\n    for (const pseudo of cssRule.pseudos) {\n      uiAutomatorSelector += parsePseudo(pseudo);\n    }\n  }\n  if (cssRule.rule) {\n    uiAutomatorSelector += `.childSelector(${parseCssRule(cssRule.rule)})`;\n  }\n  return uiAutomatorSelector;\n}\n\n/**\n * @typedef {Object} CssObject\n * @property {?string} type Type of CSS object. 'rule', 'ruleset' or 'selectors'\n */\n\n/**\n * Convert CSS object to UiAutomator2 selector\n * @param {CssObject} css CSS object\n * @returns {string} The CSS object parsed as a UiSelector\n */\nfunction parseCssObject (css) {\n  switch (css.type) {\n    case 'rule':\n      return parseCssRule(css);\n    case 'ruleSet':\n      return parseCssObject(css.rule);\n    case 'selectors':\n      return css.selectors.map((selector) => parseCssObject(selector)).join('; ');\n\n    default:\n      // This is never reachable, but if it ever is do this.\n      throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);\n  }\n}\n\n/**\n * Convert a CSS selector to a UiAutomator2 selector\n * @param {string} cssSelector CSS Selector\n * @returns {string} The CSS selector converted to a UiSelector\n */\nCssConverter.toUiAutomatorSelector = function toUiAutomatorSelector (cssSelector) {\n  let cssObj;\n  try {\n    cssObj = parser.parse(cssSelector);\n  } catch (e) {\n    throw new errors.InvalidSelectorError(`Invalid CSS selector '${cssSelector}'. Reason: '${e}'`);\n  }\n  try {\n    return parseCssObject(cssObj);\n  } catch (e) {\n    throw new errors.InvalidSelectorError(`Unsupported CSS selector '${cssSelector}'. Reason: '${e}'`);\n  }\n};\n\nexport default CssConverter;"],"file":"lib/css-converter.js","sourceRoot":"../.."}
233
+ //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["lib/css-converter.js"],"names":["CssConverter","parser","CssSelectorParser","registerSelectorPseudos","registerNestingOperators","registerAttrEqualityMods","enableSubstitutes","RESOURCE_ID","ID_LOCATOR_PATTERN","BOOLEAN_ATTRS","NUMERIC_ATTRS","STR_ATTRS","ALL_ATTRS","ATTRIBUTE_ALIASES","toSnakeCase","str","tokens","split","map","charAt","toUpperCase","slice","toLowerCase","out","join","assertGetBool","css","val","value","includes","Error","name","assertGetAttrName","attrName","officialAttr","aliasAttrs","getWordMatcherRegex","word","formatIdLocator","locator","test","parseAttr","cssAttr","valueType","methodName","operator","parsePseudo","cssPseudo","pseudoName","parseCssRule","cssRule","nestingOperator","uiAutomatorSelector","tagName","androidClass","classNames","cssClassNames","push","id","attrs","attr","pseudos","pseudo","rule","parseCssObject","type","selectors","selector","toUiAutomatorSelector","cssSelector","cssObj","parse","e","errors","InvalidSelectorError"],"mappings":"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","sourcesContent":["import { CssSelectorParser } from 'css-selector-parser';\nimport { escapeRegExp } from 'lodash';\nimport { errors } from 'appium-base-driver';\n\nconst CssConverter = {};\n\nconst parser = new CssSelectorParser();\nparser.registerSelectorPseudos('has');\nparser.registerNestingOperators('>', '+', '~');\nparser.registerAttrEqualityMods('^', '$', '*', '~');\nparser.enableSubstitutes();\n\nconst RESOURCE_ID = 'resource-id';\nconst ID_LOCATOR_PATTERN = /^[a-zA-Z_][a-zA-Z0-9._]*:id\\/[\\S]+$/;\n\nconst BOOLEAN_ATTRS = [\n  'checkable', 'checked', 'clickable', 'enabled', 'focusable',\n  'focused', 'long-clickable', 'scrollable', 'selected',\n];\n\nconst NUMERIC_ATTRS = [\n  'index', 'instance',\n];\n\nconst STR_ATTRS = [\n  'description', RESOURCE_ID, 'text', 'class-name', 'package-name'\n];\n\nconst ALL_ATTRS = [\n  ...BOOLEAN_ATTRS,\n  ...NUMERIC_ATTRS,\n  ...STR_ATTRS,\n];\n\nconst ATTRIBUTE_ALIASES = [\n  [RESOURCE_ID, ['id']],\n  ['description', [\n    'content-description', 'content-desc',\n    'desc', 'accessibility-id',\n  ]],\n  ['index', ['nth-child']],\n];\n\n/**\n * Convert hyphen separated word to snake case\n *\n * @param {string} str\n * @returns {string} The hyphen separated word translated to snake case\n */\nfunction toSnakeCase (str) {\n  if (!str) {\n    return '';\n  }\n  const tokens = str.split('-').map((str) => str.charAt(0).toUpperCase() + str.slice(1).toLowerCase());\n  const out = tokens.join('');\n  return out.charAt(0).toLowerCase() + out.slice(1);\n}\n\n/**\n * @typedef {Object} CssNameValueObject\n * @property {?name} name The name of the CSS object\n * @property {?string} value The value of the CSS object\n */\n\n/**\n * Get the boolean from a CSS object. If empty, return true. If not true/false/empty, throw exception\n *\n * @param {CssNameValueObject} css A CSS object that has 'name' and 'value'\n * @returns {string} Either 'true' or 'false'. If value is empty, return 'true'\n */\nfunction assertGetBool (css) {\n  const val = css.value?.toLowerCase() || 'true'; // an omitted boolean attribute means 'true' (e.g.: input[checked] means checked is true)\n  if (['true', 'false'].includes(val)) {\n    return val;\n  }\n  throw new Error(`'${css.name}' must be true, false or empty. Found '${css.value}'`);\n}\n\n/**\n * Get the canonical form of a CSS attribute name\n *\n * Converts to lowercase and if an attribute name is an alias for something else, return\n * what it is an alias for\n *\n * @param {Object} css CSS object\n * @returns {string} The canonical attribute name\n */\nfunction assertGetAttrName (css) {\n  const attrName = css.name.toLowerCase();\n\n  // Check if it's supported and if it is, return it\n  if (ALL_ATTRS.includes(attrName)) {\n    return attrName.toLowerCase();\n  }\n\n  // If attrName is an alias for something else, return that\n  for (const [officialAttr, aliasAttrs] of ATTRIBUTE_ALIASES) {\n    if (aliasAttrs.includes(attrName)) {\n      return officialAttr;\n    }\n  }\n  throw new Error(`'${attrName}' is not a valid attribute. ` +\n    `Supported attributes are '${ALL_ATTRS.join(', ')}'`);\n}\n\n/**\n * Get a regex that matches a whole word. For the ~= CSS attribute selector.\n *\n * @param {string} word\n * @returns {string} A regex \"word\" matcher\n */\nfunction getWordMatcherRegex (word) {\n  return `\\\\b(\\\\w*${escapeRegExp(word)}\\\\w*)\\\\b`;\n}\n\n/**\n * Add android:id/ to beginning of string if it's not there already\n *\n * @param {string} locator The initial locator\n * @returns {string} String with `android:id/` prepended (if it wasn't already)\n */\nfunction formatIdLocator (locator) {\n  return ID_LOCATOR_PATTERN.test(locator) ? locator : `android:id/${locator}`;\n}\n\n/**\n * @typedef {Object} CssAttr\n * @property {?string} valueType Type of attribute (must be string or empty)\n * @property {?string} value Value of the attribute\n * @property {?string} operator The operator between value and value type (=, *=, , ^=, $=)\n */\n\n/**\n * Convert a CSS attribute into a UiSelector method call\n *\n * @param {CssAttr} cssAttr CSS attribute object\n * @returns {string} CSS attribute parsed as UiSelector\n */\nfunction parseAttr (cssAttr) {\n  if (cssAttr.valueType && cssAttr.valueType !== 'string') {\n    throw new Error(`'${cssAttr.name}=${cssAttr.value}' is an invalid attribute. ` +\n      `Only 'string' and empty attribute types are supported. Found '${cssAttr.valueType}'`);\n  }\n  const attrName = assertGetAttrName(cssAttr);\n  const methodName = toSnakeCase(attrName);\n\n  // Validate that it's a supported attribute\n  if (!STR_ATTRS.includes(attrName) && !BOOLEAN_ATTRS.includes(attrName)) {\n    throw new Error(`'${attrName}' is not supported. Supported attributes are ` +\n      `'${[...STR_ATTRS, ...BOOLEAN_ATTRS].join(', ')}'`);\n  }\n\n  // Parse boolean, if it's a boolean attribute\n  if (BOOLEAN_ATTRS.includes(attrName)) {\n    return `.${methodName}(${assertGetBool(cssAttr)})`;\n  }\n\n  // Otherwise parse as string\n  let value = cssAttr.value || '';\n  if (attrName === RESOURCE_ID) {\n    value = formatIdLocator(value);\n  }\n  if (value === '') {\n    return `.${methodName}Matches(\"\")`;\n  }\n\n  switch (cssAttr.operator) {\n    case '=':\n      return `.${methodName}(\"${value}\")`;\n    case '*=':\n      if (['description', 'text'].includes(attrName)) {\n        return `.${methodName}Contains(\"${value}\")`;\n      }\n      return `.${methodName}Matches(\"${escapeRegExp(value)}\")`;\n    case '^=':\n      if (['description', 'text'].includes(attrName)) {\n        return `.${methodName}StartsWith(\"${value}\")`;\n      }\n      return `.${methodName}Matches(\"^${escapeRegExp(value)}\")`;\n    case '$=':\n      return `.${methodName}Matches(\"${escapeRegExp(value)}$\")`;\n    case '~=':\n      return `.${methodName}Matches(\"${getWordMatcherRegex(value)}\")`;\n    default:\n      // Unreachable, but adding error in case a new CSS attribute is added.\n      throw new Error(`Unsupported CSS attribute operator '${cssAttr.operator}'. ` +\n        ` '=', '*=', '^=', '$=' and '~=' are supported.`);\n  }\n}\n\n/**\n * @typedef {Object} CssPseudo\n * @property {?string} valueType The type of CSS pseudo selector (https://www.npmjs.com/package/css-selector-parser for reference)\n * @property {?string} name The name of the pseudo selector\n * @property {?string} value The value of the pseudo selector\n */\n\n/**\n * Convert a CSS pseudo class to a UiSelector\n *\n * @param {CssPseudo} cssPseudo CSS Pseudo class\n * @returns {string} Pseudo selector parsed as UiSelector\n */\nfunction parsePseudo (cssPseudo) {\n  if (cssPseudo.valueType && cssPseudo.valueType !== 'string') {\n    throw new Error(`'${cssPseudo.name}=${cssPseudo.value}'. ` +\n      `Unsupported css pseudo class value type: '${cssPseudo.valueType}'. Only 'string' type or empty is supported.`);\n  }\n\n  const pseudoName = assertGetAttrName(cssPseudo);\n\n  if (BOOLEAN_ATTRS.includes(pseudoName)) {\n    return `.${toSnakeCase(pseudoName)}(${assertGetBool(cssPseudo)})`;\n  }\n\n  if (NUMERIC_ATTRS.includes(pseudoName)) {\n    return `.${pseudoName}(${cssPseudo.value})`;\n  }\n}\n\n/**\n * @typedef {Object} CssRule\n * @property {?string} nestingOperator The nesting operator (aka: combinator https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Selectors)\n * @property {?string} tagName The tag name (aka: type selector https://developer.mozilla.org/en-US/docs/Web/CSS/Type_selectors)\n * @property {?string[]} classNames An array of CSS class names\n * @property {?CssAttr[]} attrs An array of CSS attributes\n * @property {?CssPseudo[]} attrs An array of CSS pseudos\n * @property {?string} id CSS identifier\n * @property {?CssRule} rule A descendant of this CSS rule\n */\n\n/**\n * Convert a CSS rule to a UiSelector\n * @param {CssRule} cssRule CSS rule definition\n */\nfunction parseCssRule (cssRule) {\n  const { nestingOperator } = cssRule;\n  if (nestingOperator && nestingOperator !== ' ') {\n    throw new Error(`'${nestingOperator}' is not a supported combinator. ` +\n      `Only child combinator (>) and descendant combinator are supported.`);\n  }\n\n  let uiAutomatorSelector = 'new UiSelector()';\n  if (cssRule.tagName && cssRule.tagName !== '*') {\n    let androidClass = [cssRule.tagName];\n    if (cssRule.classNames) {\n      for (const cssClassNames of cssRule.classNames) {\n        androidClass.push(cssClassNames);\n      }\n      uiAutomatorSelector += `.className(\"${androidClass.join('.')}\")`;\n    } else {\n      uiAutomatorSelector += `.classNameMatches(\"${cssRule.tagName}\")`;\n    }\n  } else if (cssRule.classNames) {\n    uiAutomatorSelector += `.classNameMatches(\"${cssRule.classNames.join('\\\\.')}\")`;\n  }\n  if (cssRule.id) {\n    uiAutomatorSelector += `.resourceId(\"${formatIdLocator(cssRule.id)}\")`;\n  }\n  if (cssRule.attrs) {\n    for (const attr of cssRule.attrs) {\n      uiAutomatorSelector += parseAttr(attr);\n    }\n  }\n  if (cssRule.pseudos) {\n    for (const pseudo of cssRule.pseudos) {\n      uiAutomatorSelector += parsePseudo(pseudo);\n    }\n  }\n  if (cssRule.rule) {\n    uiAutomatorSelector += `.childSelector(${parseCssRule(cssRule.rule)})`;\n  }\n  return uiAutomatorSelector;\n}\n\n/**\n * @typedef {Object} CssObject\n * @property {?string} type Type of CSS object. 'rule', 'ruleset' or 'selectors'\n */\n\n/**\n * Convert CSS object to UiAutomator2 selector\n * @param {CssObject} css CSS object\n * @returns {string} The CSS object parsed as a UiSelector\n */\nfunction parseCssObject (css) {\n  switch (css.type) {\n    case 'rule':\n      return parseCssRule(css);\n    case 'ruleSet':\n      return parseCssObject(css.rule);\n    case 'selectors':\n      return css.selectors.map((selector) => parseCssObject(selector)).join('; ');\n\n    default:\n      // This is never reachable, but if it ever is do this.\n      throw new Error(`UiAutomator does not support '${css.type}' css. Only supports 'rule', 'ruleSet', 'selectors' `);\n  }\n}\n\n/**\n * Convert a CSS selector to a UiAutomator2 selector\n * @param {string} cssSelector CSS Selector\n * @returns {string} The CSS selector converted to a UiSelector\n */\nCssConverter.toUiAutomatorSelector = function toUiAutomatorSelector (cssSelector) {\n  let cssObj;\n  try {\n    cssObj = parser.parse(cssSelector);\n  } catch (e) {\n    throw new errors.InvalidSelectorError(`Invalid CSS selector '${cssSelector}'. Reason: '${e}'`);\n  }\n  try {\n    return parseCssObject(cssObj);\n  } catch (e) {\n    throw new errors.InvalidSelectorError(`Unsupported CSS selector '${cssSelector}'. Reason: '${e}'`);\n  }\n};\n\nexport default CssConverter;"],"file":"lib/css-converter.js","sourceRoot":"../.."}