View Source PhoenixFormAwesomplete (PhoenixFormAwesomplete v1.0.2)

PhoenixFormAwesomplete is a Phoenix function component that utilizes Lea Verou's autocomplete / autosuggest / typeahead / inputsearch Awesomplete widget, complying to accessibility standards (WCAG 2, Section 508).

It comes with an AwesompleteUtil javascript library which adds the following features:

  • Dynamic remote data loading; based on what is typed-in it performs an ajax lookup.
  • Allow HTML markup in the shown items. Show value with description. Optionally search in the description text.
  • Show when there is an exact match.
  • Show when there isn't a match.
  • When there is an exact match show related data (supplied in the remote data) in other parts of the page.
  • Select the highlighted item with the tab-key.

Online demo

Use cases

An autocomplete component is a free text input that provides suggestions while typing. It is not necessary to choose one of the suggestions, but of course a form validator can reject disallowed input.

The HTML combobox is very suitable for this. The combobox looks like a <select> but it doesn't limit the input. It consists of a <input type="text"> or <input type="search"> element combined with a <datalist> with <option> elements. The options cannot contain HTML markup.

When using LiveView the datalist of the HTML combobox can be made dynamic, with different suggestions based on the typed input. Like this dictionary search demo. However, when options are used with a text that differs from the value, the combobox will behave different in different browsers.

When the standard combobox doesn't meet the requirements, a solution involving javascript can be used. For LiveView there is for example this Live Select component, but this component is not compliant with accessibility standards.

This Awesomplete component can be applied for any of these cases:

  • Use outside and inside LiveView. It is even possible to make a HEEx fragment with autocomplete that works in both. See the next chapter.
  • It is specially suitable for suggestions supplied by HTTP web services that produce JSON. Although the default is to use Ajax calls, it is possible to use Phoenix channels. And, as long as the responses can be converted to Javascript arrays, it is also fine.
  • When accessibility is important; the widget has been tested for accessibility (WCAG 2, Section 508).
  • The list with suggestions can be customized, for example to show an extra description. Any HTML can be used in the suggestions.
  • It can give suggestions for an input field with multiple values.
  • It doesn't force the user to pick one of the suggestions; other values can be entered.
  • It can highlight the input field when there is a match (green) or when there isn't (red).
  • The client stops interacting with the backend, and filters on it's own when enough characters have been typed. This is when the suggestion list has become smaller than the search result limit. The client side filter is not affected by network latency, so it responds really quick.
  • Search requests with their responses can be cached by the browser, if the web service sets HTTP Cache headers.
  • A service worker can also be used for caching, if the standard ajax function is replaced with a function that uses the fetch api.
  • It can fill dependent readonly fields/tags. The typical example would be a productcode with a product description shown in order lines. For the existing order lines the database can join the product description to be shown on the screen, but for new entries and when changing the productcode it has te be dynamicly looked up. The description can be shown in the suggestion list, and when one suggestion is selected the description can be shown near the combobox.

Phoenix function components

In HEEx templates and ~H sigils, function components offer a method of reuse. And they make the HEEx markup arguable more readable.

Use both inside and outside LiveView - via hooks

The beauty of the client hooks is that this can be used both inside and outside LiveView as they both use the mounted callback. This makes it possible define a reusable component with autocomplete fields, which can be incorporated inside and outside LiveView.

Example:

<.simple_form
  for={@form}
  id="list-form"
  phx-target={@myself}
  phx-change="validate"
  phx-submit="save"
>

  <.input field={@form[:country]} type="autocomplete" placeholder="Country" phx-debounce="blur" />

  <.autocomplete forField={@form[:country]}
                 url="https://restcountries.com/v2/all"
                 loadall="true"
                 prepop="true"
                 minChars="1"
                 maxItems="8"
                 value="name"
                 />

</.simple_form>

Why are hooks used instead of dynamically generated scripts?

For security reasons, LiveView doesn't execute the javascript in dynamicly loaded script tags. Adding new javascript after the page is loaded via the HTTP-request is what every malicious Cross Site Scripting (XSS) code tries to do. Via the Content-Security-Policy HTTP-header, it is possible to prevent dynamicly loaded scripts to be executed. In LiveView the javascript code is loaded as a static asset, and client hooks via the phx-hook can be used to execute javascript code for dynamicly added DOM elements. Instead of generating javascript code for every autocomplete field, a javascript hook is used which gets it's parameters at runtime.

Use outside LiveView, a.k.a. "dead" views - via page scripts

The advantage of embedded page scripts is that anonymous functions and functions defined on the same page can be used. It also offers a smoother migration path from PhoenixFormAwesomplete version 0.1. The EEx templates can be rewritten to use function components. The input tag is separated from the script tag of Awesomplete, which makes it easier to customize the style of the input field.

The <.autocomplete> tag in the HEEX template will produce a script in the HTML page.

Example:

<.simple_form :let={f} for={@changeset} action={@action}>

  <.input field={f[:country]} type="text" label="Country" />

  <.autocomplete forField={f[:country]}
                 url="https://restcountries.com/v2/all"
                 loadall="true"
                 prepop="true"
                 minChars="1"
                 maxItems="8"
                 value="name"
                 nonce={@script_src_nonce}
                 />

</.simple_form>

The nonce in the example above, is to allow the inline script to be executed in combination with a Content-Security-Policy that doesn't allow unsafe evals or unsafe inline scripts.

Security

Content-Security-Policy

As mentioned before, use a safe Content-Security-Policy (CSP):

  • do not allow unsafe eval
  • do not allow unsafe inline scripts
  • In the connect_src whitelist only the url's of trusted sites. Make sure that the url's of Awesomplete cannot be tampered with.

Trusted web services

Use only trusted web services. As a safety measure against cross-site scripting (XSS) it is possible to sanatize or escape HTML in the JSON responses via a convertResponse function.

When an external web service is used directly, than this service will not only see the searched text but also the client IP address.

Most web services can only be accessed with a session cookie or a valid token in the request header. Use a custom ajax function to set withCredential to true on the XMLHttpRequest object to send session cookies, or set the necessairy request headers when XHR is in opened state.

Installation

Installation for using both inside and outside LiveView

  • Add phoenix_form_awesompleteto the list of dependencies in mix.exs:

    def deps do
      [
      {:phoenix_form_awesomplete, "~> 1.0"}
      ]
    end
  • Run

    mix deps.get
    
  • Add the import statement in assets/js/app.js

    import { AwesompleteUtil, attachAwesomplete, copyValueToId } from "phoenix_form_awesomplete"
  • Add the Hooks Autocomplete and AutocompleteCopyValueToId in assets/js/app.js and pass the hooks to the LiveSocket

    let Hooks = {}
    
    // modified the LiveSocket params to add hooks
    let liveSocket = new LiveSocket("/live", Socket, {
      longPollFallbackMs: 2500,
      params: {_csrf_token: csrfToken},
      hooks: Hooks
    })
    
    const AU = AwesompleteUtil
        , customAwesompleteContext = {
    
      // These functions and objects can be referenced by name in the autocomplete function components.
      // This list can be customized.
    
      filterContains:   AU.filterContains
    , filterStartsWith: AU.filterStartsWith
    , filterWords:      AU.filterWords
    , filterOff:        AU.filterOff
    
    , item:             AU.item          // does NOT mark matching text
    // , itemContains:     AU.itemContains  // this is the default, no need to specify it.
    , itemStartsWith:   AU.itemStartsWith
    , itemMarkAll:      AU.itemMarkAll   // also mark matching text inside the description
    , itemWords:        AU.itemWords     // mark matching words
    
    , jsonFlatten:      AU.jsonFlatten   // Utility to flatten deep JSON structures
    
      // add your custom functions and/or lists here
    
    }
    
    Hooks.Autocomplete = {
      mounted() { attachAwesomplete(this.el, customAwesompleteContext, {} /* defaultSettings */ ) }
    }
    
    Hooks.AutocompleteCopyValueToId = {
      mounted() { copyValueToId(this.el) }
    }
  • Add these function components in lib/<your_project>_web/components/core_components.ex:

    @awesomplete PhoenixFormAwesomplete.HookComponent
    
    defdelegate autocomplete(assigns), to: @awesomplete
    defdelegate copy_value_to_id(assigns), to: @awesomplete
    defdelegate copy_value_to_field(assigns), to: @awesomplete
  • Add a new input type called autocomplete in lib/<your_project>_web/components/core_components.ex:

    The core_components.ex file may look different for every project, but here we assume there is a function to handle input for type="text". Copy this function to handle the new autocomplete type like in the example below. Pattern match the new type in the function argument.

    Surrounding the input tag there must be a span or div tag with a unique id, and with the phx‑update="ignore" attribute. LiveView should not do DOM updates on the input tag that is manipulated by the Awesomplete widget. The errors below the input field can and should be updated by LiveView.

    def input(%{type: "autocomplete"} = assigns) do
      assigns = assign(assigns, span_id: assigns.id <> "-domspace")
      ~H"""
      <div>
        <.label for={@id}><%= @label %></.label>
        <span phx-update="ignore" id={@span_id}>
          <input
            type="text"
            name={@name}
            id={@id}
            value={Phoenix.HTML.Form.normalize_value(@type, @value)}
            class={[
              "mt-2 block w-full rounded-lg text-zinc-900 focus:ring-0 sm:text-sm sm:leading-6"
            ]}
            {@rest}
          />
        </span>
        <.error :for={msg <- @errors}><%= msg %></.error>
      </div>
      """
    end

    and add the autocomplete in the allowed types:

    attr :type, :string,
      default: "text",
      values: ~w(autocomplete checkbox color date datetime-local email file hidden month number password
                 range radio search select tel text textarea time url week)
  • Add in assets/css/app.css

    @import "../../deps/phoenix_form_awesomplete/priv/static/awesomplete_bundle.css";

    To modify this file, copy it to your assets/css directory and import that css file.

  • Run

    mix phx.server
    

Installation for use outside LiveView only

  • Add phoenix_form_awesompleteto the list of dependencies in mix.exs:
    def deps do
      [
      {:phoenix_form_awesomplete, "~> 1.0"}
      ]
    end
  • Run
    mix deps.get
    
  • Add these function components in lib/<your_project>_web/components/core_components.ex:
    @awesomplete PhoenixFormAwesomplete.EmbedScriptComponent
    
    defdelegate autocomplete(assigns), to: @awesomplete
    defdelegate copy_value_to_id(assigns), to: @awesomplete
    defdelegate copy_value_to_field(assigns), to: @awesomplete
  • Add this code in assets/js/app.js
    import { Awesomplete, AwesompleteUtil } from "phoenix_form_awesomplete"
  • In lib/<your_project>_web/components/layouts/root.html.heex remove 'defer' for app.js:
    <script phx-track-static src={~p"/assets/js/app.js"}></script>
    Awesomplete and AwesompleteUtil must be loaded before running the inline scripts.
  • Add in assets/css/app.css
    @import "../../deps/phoenix_form_awesomplete/priv/static/awesomplete_bundle.css";
    To modify this file, copy it to your assets/css directory and import that css file.
  • Run
    mix phx.server
    

Accessibility

The Awesomplete widget is accessible (Section 508, WCAG). However, when using custom HTML in the suggestion list, this solution must be tested separately for compliance.

The red/green border color to indicate if there is a match or not is not helpful for people with red-green color blindness. The 2 pixel border size might be not enough for people with a low vision. And if they use a screen reader, the screen reader will paint it's own border around the focused element, hiding the red/green border color.

FAQ

Is it possible to use Phoenix channels instead of ajax web service calls?

Yes, the ajax call can be replaced.

As url we will use the keyword livesocket. Add this in assets/js/app.js after the declaration of liveSocket

function ajax2live(url, urlEnd, val, fn, xhr) {
  if (url && url.startsWith('livesocket:')) {
      const awe = this
          , phxEvent = url.substr(url.indexOf(':') + 1)
          , phxData = {'value':val, 'id':awe.input.id};
      // secretly use this internal function to push events
      liveSocket.execJSHookPush(awe.input, phxEvent, phxData, () => { console.log('sent ' + val); } );
  }
  else
  {
      AwesompleteUtil.ajax(url, urlEnd, val, fn, xhr);
  }
};

In assets/js/app.js add in customAwesompleteContext the name of the function above

  , ajax2live:  ajax2live

In assets/js/app.js add in the mounted function, add a function to handle server response

const awe = attachAwesomplete(this.el, customAwesompleteContext, {} /* defaultSettings */ );
this.handleEvent(`update-list-${awe.input.id}`,
  ({searchResult, searchPhrase}) => { AwesompleteUtil.updateList(awe, searchResult, searchPhrase); }
);

In the HEEx template, in the .input add phx-target and in the .autocomplete refer to the new ajax function and change the url

<.input field={@form[:country]} type="text" placeholder="Country" phx-target={@myself} phx-debounce="blur" />

<.autocomplete
  forField={@form[:country]}
  url="livesocket:update-country-list"
  minChars="1"
  maxItems="5"
  ajax="ajax2live"
  />

Handle the event in LiveView:

def handle_event("update-country-list", %{"value" => val, "id" => id}, socket) do
  newCountryList = ... your query logic here ...
  {:noreply, push_event(socket, "update-list-#{id}", %{searchResult: newCountryList, searchPhrase: val})}
end

On mobile devices the suggestions are shown behind the virtual keyboard. Do you have a solution?

With a bit of javascript it is possible to scroll up the input field to the top of the screen when suggestions are shown on small devices.

window.addEventListener('awesomplete-open', (el) => {
  if (window.innerWidth < 577 && window.innerHeight < 800) el.target.scrollIntoView();
});

Put this in scroll.js and add this javascript file in the header of the page.

Also, set maxItems to a low value when it is used on a small device.

Is it possible to group suggestions?

You can add the group in the description to show for each suggestion to which group it belongs, but the Awesomplete widget has not the option to show suggestions in groups.

Alternatives:

  • Split the input in two fields: one to select the group, and in the autocomplete field show only items of the selected group.
  • Use the HTML select element. It supports the optgroup tag.
  • This javascript component autocomplete(r) from Denis Taran has a grouping feature.

Raw core example

IEx

iex> {:safe, [input, script]} = PhoenixFormAwesomplete.awesomplete(:user, :drinks,
...> ["data-list": "beer, gin, soda, sprite, water, vodga, whine, whisky"],
...> %{ minChars: 1 } )
iex> to_string input
"<input data-list=\"beer, gin, soda, sprite, water, vodga, whine, whisky\"" <>
" id=\"user_drinks\" name=\"user[drinks]\" type=\"text\">"
iex> script
"<script>AwesompleteUtil.start('#user_drinks', {}, {minChars: 1});</script>"

The first three parameters are passed on unchanged to the Phoenix form text_input which generates the input tag. minChars is an option for the Awesomplete object which is started with inline javascript. Just adding the multiple option changes the generated javascript code completely, the PhoenixFormAwesomplete module takes care of that. Instead of an server side generated data-list it is possible to specify an url of a JSON web service and let the client-code lookup the data list on-demand while typing. Look at the live examples with code.

It is possible to use aliases for the javascript library references in the generated page scripts via the environment variables util and awesomplete. The default names, AwesompleteUtil and Awesomplete respectively, are a bit long. This can shorten the average page size. For example use this javascript:

 var AU = AwesompleteUtil, AW = Awesomplete;

and change the variables via the application config:

 :phoenix_form_awesomplete, util:         "AU"
 :phoenix_form_awesomplete, awesomplete:  "AW"

After changing the config/config.exs run:

 mix deps.compile --force phoenix_form_awesomplete

Summary

Functions

This method generates an input tag and inline javascript code that starts Awesomplete. Use this in (L)EEx templates. For HEEx templates it recommended to use <.input in combination with awesomplete_script/2.

This method generates javascript code for using Awesomplete(Util).

This method generates javascript code for using Awesomplete(Util).

This method generates a script tag with javascript code for using Awesomplete(Util).

This method generates a script tag with javascript code for using Awesomplete(Util). As awesomplete_script/2 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField. The form parameter is either a Phoenix.HTML.Form struct or an atom.

As copy_value_to_field/3 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField's. The source_form and target_form parameters are either a Phoenix.HTML.Form struct or an atom.

Same as copy_to_field/5 but with an additional last argument for the script attributes.

As copy_value_to_id/3 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField. The source_form parameter is either a Phoenix.HTML.Form struct or an atom.

As copy_value_to_id_js/3 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField. The source_form and target_form parameters are either a Phoenix.HTML.Form struct or an atom.

Same as copy_to_id/4 but with an additional last argument for the script attributes.

Create script tag with javascript that listens to awesomplete-prepop and awesomplete-match events on the source form field, and copies the data_field to the target form field.

Same as copy_value_to_field/3 but with an additional last argument for the script attributes.

Create script tag with javascript that listens to awesomplete-prepop and awesomplete-match events, and copies the data_field to the DOM element with the given target id. The target id is passed to the DOM document querySelector, and is typically set as a hash character with an element id. The target_id can also be a javascript function.

Create javascript that listens to awesomplete-prepop and awesomplete-match events, and copies the data_field to the DOM element with the given target id. The target id is passed to the DOM document querySelector, and is typically set as a hash character with an element id. The target_id can also be a javascript function.

Same as copy_value_to_id/4 but with an additional last argument for script attributes.

Create script tag with the supplied script. No defer or async because this is used for inline script.

Same as script/1 with a second argument for the script attributes.

Functions

Link to this function

awesomplete(form, field, opts \\ [], awesomplete_opts)

View Source

This method generates an input tag and inline javascript code that starts Awesomplete. Use this in (L)EEx templates. For HEEx templates it recommended to use <.input in combination with awesomplete_script/2.

Awesomplete options:

  • ajax - Replace ajax function. Supplied function receives these parameters: (url, urlEnd, val, fn, xhr). fn is the callback function. Default: AwesompleteUtil.ajax.
  • assign - Assign the Awesomplete object to a variable. true/false/name. If true the variable name will be 'awe_' + id of the input tag. Default: false
  • autoFirst - true/false. Automatically select the first element. Default: false.
  • combobox - Id of the combobox button. true/false/id. If true the assumed button id is 'awe_btn_' + id of the input tag. Default: false
  • container - Container function as defined in Awesomplete. By default a div element is added as the parent of the input element.
  • convertInput - Convert input function which receives the input text as parameter. This function is used normalize the input text. Internally convert the input text for search calls and for comparison with the suggestions. By default it trims the input and converts it to lowercase for a case-insensitive comparison. It is applied to both the input text and the suggestion text before comparing. In advanced cases like the multiple values, the convertInput is used to extract the search text.
  • convertResponse - Convert JSON response from ajax calls. This function is called with the parsed JSON, and allows conversion of the data before further processing. Default: nil - no conversion.
  • data - Data function as defined in Awesomplete
  • debounce - Time in milliseconds to wait for additional user input before doing the ajax call to retrieve suggestions. It limits the rate at which the json service is called per user session.
  • descr - Name of the field in the data list (the JSON response) that contains the description text to show below the value in the suggestion list. Default: no description
  • descrSearch - true/false. Filter must also search the input value in the description field. Default: false
  • filter - Filter function as defined in Awesomplete. Mostly use Awesomplete.FILTER_STARTSWITH or Awesomplete.FILTER_CONTAINS. If label is different as value, filter on value with AweompleteUtil.filterStartsWith, AwesompleteUtil.filterContains or AwesompleteUtil.filterWords. To turn off filtering, use AwesompleteUtil.filterOff.
  • item - Item function as defined in Awesomplete with parameters text, input and itemId. Default is to highlight all occurrences of the input text. Use AwesompleteUtil.itemStartsWith or AwesompleteUtil.itemWords if that matches with the used filter.
  • label - Name of the field in the data list (the JSON response) that contains the text that should be shown instead of the value.
  • list - Data list as defined in Awesomplete.
  • listLabel - Denotes a label to be used as aria-label on the generated autocomplete list.
  • loadall - true/false. Use true if the data list contains all items, and the input value should not be used in ajax calls. Default: false
  • limit - number. If a limit is specified, and the number of items returned by the server is equal or more as this limit, the AwesompleteUtil code assumes that there are more results, so it will re-query if more characters are typed to get more refined results. The limit:1 tells that not more than 1 result is expected, so the json service doesn’t have to return an array. With limit:0 it will always re-query if more characters are typed and the result doesn't have to be an array either. Limit:-1 will always requery and the expected result is an array. When no limit is specified, the code assumes that all possible suggestions are returned based on the typed characters, and it will not re-query if more characters are typed. It uses the specified filter for the suggestions in the dropdown. Default: no limit
  • maxItems - Maximum number of suggestions to display. Default: 10
  • minChars - Minimum characters the user has to type before the autocomplete popup shows up. Default: 2
  • multiple - true/false/characters. Separators to allow multiple values. If true, the separator will be the space character. Default: false
  • nonce - Content-Security-Policy nonce attribute for the script tag. Default: no nonce. If specified it must contain a non-empty value.
  • prepop - true/false. If true do lookup initial/autofilled value and send awesomplete-prepop event. Default: false
  • replace - Replace function as defined in Awesomplete. The replace function will be called for suggestions, to determine whether the input text matches a suggestion after replacement.
  • sort - Sort function as defined in Awesomplete
  • statusNoResults - Screen reader text to replace the default: 'No results found'
  • statusTypeXChar - Screen reader text to replace the default: 'Type {0} or more characters for results'. The placeholder {0} will be replaced with the minimum number of characters (minChars).
  • statusXResults - Screen reader text to replace the default: '{0} results found'. The placeholder {0} will be replaced with the number of results.
  • url - url for ajax calls.
  • urlEnd - Addition at the end of the url for the ajax call, after the input value. Or a function, which receives the value and must return the last part of the url.
  • value - Name of the field in the data list (the JSON response) that contains the value.

Example

iex> {:safe, [inp, scr]} = PhoenixFormAwesomplete.awesomplete(:user, :eyes,
...> ["data-list": "blue, brown, green"],
...>  %{ minChars: 1, multiple: ",;", nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S" } )
iex> to_string inp
"<input data-list=\"blue, brown, green\" id=\"user_eyes\" name=\"user[eyes]\" type=\"text\">"
iex> scr
"<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.start('#user_eyes', " <>
"{convertInput: function(input) {" <>
" return input.replace(/[,;]\\s*$/, '').match(/[^,;]*$/)[0].trim().toLowerCase(); }}, " <>
"{minChars: 1, " <>
"replace: function(data) {" <>
" var text=data.value;" <>
" this.input.value = this.input.value.match(/^.+[,;]\\s*|/)[0] + text + ', '; }, " <>
"filter: function(data, input) {" <>
" return Awesomplete.FILTER_CONTAINS(data, input.match(/[^,;]*([,;]\\s*)?$/)[0]); }, " <>
"item: function(text, input) {" <>
" return AwesompleteUtil.itemContains(text, input.match(/[^,;]*([,;]\\s*)?$/)[0]); }});" <>
"</script>"
Link to this function

awesomplete_js(ff, awesomplete_opts)

View Source

This method generates javascript code for using Awesomplete(Util).

Example

iex> ff = %Phoenix.HTML.FormField{form: "user", field: "hobby", id: "user_hobby", name: "user[hobby]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.awesomplete_js(ff , %{ minChars: 1 } )
"AwesompleteUtil.start('#user_hobby', {}, {minChars: 1});"
Link to this function

awesomplete_js(form, field, awesomplete_opts)

View Source

This method generates javascript code for using Awesomplete(Util).

Example

iex> PhoenixFormAwesomplete.awesomplete_js(:user, :hobby, %{ minChars: 1 } )
"AwesompleteUtil.start('#user_hobby', {}, {minChars: 1});"
Link to this function

awesomplete_script(ff, awesomplete_opts)

View Source

This method generates a script tag with javascript code for using Awesomplete(Util).

Example

iex> ff = %Phoenix.HTML.FormField{form: "user", field: "hobby", id: "user_hobby", name: "user[hobby]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.awesomplete_script(ff, %{ minChars: 1 } )
{:safe,
 "<script>AwesompleteUtil.start('#user_hobby', {}, {minChars: 1});</script>"}

iex> PhoenixFormAwesomplete.awesomplete_script(:user, :hobby, %{ minChars: 1, nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S" } )
{:safe,
 "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.start('#user_hobby', {}, {minChars: 1});</script>"}
Link to this function

awesomplete_script(form, field, awesomplete_opts)

View Source

This method generates a script tag with javascript code for using Awesomplete(Util). As awesomplete_script/2 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField. The form parameter is either a Phoenix.HTML.Form struct or an atom.

Example

iex> PhoenixFormAwesomplete.awesomplete_script(:user, :hobby, %{ minChars: 1 } )
{:safe,
 "<script>AwesompleteUtil.start('#user_hobby', {}, {minChars: 1});</script>"}

iex> PhoenixFormAwesomplete.awesomplete_script(:user, :hobby, %{ minChars: 1, nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S" } )
{:safe,
 "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.start('#user_hobby', {}, {minChars: 1});</script>"}
Link to this function

copy_to_field(source_form, source_field, data_field \\ nil, target_form, target_field)

View Source

As copy_value_to_field/3 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField's. The source_form and target_form parameters are either a Phoenix.HTML.Form struct or an atom.

Example

iex> PhoenixFormAwesomplete.copy_to_field(:user, :color, "label", :door, :paint)
{:safe,
 "<script>AwesompleteUtil.startCopy('#user_color', 'label', '#door_paint');</script>"}
Link to this function

copy_to_field_script(source_form, source_field, data_field \\ nil, target_form, target_field, script_attributes)

View Source

Same as copy_to_field/5 but with an additional last argument for the script attributes.

Example

iex> PhoenixFormAwesomplete.copy_to_field_script(:user, :color, "label", :door, :paint, [nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S"])
{:safe,
 "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.startCopy('#user_color', 'label', '#door_paint');</script>"}
Link to this function

copy_to_id(source_form, source_field, data_field \\ nil, target_id)

View Source

As copy_value_to_id/3 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField. The source_form parameter is either a Phoenix.HTML.Form struct or an atom.

Example

iex> PhoenixFormAwesomplete.copy_to_id(:user, :color, "label", "#awe-color-result")
{:safe,
 "<script>AwesompleteUtil.startCopy('#user_color', 'label', '#awe-color-result');</script>"}
Link to this function

copy_to_id_js(source_form, source_field, data_field \\ nil, target_id)

View Source

As copy_value_to_id_js/3 but with form and field parameters as used in Phoenix.HTML.Form functions instead of the Phoenix.HTML.FormField. The source_form and target_form parameters are either a Phoenix.HTML.Form struct or an atom.

Example

iex> PhoenixFormAwesomplete.copy_to_id_js(:user, :color, "label", "#awe-color-result")
"AwesompleteUtil.startCopy('#user_color', 'label', '#awe-color-result');"
Link to this function

copy_to_id_script(source_form, source_field, data_field \\ nil, target_id, script_attributes)

View Source

Same as copy_to_id/4 but with an additional last argument for the script attributes.

Example

iex> PhoenixFormAwesomplete.copy_to_id_script(:user, :color, "label", "#awe-color-result", [nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S"])
{:safe,
 "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.startCopy('#user_color', 'label', '#awe-color-result');</script>"}
Link to this function

copy_value_to_field(source_ff, data_field \\ nil, target_ff)

View Source

Create script tag with javascript that listens to awesomplete-prepop and awesomplete-match events on the source form field, and copies the data_field to the target form field.

Example

iex> ff_source = %Phoenix.HTML.FormField{form: "palet", field: "color", id: "palet_color", name: "palet[color]", errors: [], value: nil}
iex> ff_target = %Phoenix.HTML.FormField{form: "palet", field: "paint", id: "palet_paint", name: "palet[paint]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.copy_value_to_field(ff_source, "label", ff_target)
{:safe,
 "<script>AwesompleteUtil.startCopy('#palet_color', 'label', '#palet_paint');</script>"}
Link to this function

copy_value_to_field_script(source_ff, data_field \\ nil, target_ff, script_attributes)

View Source

Same as copy_value_to_field/3 but with an additional last argument for the script attributes.

Example

iex> ff_source = %Phoenix.HTML.FormField{form: "palet", field: "color", id: "palet_color", name: "palet[color]", errors: [], value: nil}
iex> ff_target = %Phoenix.HTML.FormField{form: "palet", field: "paint", id: "palet_paint", name: "palet[paint]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.copy_value_to_field_script(ff_source, "label", ff_target, [nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S"])
{:safe,
 "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.startCopy('#palet_color', 'label', '#palet_paint');</script>"}
Link to this function

copy_value_to_id(ff, data_field \\ nil, target_id)

View Source

Create script tag with javascript that listens to awesomplete-prepop and awesomplete-match events, and copies the data_field to the DOM element with the given target id. The target id is passed to the DOM document querySelector, and is typically set as a hash character with an element id. The target_id can also be a javascript function.

Example

iex> ff = %Phoenix.HTML.FormField{form: "palet", field: "color", id: "palet_color", name: "palet[color]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.copy_value_to_id(ff, "label", "#awe-color-result")
{:safe,
 "<script>AwesompleteUtil.startCopy('#palet_color', 'label', '#awe-color-result');</script>"}
Link to this function

copy_value_to_id_js(ff, data_field \\ nil, target_id)

View Source

Create javascript that listens to awesomplete-prepop and awesomplete-match events, and copies the data_field to the DOM element with the given target id. The target id is passed to the DOM document querySelector, and is typically set as a hash character with an element id. The target_id can also be a javascript function.

Example

iex> ff = %Phoenix.HTML.FormField{form: "palet", field: "color", id: "palet_color", name: "palet[color]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.copy_value_to_id_js(ff, "label", "#awe-color-result")
"AwesompleteUtil.startCopy('#palet_color', 'label', '#awe-color-result');"
Link to this function

copy_value_to_id_script(ff, data_field \\ nil, target_id, script_attributes)

View Source

Same as copy_value_to_id/4 but with an additional last argument for script attributes.

Example

iex> ff = %Phoenix.HTML.FormField{form: "palet", field: "color", id: "palet_color", name: "palet[color]", errors: [], value: nil}
iex> PhoenixFormAwesomplete.copy_value_to_id_script(ff, "label", "#awe-color-result", [nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S"])
{:safe,
 "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">AwesompleteUtil.startCopy('#palet_color', 'label', '#awe-color-result');</script>"}

Create script tag with the supplied script. No defer or async because this is used for inline script.

Example

iex> PhoenixFormAwesomplete.script("alert(1);")
{:safe, "<script>alert(1);</script>"}
Link to this function

script(script, script_attributes)

View Source

Same as script/1 with a second argument for the script attributes.

Example

iex> PhoenixFormAwesomplete.script("alert(1);" , [nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S"])
{:safe, "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">alert(1);</script>"}

iex> PhoenixFormAwesomplete.script("alert(2);" , %{nonce: "KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S"})
{:safe, "<script nonce=\"KG2FJFSN4LaCNyVRwTxRJjCB94Bdc41S\">alert(2);</script>"}

iex> PhoenixFormAwesomplete.script("alert(3);" , %{type: "module", id: "x42"})
{:safe, "<script id=\"x42\" type=\"module\">alert(3);</script>"}