1 /* global shortcut delBasket Sticky AUDIO_ALERT_PATH Cookies */
2 /* exported addBibToContext delBibToContext escape_str escape_price openWindow _ removeFocus toUC confirmDelete confirmClone playSound */
3 if ( KOHA === undefined ) var KOHA = {};
5 function _(s) { return s; } // dummy function for gettext
7 // http://stackoverflow.com/questions/1038746/equivalent-of-string-format-in-jquery/5341855#5341855
8 String.prototype.format = function() { return formatstr(this, arguments); };
9 function formatstr(str, col) {
10 col = typeof col === 'object' ? col : Array.prototype.slice.call(arguments, 1);
12 return str.replace(/%%|%s|%(\d+)\$s/g, function (m, n) {
13 if (m == "%%") { return "%"; }
14 if (m == "%s") { return col[idx++]; }
19 var HtmlCharsToEscape = {
24 String.prototype.escapeHtml = function() {
25 return this.replace(/[&<>]/g, function(c) {
26 return HtmlCharsToEscape[c] || c;
29 function escape_str(s){
30 return s != null ? s.escapeHtml() : "";
34 * Void method for numbers, for consistency
36 Number.prototype.escapeHtml = function() {
39 function escape_price(p){
40 return p != null ? p.escapeHtml().format_price() : "";
43 // http://stackoverflow.com/questions/14859281/select-tab-by-name-in-jquery-ui-1-10-0/16550804#16550804
44 $.fn.tabIndex = function () {
45 return $(this).parent().children('div').index(this);
47 $.fn.selectTabByID = function (tabID) {
48 $("a[href='" + tabID + "']", $(this) ).tab("show");
51 $(document).ready(function() {
53 //check for a hash before setting focus
54 let hash = window.location.hash;
56 $(".tab-pane.active input:text:first").focus();
58 $("#header_search a[data-toggle='tab']").on("shown.bs.tab", function (e) {
59 $( e.target.hash ).find("input:text:first").focus();
62 $(".close").click(function(){ window.close(); });
64 $("#checkin_search form").preventDoubleFormSubmit();
66 if($("#header_search #checkin_search").length > 0){
67 shortcut.add('Alt+r',function (){
68 $("#header_search").selectTabByID("#checkin_search");
69 $("#ret_barcode").focus();
72 shortcut.add('Alt+r',function (){
73 location.href="/cgi-bin/koha/circ/returns.pl"; });
75 if($("#header_search #circ_search").length > 0){
76 shortcut.add('Alt+u',function (){
77 $("#header_search").selectTabByID("#circ_search");
78 $("#findborrower").focus();
81 shortcut.add('Alt+u',function(){ location.href="/cgi-bin/koha/circ/circulation.pl"; });
83 if($("#header_search #catalog_search").length > 0){
84 shortcut.add('Alt+q',function (){
85 $("#header_search").selectTabByID("#catalog_search");
86 $("#search-form").focus();
89 shortcut.add('Alt+q',function(){ location.href="/cgi-bin/koha/catalogue/search.pl"; });
91 if($("#header_search #renew_search").length > 0){
92 shortcut.add('Alt+w',function (){
93 $("#header_search").selectTabByID("#renew_search");
94 $("#ren_barcode").focus();
97 shortcut.add('Alt+w',function(){ location.href="/cgi-bin/koha/circ/renew.pl"; });
100 $('#header_search .form-extra-content-toggle').on('click', function () {
101 const extraContent = $(this).closest('form').find('.form-extra-content');
102 if (extraContent.is(':visible')) {
104 $(this).removeClass("extra-content-open");
107 $(this).addClass("extra-content-open");
112 $(".validated").each(function() {
115 jQuery.validator.addClassRules("decimal", {
119 $("#logout").on("click",function(){
122 $("#helper").on("click",function(){
127 $("body").on("keypress", ".noEnterSubmit", function(e){
128 return checkEnter(e);
131 $(".keep_text").on("click",function(){
132 var field_index = $(this).parent().index();
133 keep_text( field_index );
136 $(".toggle_element").on("click",function(e){
138 $( $(this).data("element") ).toggle();
139 if (typeof Sticky !== "undefined" && typeof hcSticky === "function") {
140 Sticky.hcSticky('update');
144 var navmenulist = $("#navmenulist");
145 if( navmenulist.length > 0 ){
146 var path = location.pathname.substring(1);
147 var url = window.location.toString();
149 if ( url.match(/\?(.+)$/) ) {
150 params = "?" + RegExp.$1;
152 if ($("a[href$=\"/" + path + params + "\"]", navmenulist).length == 0){
153 $("a[href$=\"/" + path + "\"]", navmenulist).addClass("current");
155 $("a[href$=\"/" + path + params + "\"]", navmenulist).addClass("current");
159 $("#catalog-search-link a").on("mouseenter mouseleave", function(){
160 $("#catalog-search-dropdown a").toggleClass("catalog-search-dropdown-hover");
163 if ( localStorage.getItem("lastborrowernumber") ){
164 if( $("#hiddenborrowernumber").val() != localStorage.getItem("lastborrowernumber") ) {
165 $("#lastborrowerlink").show();
166 $("#lastborrowerlink").prop("title", localStorage.getItem("lastborrowername") + " (" + localStorage.getItem("lastborrowercard") + ")");
167 $("#lastborrowerlink").prop("href", "/cgi-bin/koha/circ/circulation.pl?borrowernumber=" + localStorage.getItem("lastborrowernumber"));
168 $("#lastborrower-window").css("display", "inline-flex");
172 if( !localStorage.getItem("lastborrowernumber") || ( $("#hiddenborrowernumber").val() != localStorage.getItem("lastborrowernumber") && localStorage.getItem("currentborrowernumber") != $("#hiddenborrowernumber").val())) {
173 if( $("#hiddenborrowernumber").val() ){
174 localStorage.setItem("lastborrowernumber", $("#hiddenborrowernumber").val() );
175 localStorage.setItem("lastborrowername", $("#hiddenborrowername").val() );
176 localStorage.setItem("lastborrowercard", $("#hiddenborrowercard").val() );
180 if( $("#hiddenborrowernumber").val() ){
181 localStorage.setItem("currentborrowernumber", $("#hiddenborrowernumber").val() );
184 $("#lastborrower-remove").click(function() {
185 removeLastBorrower();
186 $("#lastborrower-window").hide();
189 /* Search results browsing */
190 /* forms with action leading to search */
191 $("form[action*='search.pl']").submit(function(){
192 $('[name^="limit"]').each(function(){
193 if( $(this).val() == '' ){
194 $(this).prop("disabled","disabled");
197 var disabledPrior = false;
198 $(".search-term-row").each(function(){
200 $(this).find('select[name="op"]').prop("disabled","disabled");
201 disabledPrior = false;
203 if( $(this).find('input[name="q"]').val() == "" ){
204 $(this).find('input').prop("disabled","disabled");
205 $(this).find('select').prop("disabled","disabled");
206 disabledPrior = true;
209 resetSearchContext();
210 saveOrClearSimpleSearchParams();
212 /* any link to launch a search except navigation links */
213 $("[href*='search.pl?']").not(".nav").not('.searchwithcontext').click(function(){
214 resetSearchContext();
216 /* any link to a detail page from the results page. */
217 $("#bookbag_form a[href*='detail.pl?']").click(function(){
218 resetSearchContext();
223 function removeLastBorrower(){
224 localStorage.removeItem("lastborrowernumber");
225 localStorage.removeItem("lastborrowername");
226 localStorage.removeItem("lastborrowercard");
227 localStorage.removeItem("currentborrowernumber");
230 // http://jennifermadden.com/javascript/stringEnterKeyDetector.html
231 function checkEnter(e){ //e is event object passed from function invocation
232 var characterCode; // literal character code will be stored in this variable
233 if(e && e.which){ //if which property of event object is supported (NN4)
234 characterCode = e.which; //character code is contained in NN4's which property
236 characterCode = e.keyCode; //character code is contained in IE's keyCode property
238 if( characterCode == 13 //if generated character code is equal to ascii 13 (if enter key)
239 && e.target.nodeName == "INPUT"
240 && e.target.type != "submit" // Allow enter to submit using the submit button
248 function clearHoldFor(){
249 Cookies.remove("holdfor", { path: '/', SameSite: 'Lax' });
253 if( typeof delBasket == 'function' ){
254 delBasket('main', true);
257 removeLastBorrower();
258 localStorage.removeItem("sql_reports_activetab");
259 localStorage.removeItem("searches");
260 localStorage.removeItem("bibs_selected");
261 localStorage.removeItem("patron_search_selections");
265 window.open( "/cgi-bin/koha/help.pl", "_blank");
268 jQuery.fn.preventDoubleFormSubmit = function() {
269 jQuery(this).submit(function() {
270 $("body, form input[type='submit'], form button[type='submit'], form a").addClass('waiting');
271 if (this.beenSubmitted)
274 this.beenSubmitted = true;
278 function openWindow(link,name,width,height) {
279 name = (typeof name == "undefined")?'popup':name;
280 width = (typeof width == "undefined")?'600':width;
281 height = (typeof height == "undefined")?'400':height;
282 //IE <= 9 can't handle a "name" with whitespace
284 window.open(link,name,'width='+width+',height='+height+',resizable=yes,toolbar=false,scrollbars=yes,top');
286 window.open(link,null,'width='+width+',height='+height+',resizable=yes,toolbar=false,scrollbars=yes,top');
290 // Use this function to remove the focus from any element for
291 // repeated scanning actions on errors so the librarian doesn't
292 // continue scanning and miss the error.
293 function removeFocus() {
298 var x=f.value.toUpperCase();
303 function confirmDelete(message) {
304 return (confirm(message) ? true : false);
307 function confirmClone(message) {
308 return (confirm(message) ? true : false);
311 function playSound( sound ) {
312 if ( ! ( sound.indexOf('http://') === 0 || sound.indexOf('https://') === 0 ) ) {
313 sound = AUDIO_ALERT_PATH + sound;
315 document.getElementById("audio-alert").innerHTML = '<audio src="' + sound + '" autoplay="autoplay" autobuffer="autobuffer"></audio>';
318 // For keeping the text when navigating the search tabs
319 function keep_text(clicked_index) {
320 var searchboxes = document.getElementsByClassName("head-searchbox");
321 var persist = searchboxes[0].value;
323 for (var i = 0; i < searchboxes.length - 1; i++) {
324 if (searchboxes[i].value != searchboxes[i+1].value) {
325 if (i === searchboxes.length-2) {
326 if (searchboxes[i].value != searchboxes[0].value) {
327 persist = searchboxes[i].value;
328 } else if (searchboxes.length === 2) {
329 if (clicked_index === 0) {
330 persist = searchboxes[1].value;
333 persist = searchboxes[i+1].value;
335 } else if (searchboxes[i+1].value != searchboxes[i+2].value) {
336 persist = searchboxes[i+1].value;
341 for (i = 0; i < searchboxes.length; i++) {
342 searchboxes[i].value = persist;
346 // Extends jQuery API
347 jQuery.extend({uniqueArray:function(array){
348 return $.grep(array, function(el, index) {
349 return index === $.inArray(el, array);
353 function removeByValue(arr, val) {
354 for(var i=0; i<arr.length; i++) {
362 function addBibToContext( bibnum ) {
363 bibnum = parseInt(bibnum, 10);
364 var bibnums = getContextBiblioNumbers();
365 bibnums.push(bibnum);
366 setContextBiblioNumbers( bibnums );
367 setContextBiblioNumbers( $.uniqueArray( bibnums ) );
370 function delBibToContext( bibnum ) {
371 var bibnums = getContextBiblioNumbers();
372 removeByValue( bibnums, bibnum );
373 setContextBiblioNumbers( $.uniqueArray( bibnums ) );
376 function setContextBiblioNumbers( bibnums ) {
377 localStorage.setItem('bibs_selected', JSON.stringify( bibnums ) );
380 function getContextBiblioNumbers() {
381 var r = localStorage.getItem('bibs_selected');
383 return JSON.parse(r);
389 function resetSearchContext() {
390 setContextBiblioNumbers( new Array() );
393 function saveOrClearSimpleSearchParams() {
394 // Simple masthead search - pass value for display on details page
395 var pulldown_selection;
397 if( $("#cat-search-block select.advsearch").length ){
398 pulldown_selection = $("#cat-search-block select.advsearch").val();
400 pulldown_selection ="";
402 if( $("#cat-search-block #search-form").length ){
403 searchbox_value = $("#cat-search-block #search-form").val();
407 localStorage.setItem('cat_search_pulldown_selection', pulldown_selection );
408 localStorage.setItem('searchbox_value', searchbox_value );
411 function patron_autocomplete(node, options) {
414 let on_select_callback;
417 if (options['link-to']) {
418 link_to = options['link-to'];
420 if (options['url-params']) {
421 url_params = options['url-params'];
423 if (options['on-select-callback']) {
424 on_select_callback = options['on-select-callback'];
427 return node.autocomplete({
428 source: function (request, response) {
429 let q = buildPatronSearchQuery(request.term);
434 'q': JSON.stringify(q),
435 '_order_by': '+me.surname,+me.firstname',
440 url: '/api/v1/patrons',
442 "x-koha-embed": "library"
444 success: function (data) {
445 return response(data);
447 error: function (e) {
448 if (e.state() != 'rejected') {
449 alert(__("An error occurred. Check the logs"));
456 select: function (event, ui) {
458 window.location.href = ui.item.link;
459 } else if (on_select_callback) {
460 return on_select_callback(event, ui);
463 focus: function (event, ui) {
464 event.preventDefault(); // Don't replace the text field
467 .data("ui-autocomplete")
468 ._renderItem = function (ul, item) {
470 item.link = link_to == 'circ'
471 ? "/cgi-bin/koha/circ/circulation.pl"
472 : link_to == 'reserve'
473 ? "/cgi-bin/koha/reserve/request.pl"
474 : "/cgi-bin/koha/members/moremember.pl";
475 item.link += (url_params ? '?' + url_params + '&' : "?") + 'borrowernumber=' + item.patron_id;
481 if (item.cardnumber != "") {
482 // Display card number in parentheses if it exists
483 cardnumber = " (" + item.cardnumber + ") ";
485 if (item.library_id == loggedInLibrary) {
486 loggedInClass = "ac-currentlibrary";
490 return $("<li></li>")
491 .addClass(loggedInClass)
492 .data("ui-autocomplete-item", item)
495 + (item.link ? "<a href=\"" + item.link + "\">" : "<a>")
496 + (item.surname ? item.surname.escapeHtml() : "") + ", "
497 + (item.firstname ? item.firstname.escapeHtml() : "")
498 + cardnumber.escapeHtml()
500 + (item.date_of_birth
501 ? $date(item.date_of_birth)
502 + "<span class=\"age_years\"> ("
503 + $get_age(item.date_of_birth)
509 + $format_address(item, { no_line_break: true, include_li: false }) + " "
512 "<span class=\"ac-library\">"
513 + item.library.name.escapeHtml()
523 * Build patron search query
524 * - term: The full search term input by the user
525 * You can then pass a list of options:
526 * - search_type: String 'contains' or 'starts_with', defaults to DefaultPatronSearchMethod system preference
527 * - search_fields: String comma-separated list of specific fields, defaults to DefaultPatronSearchFields system preference
528 * - extended_attribute_types: JSON object containing the patron attribute types to be searched on
530 function buildPatronSearchQuery(term, options) {
533 let leading_wildcard;
535 let patterns = term.split(/[\s,]+/).filter(function (s) { return s.length });
537 // Bail if no patterns
538 if (patterns.length == 0) {
542 // Leading wildcard: If search_type option exists, we use that
543 if (typeof options !== 'undefined' && options.search_type) {
544 leading_wildcard = options.search_type === "contains" ? '%' : '';
545 // If not, we use DefaultPatronSearchMethod system preference instead
547 leading_wildcard = defaultPatronSearchMethod === 'contains' ? '%' : '';
550 // Search fields: If search_fields option exists, we use that
551 if (typeof options !== 'undefined' && options.search_fields) {
552 search_fields = options.search_fields;
553 // If not, we use DefaultPatronSearchFields system preference instead
555 search_fields = defaultPatronSearchFields;
558 // Add each pattern for each search field
559 let pattern_subquery_and = [];
560 patterns.forEach(function (pattern, i) {
561 let pattern_subquery_or = [];
562 search_fields.split(',').forEach(function (field, i) {
563 pattern_subquery_or.push(
564 { ["me." + field]: { 'like': leading_wildcard + pattern + '%' } }
566 if (field == 'dateofbirth') {
568 let d = $date_to_rfc3339(pattern);
569 pattern_subquery_or.push({ ["me." + field]: d });
571 // Hide the warning if the date is not correct
575 pattern_subquery_and.push(pattern_subquery_or);
577 q.push({ "-and": pattern_subquery_and });
579 // Add full search term for each search field
580 let term_subquery_or = [];
581 search_fields.split(',').forEach(function (field, i) {
582 term_subquery_or.push(
583 { ["me." + field]: { 'like': leading_wildcard + term + '%' } }
586 q.push({ "-or": term_subquery_or });
588 // Add each pattern for each extended patron attributes
589 if (typeof options !== 'undefined' && options.extended_attribute_types && extendedPatronAttributes) {
590 extended_attribute_subquery_and = [];
591 patterns.forEach(function (pattern, i) {
592 let extended_attribute_sub_or = [];
593 extended_attribute_sub_or.push({
594 "extended_attributes.value": { "like": leading_wildcard + pattern + '%' },
595 "extended_attributes.code": options.extended_attribute_types
597 extended_attribute_subquery_and.push(extended_attribute_sub_or);
599 q.push({ "-and": extended_attribute_subquery_and });
604 function selectBsTabByHash( tabs_container_id ){
605 /* Check for location.hash in the page URL */
606 /* If present the location hash will be used to activate the correct tab */
607 var hash = document.location.hash;
609 $('#' + tabs_container_id + ' a[href="' + hash + '"]').tab('show');
611 $('#' + tabs_container_id + ' a:first').tab('show');