san650/ember-cli-page-object

View on GitHub
addon/src/properties/value.js

Summary

Maintainability
A
0 mins
Test Coverage
import { findOne } from '../-private/finders';
import { getter } from '../macros/index';

/**
 * @public
 *
 * Returns the value of a matched element, or an array of values of all
 * matched elements. If a matched element is contenteditable, this helper
 * will return the html content of the element.
 *
 * @example
 *
 * // <input value="Lorem ipsum">
 *
 * import { create, value } from 'ember-cli-page-object';
 *
 * const page = create({
 *   value: value('input')
 * });
 *
 * assert.equal(page.value, 'Lorem ipsum');
 *
 * @example
 *
 * // <div contenteditable="true"><b>Lorem ipsum</b></div>
 *
 * import { create, value } from 'ember-cli-page-object';
 *
 * const page = create({
 *   value: value('[contenteditable]')
 * });
 *
 * assert.equal(page.value, '<b>Lorem ipsum</b>');
 *
 * @example
 *
 * // <div><input value="lorem"></div>
 * // <div class="scope"><input value="ipsum"></div>
 *
 * import { create, value } from 'ember-cli-page-object';
 *
 * const page = create({
 *   value: value('input', { scope: '.scope' })
 * });
 *
 * assert.equal(page.value, 'ipsum');
 *
 * @example
 *
 * // <div><input value="lorem"></div>
 * // <div class="scope"><input value="ipsum"></div>
 *
 * import { create, value } from 'ember-cli-page-object';
 *
 * const page = create({
 *   scope: '.scope',
 *   value: value('input')
 * });
 *
 * assert.equal(page.value, 'ipsum');
 *
 * @public
 *
 * @param {string} selector - CSS selector of the element to check
 * @param {Object} options - Additional options
 * @param {string} options.scope - Nests provided scope within parent's scope
 * @param {boolean} options.resetScope - Override parent's scope
 * @param {number} options.at - Reduce the set of matched elements to the one at the specified index
 * @param {string} options.testContainer - Context where to search elements in the DOM
 * @return {Descriptor}
 *
 * @throws Will throw an error if no element matches selector
 * @throws Will throw an error if multiple elements are matched by selector
 */
export function value(selector, userOptions = {}) {
  return getter(function (key) {
    let options = {
      pageObjectKey: key,
      ...userOptions,
    };

    const element = findOne(this, selector, options);

    return element.hasAttribute('contenteditable')
      ? element.innerHTML
      : getValue(element);
  });
}

function getValue(element) {
  const { value } = element;
  if (value !== undefined && element.tagName.toLowerCase() === 'select') {
    return selectValue(element);
  }

  return element.value;
}

function selectValue(element) {
  const selectedOptions = Array.from(element.selectedOptions).filter(
    (option) =>
      !option.disabled &&
      (option.parentNode.tagName.toLowerCase() !== 'optgroup' ||
        !option.parentNode.disabled)
  );

  if (element.multiple) {
    return selectedOptions.map((option) => option.value);
  } else if (selectedOptions.length === 0) {
    return null;
  }

  return element.value;
}