123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require_relative 'test_flags'
- # Form helpers below are loaded for the new stack app only and provide functions for returning the form field elements.
- module FormHelpers
- @form_context = nil
- # Returns the outer container element of the form field via its label.
- # The returned object is always an instance of `Capybara::Node::Element``, with some added sugar on top.
- def find_outer(label, **find_options)
- ZammadFormFieldCapybaraElementDelegator.new(find('.formkit-outer') { |element| element.has_css?('label', text: label, **find_options) }, @form_context)
- end
- # Usage:
- #
- # find_input('Title')
- # find_select('Owner')
- # find_treeselect('Category')
- # find_autocomplete('Customer')
- # find_editor('Text')
- # find_datepicker('Pending till')
- # find_toggle('Remember me')
- #
- # # In case of ambiguous labels, make sure to pass `exact_text` option
- # find_datepicker(nil, exact_text: 'Date')
- #
- alias find_input find_outer
- alias find_select find_outer
- alias find_treeselect find_outer
- alias find_autocomplete find_outer
- alias find_editor find_outer
- alias find_datepicker find_outer
- alias find_toggle find_outer
- # Returns the outer container element of the form field radio via its ID.
- # The returned object is always an instance of `Capybara::Node::Element``, with some added sugar on top.
- def find_radio(name, **find_options)
- ZammadFormFieldCapybaraElementDelegator.new(first("[name^=\"#{name}\"]", **find_options).ancestor('.formkit-outer'), @form_context)
- end
- # Provides a form context for stabilizing multiple field interactions.
- # This is implemented by tracking of the expected form updater and other GraphQL responses.
- # To define custom starting form updater response number, use the `form_updater_gql_number` argument (default: nil).
- #
- # Usage:
- #
- # within_form(form_updater_gql_number: 2) do
- # find_autocomplete('CC').search_for_options([email_address_1, email_address_2])
- # find_autocomplete('Tags').search_for_options([tag_1, tag_2, tag_3]).select_options(%w[foo bar])
- # find_editor('Text').type(body)
- # end
- #
- def within_form(form_updater_gql_number: nil)
- setup_form_context(form_updater_gql_number)
- yield
- demolish_form_context
- end
- private
- def setup_form_context(form_updater_gql_number)
- @form_context = ZammadFormContext.new
- return if form_updater_gql_number.blank?
- @form_context.init_form_updater(form_updater_gql_number)
- end
- def demolish_form_context
- @form_context = nil
- end
- end
- # Extension below allows for execution of custom actions on the returned form field elements.
- # This class delegates any missing methods upstream to `Capybara::Node::Element` class.
- class ZammadFormFieldCapybaraElementDelegator < SimpleDelegator
- attr_reader :element, :form_context
- include Capybara::DSL
- include BrowserTestHelper
- include TestFlags
- def initialize(element, form_context)
- @element = element
- @form_context = form_context
- super(element)
- end
- # Returns identifier of the form field.
- def field_id
- return element.find('.formkit-input', visible: :all)['id'] if input? || type_date? || type_datetime?
- return element.find('textarea')['id'] if type_textarea?
- return element.find('.formkit-fieldset')['id'] if type_radio?
- return element.find('[role="textbox"]')['id'] if type_editor?
- return element.find('[role="switch"], input[type="checkbox"]', visible: :all)['id'] if type_toggle? || type_checkbox?
- element.find('output', visible: :all)['id']
- end
- # Returns (hidden) input element used by several form field implementations to track the current value.
- # NOTE: A returned element might not be a regular INPUT field due to custom implementation.
- def input_element
- element.find("##{field_id}", visible: :all)
- end
- # Searches treeselect and autocomplete fields for supplied option via its label and selects it.
- #
- # Usage:
- #
- # find_treeselect('Tree Select').search_for_option('Parent 1::Option A')
- # find_autocomplete('Tags').search_for_option(tag_1)
- #
- # # To wait for a custom GraphQL response, you can provide expected `gql_filename` and/or `gql_number`.
- # find_autocomplete('Custom').search_for_option('foo', gql_filename: 'shared/entities/user/graphql/queries/user.graphql', gql_number: 4)
- #
- # # To select an autocomplete option with a different text than the query, provide an optional `label` parameter.
- # find autocomplete('Customer').search_for_option(customer.email, label: customer.fullname)
- #
- def search_for_option(query, label: query, gql_filename: '', gql_number: 1, use_action: false, **find_options)
- return search_for_options(query, gql_filename: gql_filename, gql_number: gql_number, **find_options) if query.is_a?(Array)
- return search_for_tags_option(query, gql_filename: gql_filename, gql_number: gql_number) if type_tags?
- return search_for_autocomplete_option(query, label: label, gql_filename: gql_filename, gql_number: gql_number, use_action: use_action, **find_options) if autocomplete?
- raise 'Field does not support searching for options' if !type_treeselect?
- element.click
- wait_until_opened
- # calculate before closing, since we cannot access it, if dialog is closed
- is_multi_select = multi_select?
- browse_for_option(query, **find_options) do |option|
- # Filter input in desktop view is part of the element, not the dropdown/dialog.
- searchbox = if desktop_view?
- element.find('[role="searchbox"]')
- else
- find('[role="searchbox"]')
- end
- searchbox.fill_in with: option
- find('[role="option"]', text: option, **find_options).click
- maybe_wait_for_form_updater
- end
- send_keys(:escape) if is_multi_select
- wait_until_closed
- self # support chaining
- end
- # Searches treeselect and autocomplete fields for supplied options via their labels and selects them.
- # NOTE: The field must support multiple selection, otherwise an error will be raised.
- #
- # Usage:
- #
- # find_treeselect('Tree Select').search_for_options(['Parent 1::Option A', 'Parent 2::Option B', 'Option C'])
- # find_autocomplete('Tags').search_for_options([tag_1, tag_2, tag_3])
- #
- # # To wait for a custom GraphQL response, you can provide expected `gql_filename` and/or `gql_number`.
- # find_autocomplete('Tags').search_for_option('foo', gql_number: 3)
- #
- def search_for_options(queries, labels: queries, gql_filename: '', gql_number: 1, use_action: false, **find_options)
- return search_for_tags_options(queries, gql_filename: gql_filename, gql_number: gql_number) if type_tags?
- return search_for_autocomplete_options(queries, labels: labels, gql_filename: gql_filename, gql_number: gql_number, use_action: use_action, **find_options) if autocomplete?
- raise 'Field does not support searching for options' if !type_treeselect?
- element.click
- wait_for_test_flag("field-tree-select-#{field_id}.opened")
- raise 'Field does not support multiple selection' if !multi_select?
- queries.each do |query|
- browse_for_option(query, **find_options) do |option, rewind|
- # Filter input in desktop view is part of the element, not the dropdown/dialog.
- searchbox = if desktop_view?
- element.find('[role="searchbox"]')
- else
- find('[role="searchbox"]')
- end
- searchbox.fill_in with: option
- find('[role="option"]', text: option, **find_options).click
- maybe_wait_for_form_updater
- rewind.call
- end
- end
- send_keys(:escape)
- wait_for_test_flag("field-tree-select-#{field_id}.closed")
- self # support chaining
- end
- # Selects an option in select, treeselect and autocomplete fields via its label.
- # NOTE: The option must be part of initial options provided by the field, no autocomplete search will occur.
- #
- # Usage:
- #
- # find_select('Owner').select_option('Test Admin Agent')
- # find_treeselect('Tree Select').select_option('Parent 1::Option A')
- # find_autocomplete('Organization').select_option(secondary_organizations.last.name)
- #
- def select_option(label, **find_options)
- return select_options(label, **find_options) if label.is_a?(Array)
- return select_treeselect_option(label, **find_options) if type_treeselect?
- return select_tags_option(label, **find_options) if type_tags?
- return select_autocomplete_option(label, **find_options) if autocomplete?
- raise 'Element is not a field of type select' if !type_select?
- element.click
- wait_for_test_flag('common-select.opened')
- # calculate before closing, since we cannot access it, if dialog is closed
- is_multi_select = multi_select?
- select_option_by_label(label, **find_options)
- send_keys(:escape) if is_multi_select
- wait_for_test_flag('common-select.closed')
- self # support chaining
- end
- # Selects multiple options in select, treeselect and autocomplete fields via its label.
- # NOTE: The option must be part of initial options provided by the field, no autocomplete search will occur.
- # NOTE: The field must support multiple selection, otherwise an error will be raised.
- #
- # Usage:
- #
- # find_select('Multi Select').select_options(['Option 1', 'Option 2'])
- # find_treeselect('Multi Tree Select').select_options(['Parent 1::Option A', 'Parent 2::Option C'])
- # find_autocomplete('Tags').select_options(%w[foo bar])
- #
- def select_options(labels, **find_options)
- return select_treeselect_options(labels, **find_options) if type_treeselect?
- return select_tags_options(labels, **find_options) if type_tags?
- return select_autocomplete_options(labels, **find_options) if autocomplete?
- raise 'Element is not a field of type select' if !type_select?
- element.click
- wait_for_test_flag('common-select.opened')
- raise 'Field does not support multiple selection' if !multi_select?
- labels.each do |label|
- select_option_by_label(label, **find_options)
- end
- send_keys(:escape)
- wait_for_test_flag('common-select.closed')
- self # support chaining
- end
- # Clears selection in select, treeselect and autocomplete fields.
- # NOTE: The field must support selection clearing, otherwise an error will be raised.
- def clear_selection
- raise 'Field does not support clearing selection' if !type_select? && !type_treeselect? && !autocomplete?
- element.find('[role="button"][aria-label="Clear Selection"]').click
- maybe_wait_for_form_updater
- self # support chaining
- end
- # Types the provided text into an input or editor field.
- #
- # Usage:
- #
- # find_input('Title').type(body)
- # find_editor('Text').type(body)
- #
- def type(text, **type_options)
- return type_editor(text, **type_options) if type_editor?
- input_element.fill_in with: text
- maybe_wait_for_form_updater
- self # support chaining
- end
- def type_editor(text, click: true)
- raise 'Field does not support typing' if !type_editor?
- cursor_home_shortcut = mac_platform? ? %i[command up] : %i[control home]
- input_element.click.send_keys(cursor_home_shortcut) if click
- input_element.send_keys(text)
- maybe_wait_for_form_updater
- self # support chaining
- end
- # Clears the input of text, editor, date and datetime fields.
- def clear
- return clear_date if type_date? || type_datetime?
- raise 'Field does not support clearing' if !input? && !type_editor?
- input_element.click.send_keys([magic_key, 'a'], :backspace)
- maybe_wait_for_form_updater
- self # support chaining
- end
- # Selects a date in a date picker field.
- #
- # Usage:
- # find_datepicker('Date Picker').select_date(Date.today)
- # find_datepicker('Date Picker').select_date('2023-01-01')
- #
- def select_date(date)
- raise 'Field does not support selecting dates' if !type_date? && !type_datetime?
- element.click
- wait_until_opened
- date = Date.parse(date) if !date.is_a?(Date) && !date.is_a?(DateTime) && !date.is_a?(Time)
- element.find('[aria-label*="Open the years overlay"]').click
- element.find('.dp__overlay_col', text: date.year).click
- element.find('[aria-label*="Open the months overlay"]').click
- element.find('.dp__overlay_col', text: date.strftime('%b')).click
- id = date.strftime('%Y-%m-%d')
- element.find_by_id(id).click # rubocop:disable Rails/DynamicFindBy
- if desktop_view?
- element.click
- wait_until_opened
- end
- yield if block_given?
- # close_date_picker(element)
- # wait_until_closed
- maybe_wait_for_form_updater
- self # support chaining
- end
- # Selects a date and enters time in a datetime picker field.
- #
- # Usage:
- # find_datepicker('Date Time').select_datetime(DateTime.now)
- # find_datepicker('Date Time').select_datetime('2023-01-01T09:00:00.000Z')
- #
- def select_datetime(datetime)
- raise 'Field does not support selecting datetimes' if !type_datetime?
- datetime = DateTime.parse(datetime) if !datetime.is_a?(DateTime) && !datetime.is_a?(Time)
- select_date(datetime) do
- element.find('[aria-label="Open the time picker"]').click
- element.find('[aria-label*="Open the hours overlay"]').click
- element.find('.dp__overlay_col', text: format('%02d', datetime.hour)).click
- element.find('[aria-label*="Open the minutes overlay"]').click
- element.find('.dp__overlay_col', text: format('%02d', datetime.min)).click
- meridian_indicator = element.find('[aria-label="Toggle AM/PM mode"]')
- meridian_indicator.click if meridian_indicator.text != datetime.strftime('%p')
- end
- end
- # Types date into a date field.
- #
- # Usage:
- # find_datepicker('Date Picker').type_date(Date.today)
- # find_datepicker('Date Picker').type_date('2023-01-01')
- #
- def type_date(date)
- raise 'Field does not support typing dates' if !type_date?
- date = Date.parse(date) if !date.is_a?(Date)
- # TODO: Support locales other than `en`, depending on the language of the current user.
- input_element.fill_in with: date.strftime('%m/%d/%Y')
- input_element.send_keys :return
- # wait_for_test_flag("field-date-time-#{field_id}.opened")
- # close_date_picker(element)
- # wait_for_test_flag("field-date-time-#{field_id}.closed")
- maybe_wait_for_form_updater
- self # support chaining
- end
- # Types date and time into a date field.
- #
- # Usage:
- # find_datepicker('Date Time').type_datetime(DateTime.now)
- # find_datepicker('Date Picker').type_datetime('2023-01-01T09:00:00.000Z')
- #
- def type_datetime(datetime)
- raise 'Field does not support typing datetimes' if !type_datetime?
- datetime = DateTime.parse(datetime) if !datetime.is_a?(DateTime) && !datetime.is_a?(Time)
- # TODO: Support locales other than `en`, depending on the language of the current user.
- input_element.fill_in with: datetime.strftime('%m/%d/%Y %-l:%M %P')
- input_element.send_keys :return
- # wait_for_test_flag("field-date-time-#{field_id}.opened")
- # close_date_picker(element)
- # wait_for_test_flag("field-date-time-#{field_id}.closed")
- maybe_wait_for_form_updater
- self # support chaining
- end
- # Selects a choice in a radio form field.
- #
- # Usage:
- #
- # find_radio('articleSenderType').select_option('Outbound Call')
- #
- def select_choice(choice, **find_options)
- raise 'Field does not support choice selection' if !type_radio?
- input_element.find('label', exact_text: choice, **find_options).click
- maybe_wait_for_form_updater
- self # support chaining
- end
- def toggle
- raise 'Field does not support toggling' if !type_toggle? && !type_checkbox?
- element.find('label').click
- self # support chaining
- end
- def toggle_on
- raise 'Field does not support toggling on' if !type_toggle? && !type_checkbox?
- element.find('label').click if input_element['aria-checked'] == 'false' || !input_element.checked?
- self # support chaining
- end
- def toggle_off
- raise 'Field does not support toggling off' if !type_toggle? && !type_checkbox?
- element.find('label').click if input_element['aria-checked'] == 'true' || input_element.checked?
- self # support chaining
- end
- def open
- element.click
- wait_until_opened
- self # support chaining
- end
- def close
- send_keys(:escape)
- wait_until_closed
- self # support chaining
- end
- def menu_element
- return dropdown_element if desktop_view?
- dialog_element
- end
- # Dropdowns are teleported to the root element, so we must search them within the document body.
- # In order to improve the test performance, we don't do any implicit waits here.
- # Instead, we do explicit waits when opening/closing dropdowns within the actions.
- def dropdown_element
- if type_select? || type_tags? || autocomplete?
- page.find('#common-select > [role="menu"]', wait: false)
- elsif type_treeselect?
- page.find('#field-tree-select-input-dropdown > [role="menu"]', wait: false)
- end
- end
- # Dialogs are teleported to the root element, so we must search them within the document body.
- # In order to improve the test performance, we don't do any implicit waits here.
- # Instead, we do explicit waits when opening/closing dialogs within the actions.
- def dialog_element
- if type_select?
- page.find('#common-select[role="dialog"]', wait: false)
- elsif type_treeselect?
- page.find("#dialog-field-tree-select-#{field_id}", wait: false)
- elsif type_tags?
- page.find("#dialog-field-tags-#{field_id}", wait: false)
- elsif autocomplete?
- page.find("#dialog-field-auto-complete-#{field_id}", wait: false)
- end
- end
- private
- def method_missing(method_name, *, &)
- # Simulate pseudo-methods in format of `#type_[name]?` in order to determine the internal type of the field.
- if method_name.to_s =~ %r{^type_(.+)\?$}
- return element['data-type'] == $1
- end
- super
- end
- def respond_to_missing?(method_name, include_private = false)
- method_name.to_s =~ %r{^type_(.+)\?$} || super
- end
- def input?
- type_text? || type_color? || type_email? || type_number? || type_tel? || type_url? || type_password?
- end
- def autocomplete?
- type_autocomplete? || type_customer? || type_organization? || type_recipient? || type_externalDataSource?
- end
- # Input elements in supported fields define data attribute for "multiple" state.
- def multi_select?
- input_element['data-multiple'] == 'true'
- end
- def wait_until_opened
- return wait_until_opened_desktop_view if desktop_view?
- return wait_for_test_flag('common-select.opened') if type_select?
- return wait_for_test_flag("field-tree-select-#{field_id}.opened") if type_treeselect?
- return wait_for_test_flag("field-date-time-#{field_id}.opened") if type_date? || type_datetime?
- return wait_for_test_flag("field-tags-#{field_id}.opened") if type_tags?
- return wait_for_test_flag("field-auto-complete-#{field_id}.opened") if autocomplete?
- raise "Couldn't detect if element was opened"
- end
- def wait_until_opened_desktop_view
- return wait_for_test_flag('common-select.opened') if type_select? || type_tags? || autocomplete?
- return wait_for_test_flag('field-tree-select-input-dropdown.opened') if type_treeselect?
- return wait_for_test_flag('field-date-time.opened') if type_date? || type_datetime?
- raise "Couldn't detect if element was opened"
- end
- def wait_until_closed
- return wait_until_closed_desktop_view if desktop_view?
- return wait_for_test_flag('common-select.closed') if type_select?
- return wait_for_test_flag("field-tree-select-#{field_id}.closed") if type_treeselect?
- return wait_for_test_flag("field-date-time-#{field_id}.closed") if type_date? || type_datetime?
- return wait_for_test_flag("field-tags-#{field_id}.closed") if type_tags?
- return wait_for_test_flag("field-auto-complete-#{field_id}.closed") if autocomplete?
- raise "Couldn't detect if element was closed"
- end
- def wait_until_closed_desktop_view
- return wait_for_test_flag('common-select.closed') if type_select? || type_tags? || autocomplete?
- return wait_for_test_flag('field-tree-select-input-dropdown.closed') if type_treeselect?
- return wait_for_test_flag('field-date-time.closed') if type_date? || type_datetime?
- raise "Couldn't detect if element was closed"
- end
- def select_option_by_label(label, **find_options)
- within menu_element do
- find('[role="option"]', text: label, **find_options).click
- maybe_wait_for_form_updater
- end
- end
- def browse_for_option(path, **find_options)
- components = path.split('::')
- # Goes back to the root page by clicking on back button multiple times.
- rewind = proc do
- depth = components.size - 1
- depth.times do
- find('[role="button"][aria-label="Back to previous page"]').click
- end
- end
- child_menu_button = if desktop_view?
- 'div[role="button"]'
- else
- 'svg[role=link]'
- end
- components.each_with_index do |option, index|
- # Child option is always the last item.
- if index == components.size - 1
- within menu_element do
- yield option, rewind
- end
- next
- end
- # Parents come before.
- within menu_element do
- find('[role="option"] span', text: option, **find_options).sibling(child_menu_button).click
- end
- end
- end
- def search_for_tags_option(query, gql_filename: '', gql_number: 1)
- element.click
- wait_until_opened
- within menu_element do
- # Filter input in desktop view is part of the element, not the dropdown/dialog.
- searchbox = if desktop_view?
- element.find('[role="searchbox"]')
- else
- find('[role="searchbox"]')
- end
- searchbox.fill_in with: query
- send_keys(:tab)
- wait_for_autocomplete_gql(gql_filename, gql_number)
- end
- send_keys(:escape)
- wait_until_closed
- maybe_wait_for_form_updater
- self # support chaining
- end
- def search_for_autocomplete_option(query, label: query, gql_filename: '', gql_number: 1, already_open: false, use_action: false, **find_options)
- if !already_open
- element.click
- wait_until_opened
- end
- # calculate before closing, since we cannot access it, if dialog is closed
- is_multi_select = multi_select?
- within menu_element do
- # Filter input in desktop view is part of the element, not the dropdown/dialog.
- searchbox = if desktop_view?
- element.find('[role="searchbox"]')
- else
- find('[role="searchbox"]')
- end
- searchbox.fill_in with: query
- wait_for_autocomplete_gql(gql_filename, gql_number)
- if use_action
- find('[role="button"]', **find_options).click
- else
- find('[role="option"]', text: label, **find_options).click
- end
- maybe_wait_for_form_updater
- end
- send_keys(:escape) if is_multi_select
- wait_until_closed
- self # support chaining
- end
- def search_for_tags_options(queries, gql_filename: '', gql_number: 1)
- element.click
- wait_until_opened
- raise 'Field does not support multiple selection' if !multi_select?
- within menu_element do
- queries.each do |query|
- # Filter input in desktop view is part of the element, not the dropdown/dialog.
- searchbox = if desktop_view?
- element.find('[role="searchbox"]')
- else
- find('[role="searchbox"]')
- end
- searchbox.fill_in with: query
- send_keys(:tab)
- wait_for_autocomplete_gql(gql_filename, gql_number)
- end
- end
- send_keys(:escape)
- wait_until_closed
- maybe_wait_for_form_updater
- self # support chaining
- end
- def search_for_autocomplete_options(queries, labels: queries, gql_filename: '', gql_number: 1, use_action: false, **find_options)
- element.click
- wait_until_opened
- within menu_element do
- queries.each_with_index do |query, index|
- # Filter input in desktop view is part of the element, not the dropdown/dialog.
- searchbox = if desktop_view?
- element.find('[role="searchbox"]')
- else
- find('[role="searchbox"]')
- end
- searchbox.fill_in with: query
- wait_for_autocomplete_gql(gql_filename, gql_number + index)
- raise 'Field does not support multiple selection' if !multi_select?
- if use_action
- find('[role="button"]', text: 'add new email address', **find_options).click
- else
- find('[role="option"]', text: labels[index], **find_options).click
- end
- maybe_wait_for_form_updater
- if !use_action
- find('[aria-label="Clear Search"]').click
- end
- end
- end
- send_keys(:escape)
- wait_until_closed
- self # support chaining
- end
- def select_treeselect_option(label, **find_options)
- element.click
- wait_until_opened
- # calculate before closing, since we cannot access it, if dialog is closed
- is_multi_select = multi_select?
- browse_for_option(label, **find_options) do |option|
- find('[role="option"]', text: option, **find_options).click
- maybe_wait_for_form_updater
- end
- send_keys(:escape) if is_multi_select
- wait_until_closed
- self # support chaining
- end
- def select_tags_option(label, **find_options)
- element.click
- wait_until_opened
- select_option_by_label(label, **find_options)
- send_keys(:escape)
- wait_until_closed
- self # support chaining
- end
- def select_autocomplete_option(label, **find_options)
- element.click
- wait_until_opened
- # calculate before closing, since we cannot access it, if dialog is closed
- is_multi_select = multi_select?
- select_option_by_label(label, **find_options)
- send_keys(:escape) if is_multi_select
- wait_until_closed
- self # support chaining
- end
- def select_treeselect_options(labels, **find_options)
- element.click
- wait_until_opened
- raise 'Field does not support multiple selection' if !multi_select?
- labels.each do |label|
- browse_for_option(label, **find_options) do |option, rewind|
- find('[role="option"]', text: option, **find_options).click
- maybe_wait_for_form_updater
- rewind.call
- end
- end
- send_keys(:escape)
- wait_until_closed
- self # support chaining
- end
- def select_tags_options(labels, **find_options)
- element.click
- wait_until_opened
- raise 'Field does not support multiple selection' if !multi_select?
- labels.each do |label|
- select_option_by_label(label, **find_options)
- end
- send_keys(:escape)
- wait_until_closed
- self # support chaining
- end
- def select_autocomplete_options(labels, **find_options)
- element.click
- wait_until_opened
- raise 'Field does not support multiple selection' if !multi_select?
- labels.each do |label|
- select_option_by_label(label, **find_options)
- end
- send_keys(:escape)
- wait_until_closed
- self # support chaining
- end
- # If a GraphQL filename is passed, we will explicitly wait for it here.
- # Otherwise, we will implicitly wait for a query depending on the type of the field.
- # If no waits are to be done, we display a friendly warning to devs, since this can lead to some instability.
- # In form context, expected response number will be automatically increased and tracked.
- def wait_for_autocomplete_gql(gql_filename, gql_number)
- gql_number = autocomplete_gql_number(gql_filename) || gql_number
- if gql_filename.present?
- wait_for_gql(gql_filename, number: gql_number)
- elsif type_customer?
- query_name = if desktop_view?
- 'shared/components/Form/fields/FieldCustomer/graphql/queries/autocompleteSearch/generic.graphql'
- else
- 'shared/components/Form/fields/FieldCustomer/graphql/queries/autocompleteSearch/user.graphql'
- end
- wait_for_gql(query_name, number: gql_number)
- elsif type_organization?
- wait_for_gql('shared/components/Form/fields/FieldOrganization/graphql/queries/autocompleteSearch/organization.graphql', number: gql_number)
- elsif type_recipient?
- wait_for_gql('shared/components/Form/fields/FieldRecipient/graphql/queries/autocompleteSearch/recipient.graphql', number: gql_number)
- elsif type_externalDataSource?
- wait_for_gql('shared/components/Form/fields/FieldExternalDataSource/graphql/queries/autocompleteSearchObjectAttributeExternalDataSource.graphql', number: gql_number)
- elsif type_tags?
- # NB: tags autocomplete query fires only once?!
- wait_for_gql('shared/entities/tags/graphql/queries/autocompleteTags.graphql', number: 1, skip_clearing: true)
- else
- warn 'Warning: missing `wait_for_gql` in `search_for_autocomplete_option()`, might lead to instability'
- end
- end
- def autocomplete_gql_number(gql_filename)
- return nil if form_context.nil?
- return form_context.form_gql_number(:autocomplete) if gql_filename.present?
- return form_context.form_gql_number(:customer) if type_customer?
- return form_context.form_gql_number(:organization) if type_organization?
- return form_context.form_gql_number(:recipient) if type_recipient?
- return form_context.form_gql_number(:externalDataSource) if type_externalDataSource?
- form_context.form_gql_number(:tags) if type_tags?
- end
- def triggers_form_updater?
- element['data-triggers-form-updater'] == 'true'
- end
- def maybe_wait_for_form_updater
- return if form_context.nil? || !triggers_form_updater?
- gql_number = form_context.form_gql_number(:form_updater)
- wait_for_form_updater(gql_number)
- end
- # Click on the upper left corner of the date picker field to close it.
- def close_date_picker(element)
- element_width = element.native.size.width.to_i
- element_height = element.native.size.height.to_i
- element.click(x: -element_width / 2, y: -element_height / 2)
- end
- def clear_date
- element.find('[role="button"][aria-label="Clear Selection"]').click
- maybe_wait_for_form_updater
- self # support chaining
- end
- def desktop_view?
- RSpec.current_example.metadata[:app] == :desktop_view
- end
- end
- class ZammadFormContext
- attr_reader :context
- def initialize
- @context = {}
- end
- def init_form_updater(number)
- context[:gql_number] = {}
- context[:gql_number][:form_updater] = number
- end
- def form_gql_number(name)
- if context[:gql_number].nil?
- context[:gql_number] = {}
- end
- if context[:gql_number][name].nil?
- context[:gql_number][name] = 1
- else
- context[:gql_number][name] += 1
- end
- context[:gql_number][name]
- end
- end
- RSpec.configure do |config|
- config.include FormHelpers, type: :system, app: :mobile
- config.include FormHelpers, type: :system, app: :desktop_view
- end
|