blacklight 8.0.0.beta5 → 8.0.0.beta6
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.
- checksums.yaml +4 -4
- data/.gitignore +1 -0
- data/Gemfile +0 -4
- data/README.md +1 -1
- data/Rakefile +18 -0
- data/VERSION +1 -1
- data/app/assets/config/blacklight/manifest.js +3 -0
- data/app/assets/stylesheets/blacklight/_facets.scss +32 -5
- data/app/assets/stylesheets/blacklight/blacklight_defaults.scss +1 -1
- data/app/components/blacklight/document_component.rb +21 -2
- data/app/components/blacklight/response/facet_group_component.html.erb +2 -1
- data/app/helpers/blacklight/blacklight_helper_behavior.rb +0 -5
- data/app/javascript/blacklight/bookmark_toggle.js +8 -15
- data/app/javascript/blacklight/button_focus.js +10 -10
- data/app/javascript/blacklight/index.js +5 -5
- data/app/javascript/blacklight/modal.js +6 -11
- data/app/javascript/blacklight/search_context.js +51 -57
- data/app/models/concerns/blacklight/document.rb +11 -0
- data/app/views/catalog/_document.html.erb +2 -2
- data/app/views/layouts/blacklight/base.html.erb +4 -4
- data/blacklight.gemspec +1 -1
- data/config/importmap.rb +1 -1
- data/config/locales/blacklight.ar.yml +209 -226
- data/config/locales/blacklight.de.yml +197 -209
- data/config/locales/blacklight.en.yml +2 -0
- data/config/locales/blacklight.es.yml +196 -209
- data/config/locales/blacklight.fr.yml +196 -213
- data/config/locales/blacklight.hu.yml +193 -206
- data/config/locales/blacklight.it.yml +197 -209
- data/config/locales/blacklight.nl.yml +193 -206
- data/config/locales/blacklight.pt-BR.yml +195 -209
- data/config/locales/blacklight.sq.yml +193 -206
- data/config/locales/blacklight.zh.yml +194 -207
- data/lib/blacklight/configuration/view_config.rb +0 -4
- data/lib/blacklight/engine.rb +6 -7
- data/lib/generators/blacklight/assets/importmap_generator.rb +9 -2
- data/lib/generators/blacklight/assets/propshaft_generator.rb +11 -2
- data/lib/generators/blacklight/assets/sprockets_generator.rb +2 -0
- data/package.json +7 -4
- data/rollup.config.js +11 -1
- data/spec/features/sitelinks_search_box_spec.rb +13 -0
- data/spec/models/blacklight/icon_spec.rb +0 -2
- data/spec/spec_helper.rb +1 -1
- data/spec/support/view_component_test_helpers.rb +21 -0
- data/spec/test_app_templates/Gemfile.extra +4 -0
- data/spec/views/catalog/_document.html.erb_spec.rb +21 -0
- data/tasks/blacklight.rake +1 -1
- metadata +7 -10
- data/app/assets/javascripts/blacklight/blacklight.esm.js +0 -390
- data/app/assets/javascripts/blacklight/blacklight.esm.js.map +0 -1
- data/app/assets/javascripts/blacklight/blacklight.js +0 -398
- data/app/assets/javascripts/blacklight/blacklight.js.map +0 -1
- data/spec/features/sitelinks_search_box.rb +0 -13
- data/spec/support/view_component_capybara_test_helpers.rb +0 -8
| @@ -1,398 +0,0 @@ | |
| 1 | 
            -
            (function (global, factory) {
         | 
| 2 | 
            -
              typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
         | 
| 3 | 
            -
              typeof define === 'function' && define.amd ? define(factory) :
         | 
| 4 | 
            -
              (global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Blacklight = factory());
         | 
| 5 | 
            -
            })(this, (function () { 'use strict';
         | 
| 6 | 
            -
             | 
| 7 | 
            -
              const Blacklight = function() {
         | 
| 8 | 
            -
                const buffer = new Array;
         | 
| 9 | 
            -
                return {
         | 
| 10 | 
            -
                  onLoad: function(func) {
         | 
| 11 | 
            -
                    buffer.push(func);
         | 
| 12 | 
            -
                  },
         | 
| 13 | 
            -
             | 
| 14 | 
            -
                  activate: function() {
         | 
| 15 | 
            -
                    for(let i = 0; i < buffer.length; i++) {
         | 
| 16 | 
            -
                      buffer[i].call();
         | 
| 17 | 
            -
                    }
         | 
| 18 | 
            -
                  },
         | 
| 19 | 
            -
             | 
| 20 | 
            -
                  listeners: function () {
         | 
| 21 | 
            -
                    const listeners = [];
         | 
| 22 | 
            -
                    if (typeof Turbo !== 'undefined') {
         | 
| 23 | 
            -
                      listeners.push('turbo:load', 'turbo:frame-load');
         | 
| 24 | 
            -
                    } else if (typeof Turbolinks !== 'undefined' && Turbolinks.supported) {
         | 
| 25 | 
            -
                      // Turbolinks 5
         | 
| 26 | 
            -
                      if (Turbolinks.BrowserAdapter) {
         | 
| 27 | 
            -
                        listeners.push('turbolinks:load');
         | 
| 28 | 
            -
                      } else {
         | 
| 29 | 
            -
                        // Turbolinks < 5
         | 
| 30 | 
            -
                        listeners.push('page:load', 'DOMContentLoaded');
         | 
| 31 | 
            -
                      }
         | 
| 32 | 
            -
                    } else {
         | 
| 33 | 
            -
                      listeners.push('DOMContentLoaded');
         | 
| 34 | 
            -
                    }
         | 
| 35 | 
            -
             | 
| 36 | 
            -
                    return listeners;
         | 
| 37 | 
            -
                  }
         | 
| 38 | 
            -
                };
         | 
| 39 | 
            -
              }();
         | 
| 40 | 
            -
             | 
| 41 | 
            -
              // turbolinks triggers page:load events on page transition
         | 
| 42 | 
            -
              // If app isn't using turbolinks, this event will never be triggered, no prob.
         | 
| 43 | 
            -
              Blacklight.listeners().forEach(function(listener) {
         | 
| 44 | 
            -
                document.addEventListener(listener, function() {
         | 
| 45 | 
            -
                  Blacklight.activate();
         | 
| 46 | 
            -
                });
         | 
| 47 | 
            -
              });
         | 
| 48 | 
            -
             | 
| 49 | 
            -
              Blacklight.onLoad(function () {
         | 
| 50 | 
            -
                const elem = document.querySelector('.no-js');
         | 
| 51 | 
            -
             | 
| 52 | 
            -
                // The "no-js" class may already have been removed because this function is
         | 
| 53 | 
            -
                // run on every turbo:load event, in that case, it won't find an element.
         | 
| 54 | 
            -
                if (!elem) return;
         | 
| 55 | 
            -
             | 
| 56 | 
            -
                elem.classList.remove('no-js');
         | 
| 57 | 
            -
                elem.classList.add('js');
         | 
| 58 | 
            -
              });
         | 
| 59 | 
            -
             | 
| 60 | 
            -
              /* Converts a "toggle" form, with single submit button to add/remove
         | 
| 61 | 
            -
                 something, like used for Bookmarks, into an AJAXy checkbox instead.
         | 
| 62 | 
            -
                 Apply to a form. Does require certain assumption about the form:
         | 
| 63 | 
            -
                  1) The same form 'action' href must be used for both ADD and REMOVE
         | 
| 64 | 
            -
                     actions, with the different being the hidden input name="_method"
         | 
| 65 | 
            -
                     being set to "put" or "delete" -- that's the Rails method to pretend
         | 
| 66 | 
            -
                     to be doing a certain HTTP verb. So same URL, PUT to add, DELETE
         | 
| 67 | 
            -
                     to remove. This plugin assumes that.
         | 
| 68 | 
            -
                     Plus, the form this is applied to should provide a data-doc-id
         | 
| 69 | 
            -
                     attribute (HTML5-style doc-*) that contains the id/primary key
         | 
| 70 | 
            -
                     of the object in question -- used by plugin for a unique value for
         | 
| 71 | 
            -
                     DOM id's.
         | 
| 72 | 
            -
                Uses HTML for a checkbox compatible with Bootstrap 4.
         | 
| 73 | 
            -
                 new CheckboxSubmit(document.querySelector('form.something')).render()
         | 
| 74 | 
            -
              */
         | 
| 75 | 
            -
              class CheckboxSubmit {
         | 
| 76 | 
            -
                constructor(form) {
         | 
| 77 | 
            -
                  this.form = form;
         | 
| 78 | 
            -
                }
         | 
| 79 | 
            -
             | 
| 80 | 
            -
                async clicked(evt) {
         | 
| 81 | 
            -
                  this.spanTarget.innerHTML = this.form.getAttribute('data-inprogress');
         | 
| 82 | 
            -
                  this.labelTarget.setAttribute('disabled', 'disabled');
         | 
| 83 | 
            -
                  this.checkboxTarget.setAttribute('disabled', 'disabled');
         | 
| 84 | 
            -
                  const response = await fetch(this.formTarget.getAttribute('action'), {
         | 
| 85 | 
            -
                    body: new FormData(this.formTarget),
         | 
| 86 | 
            -
                    method: this.formTarget.getAttribute('method').toUpperCase(),
         | 
| 87 | 
            -
                    headers: {
         | 
| 88 | 
            -
                      'Accept': 'application/json',
         | 
| 89 | 
            -
                      'X-Requested-With': 'XMLHttpRequest',
         | 
| 90 | 
            -
                      'X-CSRF-Token': document.querySelector('meta[name=csrf-token]')?.content
         | 
| 91 | 
            -
                    }
         | 
| 92 | 
            -
                  });
         | 
| 93 | 
            -
                  this.labelTarget.removeAttribute('disabled');
         | 
| 94 | 
            -
                  this.checkboxTarget.removeAttribute('disabled');
         | 
| 95 | 
            -
                  if (response.ok) {
         | 
| 96 | 
            -
                    const json = await response.json();
         | 
| 97 | 
            -
                    this.updateStateFor(!this.checked);
         | 
| 98 | 
            -
                    document.querySelector('[data-role=bookmark-counter]').innerHTML = json.bookmarks.count;
         | 
| 99 | 
            -
                  } else {
         | 
| 100 | 
            -
                    alert('Error');
         | 
| 101 | 
            -
                  }
         | 
| 102 | 
            -
                }
         | 
| 103 | 
            -
             | 
| 104 | 
            -
                get checked() {
         | 
| 105 | 
            -
                  return (this.form.querySelectorAll('input[name=_method][value=delete]').length != 0)
         | 
| 106 | 
            -
                }
         | 
| 107 | 
            -
             | 
| 108 | 
            -
                get formTarget() {
         | 
| 109 | 
            -
                  return this.form
         | 
| 110 | 
            -
                }
         | 
| 111 | 
            -
             | 
| 112 | 
            -
                get labelTarget() {
         | 
| 113 | 
            -
                  return this.form.querySelector('[data-checkboxsubmit-target="label"]')
         | 
| 114 | 
            -
                }
         | 
| 115 | 
            -
             | 
| 116 | 
            -
                get checkboxTarget() {
         | 
| 117 | 
            -
                  return this.form.querySelector('[data-checkboxsubmit-target="checkbox"]')
         | 
| 118 | 
            -
                }
         | 
| 119 | 
            -
             | 
| 120 | 
            -
                get spanTarget() {
         | 
| 121 | 
            -
                  return this.form.querySelector('[data-checkboxsubmit-target="span"]')
         | 
| 122 | 
            -
                }
         | 
| 123 | 
            -
             | 
| 124 | 
            -
                updateStateFor(state) {
         | 
| 125 | 
            -
                  this.checkboxTarget.checked = state;
         | 
| 126 | 
            -
             | 
| 127 | 
            -
                  if (state) {
         | 
| 128 | 
            -
                    this.labelTarget.classList.add('checked');
         | 
| 129 | 
            -
                    //Set the Rails hidden field that fakes an HTTP verb
         | 
| 130 | 
            -
                    //properly for current state action.
         | 
| 131 | 
            -
                    this.formTarget.querySelector('input[name=_method]').value = 'delete';
         | 
| 132 | 
            -
                    this.spanTarget.innerHTML = this.form.getAttribute('data-present');
         | 
| 133 | 
            -
                  } else {
         | 
| 134 | 
            -
                    this.labelTarget.classList.remove('checked');
         | 
| 135 | 
            -
                    this.formTarget.querySelector('input[name=_method]').value = 'put';
         | 
| 136 | 
            -
                    this.spanTarget.innerHTML = this.form.getAttribute('data-absent');
         | 
| 137 | 
            -
                  }
         | 
| 138 | 
            -
                }
         | 
| 139 | 
            -
              }
         | 
| 140 | 
            -
             | 
| 141 | 
            -
              const BookmarkToggle = (() => {
         | 
| 142 | 
            -
                  // change form submit toggle to checkbox
         | 
| 143 | 
            -
                  Blacklight.doBookmarkToggleBehavior = function() {
         | 
| 144 | 
            -
                    document.addEventListener('click', (e) => {
         | 
| 145 | 
            -
                      if (e.target.matches('[data-checkboxsubmit-target="checkbox"]')) {
         | 
| 146 | 
            -
                        const form = e.target.closest('form');
         | 
| 147 | 
            -
                        if (form) new CheckboxSubmit(form).clicked(e);
         | 
| 148 | 
            -
                      }
         | 
| 149 | 
            -
                    });
         | 
| 150 | 
            -
                  };
         | 
| 151 | 
            -
                  Blacklight.doBookmarkToggleBehavior.selector = 'form.bookmark-toggle';
         | 
| 152 | 
            -
             | 
| 153 | 
            -
                  Blacklight.doBookmarkToggleBehavior();
         | 
| 154 | 
            -
              })();
         | 
| 155 | 
            -
             | 
| 156 | 
            -
              const ButtonFocus = (() => {
         | 
| 157 | 
            -
                document.addEventListener('click', (e) => {
         | 
| 158 | 
            -
                  // Button clicks should change focus. As of 10/3/19, Firefox for Mac and
         | 
| 159 | 
            -
                  // Safari both do not set focus to a button on button click.
         | 
| 160 | 
            -
                  // See https://zellwk.com/blog/inconsistent-button-behavior/ for background information
         | 
| 161 | 
            -
                  if (e.target.matches('[data-toggle="collapse"]') || e.target.matches('[data-bs-toggle="collapse"]')) {
         | 
| 162 | 
            -
                    e.target.focus();
         | 
| 163 | 
            -
                  }
         | 
| 164 | 
            -
                });
         | 
| 165 | 
            -
              })();
         | 
| 166 | 
            -
             | 
| 167 | 
            -
              /*
         | 
| 168 | 
            -
                The blacklight modal plugin can display some interactions inside a Bootstrap
         | 
| 169 | 
            -
                modal window, including some multi-page interactions.
         | 
| 170 | 
            -
             | 
| 171 | 
            -
                It supports unobtrusive Javascript, where a link or form that would have caused
         | 
| 172 | 
            -
                a new page load is changed to display it's results inside a modal dialog,
         | 
| 173 | 
            -
                by this plugin.  The plugin assumes there is a Bootstrap modal div
         | 
| 174 | 
            -
                on the page with id #blacklight-modal to use as the modal -- the standard Blacklight
         | 
| 175 | 
            -
                layout provides this.
         | 
| 176 | 
            -
             | 
| 177 | 
            -
                To make a link or form have their results display inside a modal, add
         | 
| 178 | 
            -
                `data-blacklight-modal="trigger"` to the link or form. (Note, form itself not submit input)
         | 
| 179 | 
            -
                With Rails link_to helper, you'd do that like:
         | 
| 180 | 
            -
             | 
| 181 | 
            -
                    link_to something, link, data: { blacklight_modal: "trigger" }
         | 
| 182 | 
            -
             | 
| 183 | 
            -
                The results of the link href or form submit will be displayed inside
         | 
| 184 | 
            -
                a modal -- they should include the proper HTML markup for a bootstrap modal's
         | 
| 185 | 
            -
                contents. Also, you ordinarily won't want the Rails template with wrapping
         | 
| 186 | 
            -
                navigational elements to be used.  The Rails controller could suppress
         | 
| 187 | 
            -
                the layout when a JS AJAX request is detected, OR the response
         | 
| 188 | 
            -
                can include a `<div data-blacklight-modal="container">` -- only the contents
         | 
| 189 | 
            -
                of the container will be placed inside the modal, the rest of the
         | 
| 190 | 
            -
                page will be ignored.
         | 
| 191 | 
            -
             | 
| 192 | 
            -
                Link or forms inside the modal will ordinarily cause page loads
         | 
| 193 | 
            -
                when they are triggered. However, if you'd like their results
         | 
| 194 | 
            -
                to stay within the modal, just add `data-blacklight-modal="preserve"`
         | 
| 195 | 
            -
                to the link or form.
         | 
| 196 | 
            -
             | 
| 197 | 
            -
                Here's an example of what might be returned, demonstrating most of the devices available:
         | 
| 198 | 
            -
             | 
| 199 | 
            -
                  <div data-blacklight-modal="container">
         | 
| 200 | 
            -
                    <div class="modal-header">
         | 
| 201 | 
            -
                      <button type="button" class="close" data-bl-dismiss="modal" aria-hidden="true">×</button>
         | 
| 202 | 
            -
                      <h3 class="modal-title">Request Placed</h3>
         | 
| 203 | 
            -
                    </div>
         | 
| 204 | 
            -
             | 
| 205 | 
            -
                    <div class="modal-body">
         | 
| 206 | 
            -
                      <p>Some message</p>
         | 
| 207 | 
            -
                      <%= link_to "This result will still be within modal", some_link, data: { blacklight_modal: "preserve" } %>
         | 
| 208 | 
            -
                    </div>
         | 
| 209 | 
            -
             | 
| 210 | 
            -
             | 
| 211 | 
            -
                    <div class="modal-footer">
         | 
| 212 | 
            -
                      <button type="button" class="btn btn-secondary" data-bl-dismiss="modal">Close</button>
         | 
| 213 | 
            -
                    </div>
         | 
| 214 | 
            -
                  </div>
         | 
| 215 | 
            -
             | 
| 216 | 
            -
             | 
| 217 | 
            -
                One additional feature. If the content returned from the AJAX form submission
         | 
| 218 | 
            -
                can be a turbo-stream that defines some HTML fragementsand where on the page to put them:
         | 
| 219 | 
            -
                https://turbo.hotwired.dev/handbook/streams
         | 
| 220 | 
            -
              */
         | 
| 221 | 
            -
             | 
| 222 | 
            -
              const Modal = (() => {
         | 
| 223 | 
            -
                // We keep all our data in Blacklight.modal object.
         | 
| 224 | 
            -
                // Create lazily if someone else created first.
         | 
| 225 | 
            -
                if (Blacklight.modal === undefined) {
         | 
| 226 | 
            -
                  Blacklight.modal = {};
         | 
| 227 | 
            -
                }
         | 
| 228 | 
            -
             | 
| 229 | 
            -
                const modal = Blacklight.modal;
         | 
| 230 | 
            -
             | 
| 231 | 
            -
                // a Bootstrap modal div that should be already on the page hidden
         | 
| 232 | 
            -
                modal.modalSelector = '#blacklight-modal';
         | 
| 233 | 
            -
             | 
| 234 | 
            -
                // Trigger selectors identify forms or hyperlinks that should open
         | 
| 235 | 
            -
                // inside a modal dialog.
         | 
| 236 | 
            -
                modal.triggerLinkSelector  = 'a[data-blacklight-modal~=trigger]';
         | 
| 237 | 
            -
             | 
| 238 | 
            -
                // preserve selectors identify forms or hyperlinks that, if activated already
         | 
| 239 | 
            -
                // inside a modal dialog, should have destinations remain inside the modal -- but
         | 
| 240 | 
            -
                // won't trigger a modal if not already in one.
         | 
| 241 | 
            -
                //
         | 
| 242 | 
            -
                // No need to repeat selectors from trigger selectors, those will already
         | 
| 243 | 
            -
                // be preserved. MUST be manually prefixed with the modal selector,
         | 
| 244 | 
            -
                // so they only apply to things inside a modal.
         | 
| 245 | 
            -
                modal.preserveLinkSelector = modal.modalSelector + ' a[data-blacklight-modal~=preserve]';
         | 
| 246 | 
            -
             | 
| 247 | 
            -
                modal.containerSelector    = '[data-blacklight-modal~=container]';
         | 
| 248 | 
            -
             | 
| 249 | 
            -
                // Called on fatal failure of ajax load, function returns content
         | 
| 250 | 
            -
                // to show to user in modal.  Right now called only for network errors.
         | 
| 251 | 
            -
                modal.onFailure = function (error) {
         | 
| 252 | 
            -
                    console.error('Server error:', this.url, error);
         | 
| 253 | 
            -
             | 
| 254 | 
            -
                    const contents = `<div class="modal-header">
         | 
| 255 | 
            -
                    <div class="modal-title">There was a problem with your request.</div>
         | 
| 256 | 
            -
                    <button type="button" class="blacklight-modal-close btn-close close" data-bl-dismiss="modal" aria-label="Close">
         | 
| 257 | 
            -
                      <span aria-hidden="true" class="visually-hidden">×</span>
         | 
| 258 | 
            -
                    </button>
         | 
| 259 | 
            -
                    </div>
         | 
| 260 | 
            -
                    <div class="modal-body">
         | 
| 261 | 
            -
                      <p>Expected a successful response from the server, but got an error</p>
         | 
| 262 | 
            -
                      <pre>${this.url}\n${error}</pre>
         | 
| 263 | 
            -
                    </div>`;
         | 
| 264 | 
            -
             | 
| 265 | 
            -
                    document.querySelector(`${modal.modalSelector} .modal-content`).innerHTML = contents;
         | 
| 266 | 
            -
             | 
| 267 | 
            -
                    modal.show();
         | 
| 268 | 
            -
                };
         | 
| 269 | 
            -
             | 
| 270 | 
            -
                // Add the passed in contents to the modal and display it.
         | 
| 271 | 
            -
                modal.receiveAjax = function (contents) {
         | 
| 272 | 
            -
                    const domparser = new DOMParser();
         | 
| 273 | 
            -
                    const dom = domparser.parseFromString(contents, "text/html");
         | 
| 274 | 
            -
                    // If there is a containerSelector on the document, use its children.
         | 
| 275 | 
            -
                    let elements = dom.querySelectorAll(`${modal.containerSelector} > *`);
         | 
| 276 | 
            -
                    if (elements.length == 0) {
         | 
| 277 | 
            -
                      // If the containerSelector wasn't found, use the whole document
         | 
| 278 | 
            -
                      elements = dom.body.childNodes;
         | 
| 279 | 
            -
                    }
         | 
| 280 | 
            -
             | 
| 281 | 
            -
                    document.querySelector(`${modal.modalSelector} .modal-content`).replaceChildren(...elements);
         | 
| 282 | 
            -
             | 
| 283 | 
            -
                    modal.show();
         | 
| 284 | 
            -
                };
         | 
| 285 | 
            -
             | 
| 286 | 
            -
             | 
| 287 | 
            -
                modal.modalAjaxLinkClick = function(e) {
         | 
| 288 | 
            -
                  e.preventDefault();
         | 
| 289 | 
            -
                  const href = e.target.getAttribute('href');
         | 
| 290 | 
            -
                  fetch(href)
         | 
| 291 | 
            -
                    .then(response => {
         | 
| 292 | 
            -
                       if (!response.ok) {
         | 
| 293 | 
            -
                         throw new TypeError("Request failed");
         | 
| 294 | 
            -
                       }
         | 
| 295 | 
            -
                       return response.text();
         | 
| 296 | 
            -
                     })
         | 
| 297 | 
            -
                    .then(data => modal.receiveAjax(data))
         | 
| 298 | 
            -
                    .catch(error => modal.onFailure(error));
         | 
| 299 | 
            -
                };
         | 
| 300 | 
            -
             | 
| 301 | 
            -
                modal.setupModal = function() {
         | 
| 302 | 
            -
                  // Register both trigger and preserve selectors in ONE event handler, combining
         | 
| 303 | 
            -
                  // into one selector with a comma, so if something matches BOTH selectors, it
         | 
| 304 | 
            -
                  // still only gets the event handler called once.
         | 
| 305 | 
            -
                  document.addEventListener('click', (e) => {
         | 
| 306 | 
            -
                    if (e.target.closest(`${modal.triggerLinkSelector}, ${modal.preserveLinkSelector}`))
         | 
| 307 | 
            -
                      modal.modalAjaxLinkClick(e);
         | 
| 308 | 
            -
                    else if (e.target.closest('[data-bl-dismiss="modal"]'))
         | 
| 309 | 
            -
                      modal.hide();
         | 
| 310 | 
            -
                  });
         | 
| 311 | 
            -
                };
         | 
| 312 | 
            -
             | 
| 313 | 
            -
                modal.hide = function (el) {
         | 
| 314 | 
            -
                  const dom = document.querySelector(Blacklight.modal.modalSelector);
         | 
| 315 | 
            -
             | 
| 316 | 
            -
                  if (!dom.open) return
         | 
| 317 | 
            -
                  dom.close();
         | 
| 318 | 
            -
                };
         | 
| 319 | 
            -
             | 
| 320 | 
            -
                modal.show = function(el) {
         | 
| 321 | 
            -
                  const dom = document.querySelector(Blacklight.modal.modalSelector);
         | 
| 322 | 
            -
             | 
| 323 | 
            -
                  if (dom.open) return
         | 
| 324 | 
            -
                  dom.showModal();
         | 
| 325 | 
            -
                };
         | 
| 326 | 
            -
             | 
| 327 | 
            -
                modal.setupModal();
         | 
| 328 | 
            -
              })();
         | 
| 329 | 
            -
             | 
| 330 | 
            -
              const SearchContext = (() => {
         | 
| 331 | 
            -
                Blacklight.doSearchContextBehavior = function() {
         | 
| 332 | 
            -
                  document.addEventListener('click', (e) => {
         | 
| 333 | 
            -
                    if (e.target.matches('[data-context-href]')) {
         | 
| 334 | 
            -
                      Blacklight.handleSearchContextMethod.call(e.target, e);
         | 
| 335 | 
            -
                    }
         | 
| 336 | 
            -
                  });
         | 
| 337 | 
            -
                };
         | 
| 338 | 
            -
             | 
| 339 | 
            -
                Blacklight.csrfToken = () => document.querySelector('meta[name=csrf-token]')?.content;
         | 
| 340 | 
            -
                Blacklight.csrfParam = () => document.querySelector('meta[name=csrf-param]')?.content;
         | 
| 341 | 
            -
             | 
| 342 | 
            -
                // this is the Rails.handleMethod with a couple adjustments, described inline:
         | 
| 343 | 
            -
                // first, we're attaching this directly to the event handler, so we can check for meta-keys
         | 
| 344 | 
            -
                Blacklight.handleSearchContextMethod = function(event) {
         | 
| 345 | 
            -
                  const link = this;
         | 
| 346 | 
            -
             | 
| 347 | 
            -
                  // instead of using the normal href, we need to use the context href instead
         | 
| 348 | 
            -
                  let href = link.getAttribute('data-context-href');
         | 
| 349 | 
            -
                  let target = link.getAttribute('target');
         | 
| 350 | 
            -
                  let csrfToken = Blacklight.csrfToken();
         | 
| 351 | 
            -
                  let csrfParam = Blacklight.csrfParam();
         | 
| 352 | 
            -
                  let form = document.createElement('form');
         | 
| 353 | 
            -
                  form.method = 'post';
         | 
| 354 | 
            -
                  form.action = href;
         | 
| 355 | 
            -
             | 
| 356 | 
            -
             | 
| 357 | 
            -
                  let formContent = `<input name="_method" value="post" type="hidden" />
         | 
| 358 | 
            -
                  <input name="redirect" value="${link.getAttribute('href')}" type="hidden" />`;
         | 
| 359 | 
            -
             | 
| 360 | 
            -
                  // check for meta keys.. if set, we should open in a new tab
         | 
| 361 | 
            -
                  if(event.metaKey || event.ctrlKey) {
         | 
| 362 | 
            -
                    target = '_blank';
         | 
| 363 | 
            -
                  }
         | 
| 364 | 
            -
             | 
| 365 | 
            -
                  if (csrfParam !== undefined && csrfToken !== undefined) {
         | 
| 366 | 
            -
                    formContent += `<input name="${csrfParam}" value="${csrfToken}" type="hidden" />`;
         | 
| 367 | 
            -
                  }
         | 
| 368 | 
            -
             | 
| 369 | 
            -
                  // Must trigger submit by click on a button, else "submit" event handler won't work!
         | 
| 370 | 
            -
                  // https://developer.mozilla.org/en-US/docs/Web/API/HTMLFormElement/submit
         | 
| 371 | 
            -
                  formContent += '<input type="submit" />';
         | 
| 372 | 
            -
             | 
| 373 | 
            -
                  if (target) { form.setAttribute('target', target); }
         | 
| 374 | 
            -
             | 
| 375 | 
            -
                  form.style.display = 'none';
         | 
| 376 | 
            -
                  form.innerHTML = formContent;
         | 
| 377 | 
            -
                  document.body.appendChild(form);
         | 
| 378 | 
            -
                  form.querySelector('[type="submit"]').click();
         | 
| 379 | 
            -
             | 
| 380 | 
            -
                  event.preventDefault();
         | 
| 381 | 
            -
                };
         | 
| 382 | 
            -
             | 
| 383 | 
            -
                Blacklight.doSearchContextBehavior();
         | 
| 384 | 
            -
              })();
         | 
| 385 | 
            -
             | 
| 386 | 
            -
              const index = {
         | 
| 387 | 
            -
                BookmarkToggle,
         | 
| 388 | 
            -
                ButtonFocus,
         | 
| 389 | 
            -
                Modal,
         | 
| 390 | 
            -
                SearchContext,
         | 
| 391 | 
            -
                Core: Blacklight,
         | 
| 392 | 
            -
                onLoad: Blacklight.onLoad
         | 
| 393 | 
            -
              };
         | 
| 394 | 
            -
             | 
| 395 | 
            -
              return index;
         | 
| 396 | 
            -
             | 
| 397 | 
            -
            }));
         | 
| 398 | 
            -
            //# sourceMappingURL=blacklight.js.map
         | 
| @@ -1 +0,0 @@ | |
| 1 | 
            -
            {"version":3,"file":"blacklight.js","sources":["../../../javascript/blacklight/core.js","../../../javascript/blacklight/checkbox_submit.js","../../../javascript/blacklight/bookmark_toggle.js","../../../javascript/blacklight/button_focus.js","../../../javascript/blacklight/modal.js","../../../javascript/blacklight/search_context.js","../../../javascript/blacklight/index.js"],"sourcesContent":["const Blacklight = function() {\n  const buffer = new Array;\n  return {\n    onLoad: function(func) {\n      buffer.push(func);\n    },\n\n    activate: function() {\n      for(let i = 0; i < buffer.length; i++) {\n        buffer[i].call();\n      }\n    },\n\n    listeners: function () {\n      const listeners = [];\n      if (typeof Turbo !== 'undefined') {\n        listeners.push('turbo:load', 'turbo:frame-load');\n      } else if (typeof Turbolinks !== 'undefined' && Turbolinks.supported) {\n        // Turbolinks 5\n        if (Turbolinks.BrowserAdapter) {\n          listeners.push('turbolinks:load');\n        } else {\n          // Turbolinks < 5\n          listeners.push('page:load', 'DOMContentLoaded');\n        }\n      } else {\n        listeners.push('DOMContentLoaded');\n      }\n\n      return listeners;\n    }\n  };\n}();\n\n// turbolinks triggers page:load events on page transition\n// If app isn't using turbolinks, this event will never be triggered, no prob.\nBlacklight.listeners().forEach(function(listener) {\n  document.addEventListener(listener, function() {\n    Blacklight.activate()\n  })\n})\n\nBlacklight.onLoad(function () {\n  const elem = document.querySelector('.no-js');\n\n  // The \"no-js\" class may already have been removed because this function is\n  // run on every turbo:load event, in that case, it won't find an element.\n  if (!elem) return;\n\n  elem.classList.remove('no-js')\n  elem.classList.add('js')\n})\n\n\nexport default Blacklight\n","/* Converts a \"toggle\" form, with single submit button to add/remove\n   something, like used for Bookmarks, into an AJAXy checkbox instead.\n   Apply to a form. Does require certain assumption about the form:\n    1) The same form 'action' href must be used for both ADD and REMOVE\n       actions, with the different being the hidden input name=\"_method\"\n       being set to \"put\" or \"delete\" -- that's the Rails method to pretend\n       to be doing a certain HTTP verb. So same URL, PUT to add, DELETE\n       to remove. This plugin assumes that.\n       Plus, the form this is applied to should provide a data-doc-id\n       attribute (HTML5-style doc-*) that contains the id/primary key\n       of the object in question -- used by plugin for a unique value for\n       DOM id's.\n  Uses HTML for a checkbox compatible with Bootstrap 4.\n   new CheckboxSubmit(document.querySelector('form.something')).render()\n*/\nexport default class CheckboxSubmit {\n  constructor(form) {\n    this.form = form\n  }\n\n  async clicked(evt) {\n    this.spanTarget.innerHTML = this.form.getAttribute('data-inprogress')\n    this.labelTarget.setAttribute('disabled', 'disabled');\n    this.checkboxTarget.setAttribute('disabled', 'disabled');\n    const response = await fetch(this.formTarget.getAttribute('action'), {\n      body: new FormData(this.formTarget),\n      method: this.formTarget.getAttribute('method').toUpperCase(),\n      headers: {\n        'Accept': 'application/json',\n        'X-Requested-With': 'XMLHttpRequest',\n        'X-CSRF-Token': document.querySelector('meta[name=csrf-token]')?.content\n      }\n    })\n    this.labelTarget.removeAttribute('disabled')\n    this.checkboxTarget.removeAttribute('disabled')\n    if (response.ok) {\n      const json = await response.json()\n      this.updateStateFor(!this.checked)\n      document.querySelector('[data-role=bookmark-counter]').innerHTML = json.bookmarks.count\n    } else {\n      alert('Error')\n    }\n  }\n\n  get checked() {\n    return (this.form.querySelectorAll('input[name=_method][value=delete]').length != 0)\n  }\n\n  get formTarget() {\n    return this.form\n  }\n\n  get labelTarget() {\n    return this.form.querySelector('[data-checkboxsubmit-target=\"label\"]')\n  }\n\n  get checkboxTarget() {\n    return this.form.querySelector('[data-checkboxsubmit-target=\"checkbox\"]')\n  }\n\n  get spanTarget() {\n    return this.form.querySelector('[data-checkboxsubmit-target=\"span\"]')\n  }\n\n  updateStateFor(state) {\n    this.checkboxTarget.checked = state\n\n    if (state) {\n      this.labelTarget.classList.add('checked')\n      //Set the Rails hidden field that fakes an HTTP verb\n      //properly for current state action.\n      this.formTarget.querySelector('input[name=_method]').value = 'delete'\n      this.spanTarget.innerHTML = this.form.getAttribute('data-present')\n    } else {\n      this.labelTarget.classList.remove('checked')\n      this.formTarget.querySelector('input[name=_method]').value = 'put'\n      this.spanTarget.innerHTML = this.form.getAttribute('data-absent')\n    }\n  }\n}\n","import Blacklight from './core'\nimport CheckboxSubmit from './checkbox_submit'\n\nconst BookmarkToggle = (() => {\n    // change form submit toggle to checkbox\n    Blacklight.doBookmarkToggleBehavior = function() {\n      document.addEventListener('click', (e) => {\n        if (e.target.matches('[data-checkboxsubmit-target=\"checkbox\"]')) {\n          const form = e.target.closest('form')\n          if (form) new CheckboxSubmit(form).clicked(e);\n        }\n      });\n    };\n    Blacklight.doBookmarkToggleBehavior.selector = 'form.bookmark-toggle';\n\n    Blacklight.doBookmarkToggleBehavior();\n})()\n\nexport default BookmarkToggle\n","const ButtonFocus = (() => {\n  document.addEventListener('click', (e) => {\n    // Button clicks should change focus. As of 10/3/19, Firefox for Mac and\n    // Safari both do not set focus to a button on button click.\n    // See https://zellwk.com/blog/inconsistent-button-behavior/ for background information\n    if (e.target.matches('[data-toggle=\"collapse\"]') || e.target.matches('[data-bs-toggle=\"collapse\"]')) {\n      e.target.focus()\n    }\n  })\n})()\n\nexport default ButtonFocus\n","/*\n  The blacklight modal plugin can display some interactions inside a Bootstrap\n  modal window, including some multi-page interactions.\n\n  It supports unobtrusive Javascript, where a link or form that would have caused\n  a new page load is changed to display it's results inside a modal dialog,\n  by this plugin.  The plugin assumes there is a Bootstrap modal div\n  on the page with id #blacklight-modal to use as the modal -- the standard Blacklight\n  layout provides this.\n\n  To make a link or form have their results display inside a modal, add\n  `data-blacklight-modal=\"trigger\"` to the link or form. (Note, form itself not submit input)\n  With Rails link_to helper, you'd do that like:\n\n      link_to something, link, data: { blacklight_modal: \"trigger\" }\n\n  The results of the link href or form submit will be displayed inside\n  a modal -- they should include the proper HTML markup for a bootstrap modal's\n  contents. Also, you ordinarily won't want the Rails template with wrapping\n  navigational elements to be used.  The Rails controller could suppress\n  the layout when a JS AJAX request is detected, OR the response\n  can include a `<div data-blacklight-modal=\"container\">` -- only the contents\n  of the container will be placed inside the modal, the rest of the\n  page will be ignored.\n\n  Link or forms inside the modal will ordinarily cause page loads\n  when they are triggered. However, if you'd like their results\n  to stay within the modal, just add `data-blacklight-modal=\"preserve\"`\n  to the link or form.\n\n  Here's an example of what might be returned, demonstrating most of the devices available:\n\n    <div data-blacklight-modal=\"container\">\n      <div class=\"modal-header\">\n        <button type=\"button\" class=\"close\" data-bl-dismiss=\"modal\" aria-hidden=\"true\">×</button>\n        <h3 class=\"modal-title\">Request Placed</h3>\n      </div>\n\n      <div class=\"modal-body\">\n        <p>Some message</p>\n        <%= link_to \"This result will still be within modal\", some_link, data: { blacklight_modal: \"preserve\" } %>\n      </div>\n\n\n      <div class=\"modal-footer\">\n        <button type=\"button\" class=\"btn btn-secondary\" data-bl-dismiss=\"modal\">Close</button>\n      </div>\n    </div>\n\n\n  One additional feature. If the content returned from the AJAX form submission\n  can be a turbo-stream that defines some HTML fragementsand where on the page to put them:\n  https://turbo.hotwired.dev/handbook/streams\n*/\nimport Blacklight from './core'\nimport ModalForm from './modalForm'\n\nconst Modal = (() => {\n  // We keep all our data in Blacklight.modal object.\n  // Create lazily if someone else created first.\n  if (Blacklight.modal === undefined) {\n    Blacklight.modal = {};\n  }\n\n  const modal = Blacklight.modal\n\n  // a Bootstrap modal div that should be already on the page hidden\n  modal.modalSelector = '#blacklight-modal';\n\n  // Trigger selectors identify forms or hyperlinks that should open\n  // inside a modal dialog.\n  modal.triggerLinkSelector  = 'a[data-blacklight-modal~=trigger]';\n\n  // preserve selectors identify forms or hyperlinks that, if activated already\n  // inside a modal dialog, should have destinations remain inside the modal -- but\n  // won't trigger a modal if not already in one.\n  //\n  // No need to repeat selectors from trigger selectors, those will already\n  // be preserved. MUST be manually prefixed with the modal selector,\n  // so they only apply to things inside a modal.\n  modal.preserveLinkSelector = modal.modalSelector + ' a[data-blacklight-modal~=preserve]';\n\n  modal.containerSelector    = '[data-blacklight-modal~=container]';\n\n  // Called on fatal failure of ajax load, function returns content\n  // to show to user in modal.  Right now called only for network errors.\n  modal.onFailure = function (error) {\n      console.error('Server error:', this.url, error);\n\n      const contents = `<div class=\"modal-header\">\n        <div class=\"modal-title\">There was a problem with your request.</div>\n        <button type=\"button\" class=\"blacklight-modal-close btn-close close\" data-bl-dismiss=\"modal\" aria-label=\"Close\">\n          <span aria-hidden=\"true\" class=\"visually-hidden\">×</span>\n        </button>\n        </div>\n        <div class=\"modal-body\">\n          <p>Expected a successful response from the server, but got an error</p>\n          <pre>${this.url}\\n${error}</pre>\n        </div>`\n\n      document.querySelector(`${modal.modalSelector} .modal-content`).innerHTML = contents\n\n      modal.show();\n  }\n\n  // Add the passed in contents to the modal and display it.\n  modal.receiveAjax = function (contents) {\n      const domparser = new DOMParser();\n      const dom = domparser.parseFromString(contents, \"text/html\")\n      // If there is a containerSelector on the document, use its children.\n      let elements = dom.querySelectorAll(`${modal.containerSelector} > *`)\n      if (elements.length == 0) {\n        // If the containerSelector wasn't found, use the whole document\n        elements = dom.body.childNodes\n      }\n\n      document.querySelector(`${modal.modalSelector} .modal-content`).replaceChildren(...elements)\n\n      modal.show();\n  };\n\n\n  modal.modalAjaxLinkClick = function(e) {\n    e.preventDefault();\n    const href = e.target.getAttribute('href')\n    fetch(href)\n      .then(response => {\n         if (!response.ok) {\n           throw new TypeError(\"Request failed\");\n         }\n         return response.text();\n       })\n      .then(data => modal.receiveAjax(data))\n      .catch(error => modal.onFailure(error))\n  };\n\n  modal.setupModal = function() {\n    // Register both trigger and preserve selectors in ONE event handler, combining\n    // into one selector with a comma, so if something matches BOTH selectors, it\n    // still only gets the event handler called once.\n    document.addEventListener('click', (e) => {\n      if (e.target.closest(`${modal.triggerLinkSelector}, ${modal.preserveLinkSelector}`))\n        modal.modalAjaxLinkClick(e)\n      else if (e.target.closest('[data-bl-dismiss=\"modal\"]'))\n        modal.hide()\n    })\n  };\n\n  modal.hide = function (el) {\n    const dom = document.querySelector(Blacklight.modal.modalSelector)\n\n    if (!dom.open) return\n    dom.close()\n  }\n\n  modal.show = function(el) {\n    const dom = document.querySelector(Blacklight.modal.modalSelector)\n\n    if (dom.open) return\n    dom.showModal()\n  }\n\n  modal.setupModal()\n})()\n\nexport default Modal\n","import Blacklight from './core'\n\nconst SearchContext = (() => {\n  Blacklight.doSearchContextBehavior = function() {\n    document.addEventListener('click', (e) => {\n      if (e.target.matches('[data-context-href]')) {\n        Blacklight.handleSearchContextMethod.call(e.target, e)\n      }\n    })\n  };\n\n  Blacklight.csrfToken = () => document.querySelector('meta[name=csrf-token]')?.content\n  Blacklight.csrfParam = () => document.querySelector('meta[name=csrf-param]')?.content\n\n  // this is the Rails.handleMethod with a couple adjustments, described inline:\n  // first, we're attaching this directly to the event handler, so we can check for meta-keys\n  Blacklight.handleSearchContextMethod = function(event) {\n    const link = this\n\n    // instead of using the normal href, we need to use the context href instead\n    let href = link.getAttribute('data-context-href')\n    let target = link.getAttribute('target')\n    let csrfToken = Blacklight.csrfToken()\n    let csrfParam = Blacklight.csrfParam()\n    let form = document.createElement('form')\n    form.method = 'post'\n    form.action = href\n\n\n    let formContent = `<input name=\"_method\" value=\"post\" type=\"hidden\" />\n      <input name=\"redirect\" value=\"${link.getAttribute('href')}\" type=\"hidden\" />`\n\n    // check for meta keys.. if set, we should open in a new tab\n    if(event.metaKey || event.ctrlKey) {\n      target = '_blank';\n    }\n\n    if (csrfParam !== undefined && csrfToken !== undefined) {\n      formContent += `<input name=\"${csrfParam}\" value=\"${csrfToken}\" type=\"hidden\" />`\n    }\n\n    // Must trigger submit by click on a button, else \"submit\" event handler won't work!\n    // https://developer.mozilla.org/en-US/docs/Web/API/HTMLFormElement/submit\n    formContent += '<input type=\"submit\" />'\n\n    if (target) { form.setAttribute('target', target); }\n\n    form.style.display = 'none'\n    form.innerHTML = formContent\n    document.body.appendChild(form)\n    form.querySelector('[type=\"submit\"]').click()\n\n    event.preventDefault()\n  };\n\n  Blacklight.doSearchContextBehavior();\n})()\n\nexport default SearchContext\n","import BookmarkToggle from './bookmark_toggle'\nimport ButtonFocus from './button_focus'\nimport Modal from './modal'\nimport SearchContext from './search_context'\nimport Core from './core'\n\nexport default {\n  BookmarkToggle,\n  ButtonFocus,\n  Modal,\n  SearchContext,\n  Core,\n  onLoad: Core.onLoad\n}\n"],"names":["Core"],"mappings":"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"}
         | 
| @@ -1,13 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            describe 'Sitelinks search box' do
         | 
| 4 | 
            -
              it 'is home page' do
         | 
| 5 | 
            -
                visit root_path
         | 
| 6 | 
            -
                expect(page).to have_css 'script[type="application/ld+json"]'
         | 
| 7 | 
            -
              end
         | 
| 8 | 
            -
             | 
| 9 | 
            -
              it 'on search page' do
         | 
| 10 | 
            -
                visit search_catalog_path q: 'book'
         | 
| 11 | 
            -
                expect(page).not_to have_css 'script[type="application/ld+json"]'
         | 
| 12 | 
            -
              end
         | 
| 13 | 
            -
            end
         | 
| @@ -1,8 +0,0 @@ | |
| 1 | 
            -
            # frozen_string_literal: true
         | 
| 2 | 
            -
             | 
| 3 | 
            -
            module ViewComponentCapybaraTestHelpers
         | 
| 4 | 
            -
              # Work around for https://github.com/teamcapybara/capybara/issues/2466
         | 
| 5 | 
            -
              def render_inline_to_capybara_node(component)
         | 
| 6 | 
            -
                Capybara::Node::Simple.new(render_inline(component).to_s)
         | 
| 7 | 
            -
              end
         | 
| 8 | 
            -
            end
         |