diff --git a/fec/fec/static/js/legal-search-ao.js b/fec/fec/static/js/legal-search-ao.js index 9e0500b409..59b8a1b008 100644 --- a/fec/fec/static/js/legal-search-ao.js +++ b/fec/fec/static/js/legal-search-ao.js @@ -279,6 +279,9 @@ LegalSearchAo.prototype.getResults = function(e) { // Set the sort param value according to this.sortOrder serializedFilters.sort = this.sortOrder == 'asc' ? 'ao_no' : '-ao_no'; + // If we're getting new results, let's reset the page offset (go back to page 1) + serializedFilters['offset'] = 0; + // Then update the URL with currently params updateQuery(serializedFilters, filterFields); @@ -295,7 +298,23 @@ LegalSearchAo.prototype.getResults = function(e) { delete fetchParams[param]; if (param == 'search') { - fetchParams['q'] = fetchParams['search']; + // We need to divide any 'search' value into q and q_exclude, split on ` -` + const qStrings = []; + const qExcludeStrings = []; + if (fetchParams['search'].indexOf(' -') >= 0) { + const allTerms = fetchParams['search'].split(' '); + allTerms.forEach(term => { + if (term.startsWith('-')) + qExcludeStrings.push(term.substring(1)); + else qStrings.push(term); + }); + if (qStrings.length > 0) + fetchParams['q'] = qStrings.join(' '); + if (qExcludeStrings.length > 0) + fetchParams['q_exclude'] = qExcludeStrings.join(' '); + } else + fetchParams['q'] = fetchParams['search']; + delete fetchParams['search']; } } @@ -604,7 +623,7 @@ LegalSearchAo.prototype.updatePagination = function(resultsCount) { if (totalNumberOfPages <= maxButtonsOnScreen) { // Yay! We can just use every button for (let i = 0; i < totalNumberOfPages; i++) { - pageNumbers.push(i + 1); + pageNumbers.push(i); } } else { const buttonsBeforeCurrent = Math.floor(maxButtonsOnScreen / 2); diff --git a/fec/fec/static/js/pages/election-reporting-dates-tables.js b/fec/fec/static/js/pages/election-reporting-dates-tables.js deleted file mode 100644 index 5559239165..0000000000 --- a/fec/fec/static/js/pages/election-reporting-dates-tables.js +++ /dev/null @@ -1,564 +0,0 @@ -const states_dropdown_template = ` -
- -
`; - -const header_notes_modal_partial = ` - -`; - -function ReportingDates() { - - this.dates_table = document.getElementsByClassName('election-dates-table')[0]; - - this.buildStaticElements(); // build header_notes dialog and states dropdown - - this.convertFootnotes(); //converts number or symbol following "~" to footnote html, in-place - - this.addStateClass(); // adds state abbr classes to rows - - this.addFootnotes(); //adds hidden footnote rows - - this.stripeByState(); //zebra strip by state - - //it only runs this logic if the page has an `.election-dates-table` on it, TODO: might not e necessary for this trmplate - if (this.dates_table) { - //get all acnhor links in TDs) - this.anchors = this.dates_table.querySelectorAll('td a[href^=\'#\']'); - - //disable default jump behavior for anchor links(#) but keep links for accessibility - for (const anchor of this.anchors) { - anchor.addEventListener('click', e => { - e.preventDefault(); - }); - } - - const header_sups = document.querySelectorAll( - 'tr:first-child th a[href^="#"]' - ); - - //add data attribute to the header sups to open AY11 dialog and disable default jump behavior for anchor links(#) but keep links for accessibility - for (const header_sup of header_sups) { - header_sup.setAttribute('data-a11y-dialog-show', 'header_notes_modal'); - header_sup.addEventListener('click', e => { - e.preventDefault(); - }); - } - - //Define media query - const mql = window.matchMedia('screen and (max-width: 650px)'); - - //call listener function explicitly at run time - this.mediaQueryResponse(mql); - - //attach listener function to listen in on state changes - mql.addListener(this.mediaQueryResponse); // TODO: .addListener() has been deprecated - - //show footnotes on click of a link that wraps the superscripts in cells - for (const anchor of this.anchors) { - anchor.addEventListener('click', this.showFootnotes.bind(this)); - } - - //handle changes on states dropdown to filter by state - this.states = document.getElementById('states'); - this.states.addEventListener('change', this.handleStateChange.bind(this)); - - //Define nexUntil() function for use in ShowFootnotes() - this.nextUntil = function(elem, selector, filter) { - // Setup siblings array - const siblings = []; - - // Get the next sibling element - elem = elem.nextElementSibling; - - // As long as a sibling exists - while (elem) { - // If we've reached our match, bail - if (elem.matches(selector)) break; - - // If filtering by a selector, check if the sibling matches - if (filter && !elem.matches(filter)) { - elem = elem.nextElementSibling; - continue; - } - - // Otherwise, push it to the siblings array - siblings.push(elem); - - // Get the next sibling element - elem = elem.nextElementSibling; - } - - return siblings; - }; - } - -} - -//create and insert states-dropdown, static header-notes-list, and dialog -ReportingDates.prototype.buildStaticElements = function() { - //Add states dropdown template to page - const dropdown_wrapper = document.createElement('div'); - dropdown_wrapper.innerHTML = states_dropdown_template; - - const table_parent = this.dates_table.parentNode; - - table_parent.insertBefore(dropdown_wrapper, this.dates_table); - - //Create header note list for modal dialogue - let hdr_str = ''; - - //Get the '#header_notes' script tag created in the template with json_script - const header_notes = document.getElementById('header_notes'); - if (header_notes) { - const header_notes_json = JSON.parse(document.getElementById('header_notes').textContent); - - if (typeof header_notes_json == 'object') { - hdr_str = `

Header notes

`; - } - - if (typeof header_notes_json == 'object') { - //Create A11Y modal dialog for header_notes popup and add innerHTML - const dialog = document.createElement('div'); - //Must add these three classes separately for IE :-( - dialog.classList.add('js-modal'); - dialog.classList.add('modal'); - dialog.classList.add('modal__content'); - dialog.setAttribute('aria-hidden', 'true'); - dialog.id = 'header_notes_modal'; - document.body.appendChild(dialog); - dialog.innerHTML = header_notes_modal_partial; - //Populate dialog with all header notes - const dialog_p = document.querySelector('.modal p'); - dialog_p.innerHTML = `${hdr_str}`; - - } - } -}; - -// Adds state classes to rows -ReportingDates.prototype.addStateClass = function() { - const all_tr = document.querySelectorAll('tr'); - const states_select = document.getElementById('states'); - Array.from(all_tr).forEach(row => { - - let state_election_name_str = row.cells[0].textContent; - // Remove extra spaces, set to lowercase to normalize human input errors - let state_election_name = state_election_name_str.replace(/\s+/g,' ').trim().toLowerCase(); - - // Match state name in the full election name with states select option to get the state abbreviation - Array.from(states_select.options).forEach(opt => { - // Use '^' to match the full state-name at beginning of string, to lowercase - let regex = `^${opt.textContent.toLowerCase()}.*$`; - // Match state-name in full state_election_string - if (state_election_name.match(regex)) { - - row.classList.add(opt.value.toLowerCase()); - } - }); - - }); -}; - -// Show chosen state rows, hide others -ReportingDates.prototype.handleStateChange = function() { - const state = this.states.value.toLowerCase(); - - //TODO: Should this be `this.dates_table.querySelectorAll` ? - const tr = document.querySelectorAll('tr'); - const ftnt = document.querySelectorAll('tr.footnote_row'); - - if (state != 'states') { - const ones = document.querySelectorAll( - `table tr.${state}:not(.footnote_row), tr:first-child` - ); - const not_ones = document.querySelectorAll(`table tr:not(.${state})`); - - for (const not_one of not_ones) { - not_one.style.display = 'none'; - //use class to handle Safari's lack of support for visibiity:visible/collapse - not_one.classList.remove('row_display'); - } - - for (const one of ones) { - one.style.display = 'table-row'; - //use class to handle Safari's lack of support for visibiity:visible/collapse - one.classList.add('row_display'); - } - } else { - for (const t of tr) { - t.style.display = 'table-row'; - t.classList.add('row_display'); - } - - for (const f of ftnt) { - f.style.display = 'none'; - f.classList.remove('row_display'); - } - } -}; - -// Add footnote rows based on existence of superscript number created by convertFootnotes() -ReportingDates.prototype.addFootnotes = function() { - //Get the '#footnotes' script tag created in the template with json_script - const footnotes = document.getElementById('footnotes'); - if (footnotes) { - const footnotes_json = JSON.parse(document.getElementById('footnotes').textContent); - const footnotes_array = footnotes_json.footnote; - - const date_sups = document.querySelectorAll('td sup'); - - if (typeof footnotes_array == 'object') { - Array.from(date_sups) - .reverse() - .forEach(node => { - const indx = node.innerText; //should this be textContent? - //Only put period after numeric footnotes - const dot = /^\d+$/.test(indx) ? '.' : ''; - const state_class = node.closest('tr').className; - const ftnt_colspan = node.closest('tr').cells.length - 1; - let current_text; - - for (let note of footnotes_array){ - if(note.value.footnote_number == indx){ - - current_text = note.value.footnote_text; - - } - } - - const ftnt_row = ` - - - ${indx}${dot} ${current_text} - `; - node.closest('tr').insertAdjacentHTML('afterend', ftnt_row); - }); - } - //hide footnotes rows initially - const footnote_rows = document.querySelectorAll('.footnote_row'); - for (const footnote_row of footnote_rows) { - footnote_row.style.display = 'none'; - } - } -}; -//Prepend header to cells in mobile ONLY, also add/remove them on resize between mobile/desktop -ReportingDates.prototype.mediaQueryResponse = function(mql) { - //get all non-footnote row cells for mobile - const all_tds = document.querySelectorAll('tr:not(.footnote_row) td'); - //get all table header cells - const all_th = document.querySelectorAll('th'); - - //If mobile - if (mql.matches) { - //Iterate over non-foonote-row cells - Array.from(all_tds).forEach(cell => { - //get the header html for each cell index - const th_append = Array.from(all_th)[cell.cellIndex].outerHTML; - - //prepend it to the cell in a span - cell.insertAdjacentHTML( - 'afterbegin', - `${th_append}` - ); - - //get all the appended header anchor links - const appended_anchors = cell.querySelectorAll('.th_append a[href^="#"]'); - //get the modal and modal-close - const jsmodal = document.getElementsByClassName('js-modal'); - const jsmodalClose = document.querySelectorAll('.modal__close'); - - //Re-stablish A11Y data-attributes to the appended header's anchor if it was removed and re-added - for (const anchor of appended_anchors) { - anchor.setAttribute('data-a11y-dialog-show', 'header_notes_modal'); - anchor.addEventListener('click', e => { - jsmodal[0].setAttribute('aria-hidden', 'false'); - e.preventDefault(); - }); - } - - for (const close of jsmodalClose) { - close.addEventListener('click', () => { - jsmodal[0].setAttribute('aria-hidden', 'true'); - close.focus(); // TODO: jQuery deprecation - }); - } - }); - } - //If not mobile - else { - //Iterate over non-foonote-row cells - Array.from(all_tds).forEach(cell => { - //get prepended header --if it exists-- and remove it - cell.getElementsByTagName('span'); - const th_appended = cell.getElementsByTagName('span'); //cell.getElementsByClassname('th_append'); - if (th_appended.length) { - th_appended[0].remove(); - } - }); - } -}; - -//Show chosen footnote or append it in Mobile view -ReportingDates.prototype.showFootnotes = function(e) { - //escape symbols and invalid CSS selectors in indx (ie: '*', '**', etc.) - const indx = CSS.escape(e.target.textContent); - //get the string representaton of indx to use elsewhere - const clean_indx = e.target.textContent; - - const el = e.target; - const notftnt = 'tr:not(.footnote_row)'; - const ftnt = `.footnote_${indx}`; - const ntlvftnt = `tr:not(.footnote_${indx})`; - const current_row = el.closest('tr'); - const el_td = el.closest('td'); - - //get background color of parent row so footnotes can match parent without zebra striping - const style = window.getComputedStyle(current_row, ''); - const bgColor = style.getPropertyValue('background-color'); - //create new no-transparency color based on background color of current row to change(hide) the btm border on selected cell - const newColor = - bgColor == 'rgba(0, 0, 0, 0)' - ? 'rgba(255,255,255,1)' - : 'rgba(248,248,248,1)'; - - //get footnote row corresponding to the one clicked - const live_note = this.nextUntil(current_row, notftnt, ftnt); - //get all other footnote rows - const not_live_note = this.nextUntil(current_row, notftnt, ntlvftnt); - - //Change border under clicked footnote in Desktop view - - const tds = current_row.cells; - - //First set borderBottom on all TDs on parent row, then below we change - //border on only the one clicked - //have to use 'Array.from' here for Safari, not sure why only in this for/of stmt and not others - for (const td of Array.from(tds)) { - td.style.borderBottom = '1px solid #ddd'; - } - - //Show/hide footnote and change border color under cell - if (live_note[0].style.display == 'none') { - //remove bottom border on first TD (state cell) - current_row.cells[0].style.borderBottom = `1px solid ${newColor}`; - //show chosen footnote - live_note[0].style.display = 'table-row'; - //set bg color on footnote rows to match parent rows - live_note[0].style.backgroundColor = bgColor; - //hide bottom border under cell in which the footnote sup was clicked - // or left border for first-column cells - if (el_td.cellIndex == '0') { - current_row.parentNode.rows[ - current_row.rowIndex + 1 - ].cells[0].setAttribute( - 'style', - `border-right:1px solid ${newColor} !important` - ); - } else { - el_td.style.borderBottom = `1px solid ${newColor}`; - } - - //hide all but the footnote clicked - for (const not of not_live_note) { - not.style.display = 'none'; - } - } else { - live_note[0].style.display = 'none'; - el_td.style.borderBottom = '1px solid #ddd'; - } - - //Dynamically add footnotes in Mobile view (under each cell, per click) - - //Add footnote innerHTML in span under clicked cell(display:block) in mobile - - //get innerHTML from chosen footnote using the results of the nextUntil function (i.e live_note const) - const live_note_text = live_note[0].cells[1].innerHTML; - - //define any existing responsive_footnote for just the current row(i.e. current_row const) - const resp_exists = current_row.querySelector(`.append${indx}`); - - //For toggling on/off the same one using superscript - if (!resp_exists) { - //if it does not already exist, add it with a class of the clicked index - el_td.insertAdjacentHTML( - 'beforeend', - `${live_note_text}` - ); - } - //else remove it - else { - resp_exists.remove(); - } - - //get all '.ftnt_append' classes per row - const resp_ftnts = current_row.getElementsByClassName('ftnt_append'); - - //For going from one footnote to another within each row - //Iterate over them and remove the ones that are not the clicked index number - Array.from(resp_ftnts).forEach(nt => { - if (nt.classList.contains(`append${clean_indx}`)) { - nt.style.display = 'block'; - } else { - nt.remove(); - } - }); - - //Extra close('x') button for mobile footnotes - if (resp_ftnts[0]) { - const close_btn = document.createElement('button'); - close_btn.classList.add('ftnt-close'); - close_btn.classList.add('button--close--primary'); - close_btn.textContent = 'x'; - resp_ftnts[0].appendChild(close_btn); - //get ftnt-close for just this row and activate it - const closer = current_row.querySelectorAll('.ftnt-close'); - closer[0].addEventListener('click', ex => { - const x = ex.target; - const this_span = x.closest('span'); - this_span.remove(); - }); - } -}; - -ReportingDates.prototype.stripeByState = function() { - const bg = 'rgba(241,241,241,.5)'; - const state_rows = this.dates_table.getElementsByTagName('tr'); - let state_class = []; - for (const tr of state_rows) { - state_class.push(tr.classList.item(0)); - } - let unique = [...new Set(state_class)]; - for (let x = 0; x < unique.length; x += 2) { - const unique_row = this.dates_table.getElementsByClassName(unique[x]); - for (const un of unique_row) { - un.style.backgroundColor = bg; - } - } -}; - -ReportingDates.prototype.convertFootnotes = function() { - -const all_hdr = this.dates_table.getElementsByTagName('th'); - Array.from(all_hdr).forEach(cell => { - const txt = cell.textContent; - - if (/~/.test(txt)) { - let txt_array = txt.split('~'); - let hdr_txt = txt_array.shift(); - let appended_hdr_notes = txt_array; - - let hdr_note_html_array = []; - for (let note of appended_hdr_notes) { - let hdr_note_html = `${note}`; - hdr_note_html_array.push(hdr_note_html); - - } - - cell.innerHTML = `${hdr_txt}${hdr_note_html_array}`; - - } - - }); - - //get all non-footnote/non-header row cells - const all_td = this.dates_table.querySelectorAll('tr:not(.footnote_row) td'); - - Array.from(all_td).forEach(cell => { - - const txt = cell.innerHTML; - - if (/~/.test(txt)) { - //Create an array from the string split on the tilda(s) - let txt_array = txt.split('~'); - ///The first item is the date text, return that as a var. Now txt_array only includes footnotes. - let date_txt = txt_array.shift(); - //Creeate a new varialble for clarity - let appended_footnotes = txt_array; - - let footnote_html_array = []; - for (let note of appended_footnotes) { - let footnote_html = `${note}`; - - footnote_html_array.push(footnote_html); - } - - cell.innerHTML = `${date_txt}${footnote_html_array}`; - - } - - }); - -}; - -new ReportingDates(); diff --git a/fec/fec/static/js/pages/reporting-dates-tables.js b/fec/fec/static/js/pages/reporting-dates-tables.js index 8c9f2a48e5..848d2a3661 100644 --- a/fec/fec/static/js/pages/reporting-dates-tables.js +++ b/fec/fec/static/js/pages/reporting-dates-tables.js @@ -11,7 +11,7 @@ const states_dropdown_template = ` - + @@ -72,39 +72,43 @@ const header_notes_modal_partial = `