Bug 26224: Prevent double submit of header check in form
[koha.git] / koha-tmpl / intranet-tmpl / prog / js / staff-global.js
1 /* global shortcut delCookie delBasket Sticky */
2 /* exported paramOfUrl addBibToContext delBibToContext */
3 if ( KOHA === undefined ) var KOHA = {};
4
5 function _(s) { return s; } // dummy function for gettext
6
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);
11     var idx = 0;
12     return str.replace(/%%|%s|%(\d+)\$s/g, function (m, n) {
13         if (m == "%%") { return "%"; }
14         if (m == "%s") { return col[idx++]; }
15         return col[n];
16     });
17 }
18
19 var HtmlCharsToEscape = {
20     '&': '&',
21     '<': '&lt;',
22     '>': '&gt;'
23 };
24 String.prototype.escapeHtml = function() {
25     return this.replace(/[&<>]/g, function(c) {
26         return HtmlCharsToEscape[c] || c;
27     });
28 };
29
30 /*
31  * Void method for numbers, for consistency
32  */
33 Number.prototype.escapeHtml = function() {
34     return this;
35 };
36
37 // http://stackoverflow.com/questions/14859281/select-tab-by-name-in-jquery-ui-1-10-0/16550804#16550804
38 $.fn.tabIndex = function () {
39     return $(this).parent().children('div').index(this);
40 };
41 $.fn.selectTabByID = function (tabID) {
42     $(this).tabs("option", "active", $( tabID ).tabIndex());
43 };
44
45  $(document).ready(function() {
46     $('#header_search').tabs().on( "tabsactivate", function(e, ui) { $(this).find("div:visible").find('input').eq(0).focus(); });
47
48     $(".close").click(function(){ window.close(); });
49
50     $("#checkin_search form").preventDoubleFormSubmit();
51
52     if($("#header_search #checkin_search").length > 0){ shortcut.add('Alt+r',function (){ $("#header_search").selectTabByID("#checkin_search"); $("#ret_barcode").focus(); }); } else { shortcut.add('Alt+r',function (){ location.href="/cgi-bin/koha/circ/returns.pl"; }); }
53     if($("#header_search #circ_search").length > 0){ shortcut.add('Alt+u',function (){ $("#header_search").selectTabByID("#circ_search"); $("#findborrower").focus(); }); } else { shortcut.add('Alt+u',function(){ location.href="/cgi-bin/koha/circ/circulation.pl"; }); }
54     if($("#header_search #catalog_search").length > 0){ shortcut.add('Alt+q',function (){ $("#header_search").selectTabByID("#catalog_search"); $("#search-form").focus(); }); } else { shortcut.add('Alt+q',function(){ location.href="/cgi-bin/koha/catalogue/search.pl"; }); }
55     if($("#header_search #renew_search").length > 0){ shortcut.add('Alt+w',function (){ $("#header_search").selectTabByID("#renew_search"); $("#ren_barcode").focus(); }); } else { shortcut.add('Alt+w',function(){ location.href="/cgi-bin/koha/circ/renew.pl"; }); }
56
57     $("#header_search > ul > li").show();
58
59     $(".focus").focus();
60     $(".validated").each(function() {
61         $(this).validate();
62     });
63
64     $("#logout").on("click",function(){
65         logOut();
66     });
67     $("#helper").on("click",function(){
68         openHelp();
69         return false;
70     });
71
72     $("body").on("keypress", ".noEnterSubmit", function(e){
73         return checkEnter(e);
74     });
75
76     $(".keep_text").on("click",function(){
77         var field_index = $(this).parent().index();
78         keep_text( field_index );
79     });
80
81     $(".toggle_element").on("click",function(e){
82         e.preventDefault();
83         $( $(this).data("element") ).toggle();
84         if (typeof Sticky !== "undefined" && typeof hcSticky === "function") {
85             Sticky.hcSticky('update');
86         }
87     });
88
89     var navmenulist = $("#navmenulist");
90     if( navmenulist.length > 0 ){
91         var path = location.pathname.substring(1);
92         var url = window.location.toString();
93         var params = '';
94         if ( url.match(/\?(.+)$/) ) {
95             params = "?" + RegExp.$1;
96         }
97         $("a[href$=\"/" + path + params + "\"]", navmenulist).addClass("current");
98     }
99
100     $("#catalog-search-link a").on("hover", function(){
101         $("#catalog-search-dropdown a").toggleClass("catalog-search-dropdown-hover");
102     });
103
104     if ( localStorage.getItem("lastborrowernumber") ){
105         if( $("#hiddenborrowernumber").val() != localStorage.getItem("lastborrowernumber") ) {
106             $("#lastborrower-window").detach().appendTo("#breadcrumbs");
107             $("#lastborrowerlink").show();
108             $("#lastborrowerlink").prop("title", localStorage.getItem("lastborrowername") + " (" + localStorage.getItem("lastborrowercard") + ")");
109             $("#lastborrowerlink").prop("href", "/cgi-bin/koha/circ/circulation.pl?borrowernumber=" + localStorage.getItem("lastborrowernumber"));
110             $("#lastborrower-window").css("display", "inline-block");
111         }
112     }
113
114     if( !localStorage.getItem("lastborrowernumber") || ( $("#hiddenborrowernumber").val() != localStorage.getItem("lastborrowernumber") && localStorage.getItem("currentborrowernumber") != $("#hiddenborrowernumber").val())) {
115         if( $("#hiddenborrowernumber").val() ){
116             localStorage.setItem("lastborrowernumber", $("#hiddenborrowernumber").val() );
117             localStorage.setItem("lastborrowername", $("#hiddenborrowername").val() );
118             localStorage.setItem("lastborrowercard", $("#hiddenborrowercard").val() );
119         }
120     }
121
122     if( $("#hiddenborrowernumber").val() ){
123         localStorage.setItem("currentborrowernumber", $("#hiddenborrowernumber").val() );
124     }
125
126     $("#lastborrower-remove").click(function() {
127         removeLastBorrower();
128         $("#lastborrower-window").hide();
129     });
130
131     /* Search results browsing */
132     /* forms with action leading to search */
133     $("form[action*='search.pl']").submit(function(){
134         resetSearchContext();
135     });
136     /* any link to launch a search except navigation links */
137     $("[href*='search.pl?']").not(".nav").not('.searchwithcontext').click(function(){
138         resetSearchContext();
139     });
140     /* any link to a detail page from the results page. */
141     $("#bookbag_form a[href*='detail.pl?']").click(function(){
142         resetSearchContext();
143     });
144
145 });
146
147 function removeLastBorrower(){
148     localStorage.removeItem("lastborrowernumber");
149     localStorage.removeItem("lastborrowername");
150     localStorage.removeItem("lastborrowercard");
151     localStorage.removeItem("currentborrowernumber");
152 }
153
154 // http://jennifermadden.com/javascript/stringEnterKeyDetector.html
155 function checkEnter(e){ //e is event object passed from function invocation
156     var characterCode; // literal character code will be stored in this variable
157     if(e && e.which){ //if which property of event object is supported (NN4)
158         characterCode = e.which; //character code is contained in NN4's which property
159     } else {
160         characterCode = e.keyCode; //character code is contained in IE's keyCode property
161     }
162     if( characterCode == 13 //if generated character code is equal to ascii 13 (if enter key)
163         && e.target.nodeName == "INPUT"
164         && e.target.type != "submit" // Allow enter to submit using the submit button
165     ){
166         return false;
167     } else {
168         return true;
169     }
170 }
171
172 function clearHoldFor(){
173     $.removeCookie("holdfor", { path: '/' });
174 }
175
176 function logOut(){
177     if( typeof delBasket == 'function' ){
178         delBasket('main', true);
179     }
180     clearHoldFor();
181     removeLastBorrower();
182 }
183
184 function openHelp(){
185     window.open( "/cgi-bin/koha/help.pl", "_blank");
186 }
187
188 jQuery.fn.preventDoubleFormSubmit = function() {
189     jQuery(this).submit(function() {
190     $("body, form input[type='submit'], form button[type='submit'], form a").addClass('waiting');
191         if (this.beenSubmitted)
192             return false;
193         else
194             this.beenSubmitted = true;
195     });
196 };
197
198 function openWindow(link,name,width,height) {
199     name = (typeof name == "undefined")?'popup':name;
200     width = (typeof width == "undefined")?'600':width;
201     height = (typeof height == "undefined")?'400':height;
202     var newwin;
203     //IE <= 9 can't handle a "name" with whitespace
204     try {
205         newin=window.open(link,name,'width='+width+',height='+height+',resizable=yes,toolbar=false,scrollbars=yes,top');
206     } catch(e) {
207         newin=window.open(link,null,'width='+width+',height='+height+',resizable=yes,toolbar=false,scrollbars=yes,top');
208     }
209 }
210
211 // Use this function to remove the focus from any element for
212 // repeated scanning actions on errors so the librarian doesn't
213 // continue scanning and miss the error.
214 function removeFocus() {
215     $(':focus').blur();
216 }
217
218 function toUC(f) {
219     var x=f.value.toUpperCase();
220     f.value=x;
221     return true;
222 }
223
224 function confirmDelete(message) {
225     return (confirm(message) ? true : false);
226 }
227
228 function confirmClone(message) {
229     return (confirm(message) ? true : false);
230 }
231
232 function playSound( sound ) {
233     if ( ! ( sound.indexOf('http://') === 0 || sound.indexOf('https://') === 0  ) ) {
234         sound = AUDIO_ALERT_PATH + sound;
235     }
236     document.getElementById("audio-alert").innerHTML = '<audio src="' + sound + '" autoplay="autoplay" autobuffer="autobuffer"></audio>';
237 }
238
239 // For keeping the text when navigating the search tabs
240 function keep_text(clicked_index) {
241     var searchboxes = document.getElementsByClassName("head-searchbox");
242     var persist = searchboxes[0].value;
243
244     for (i = 0; i < searchboxes.length - 1; i++) {
245         if (searchboxes[i].value != searchboxes[i+1].value) {
246             if (i === searchboxes.length-2) {
247                 if (searchboxes[i].value != searchboxes[0].value) {
248                     persist = searchboxes[i].value;
249                 } else if (searchboxes.length === 2) {
250                     if (clicked_index === 0) {
251                         persist = searchboxes[1].value;
252                     }
253                 } else {
254                     persist = searchboxes[i+1].value;
255                 }
256             } else if (searchboxes[i+1].value != searchboxes[i+2].value) {
257                 persist = searchboxes[i+1].value;
258             }
259         }
260     }
261
262     for (i = 0; i < searchboxes.length; i++) {
263         searchboxes[i].value = persist;
264     }
265 }
266
267 // Extends jQuery API
268 jQuery.extend({uniqueArray:function(array){
269     return $.grep(array, function(el, index) {
270         return index === $.inArray(el, array);
271     });
272 }});
273
274 function removeByValue(arr, val) {
275     for(var i=0; i<arr.length; i++) {
276         if(arr[i] == val) {
277             arr.splice(i, 1);
278             break;
279         }
280     }
281 }
282
283 function paramOfUrl( url, param ) {
284     param = param.replace(/[\[]/,"\\\[").replace(/[\]]/,"\\\]");
285     var regexS = "[\\?&]"+param+"=([^&#]*)";
286     var regex = new RegExp( regexS );
287     var results = regex.exec( url );
288     if( results == null ) {
289         return "";
290     } else {
291         return results[1];
292     }
293 }
294
295 function addBibToContext( bibnum ) {
296     bibnum = parseInt(bibnum, 10);
297     var bibnums = getContextBiblioNumbers();
298     bibnums.push(bibnum);
299     setContextBiblioNumbers( bibnums );
300     setContextBiblioNumbers( $.uniqueArray( bibnums ) );
301 }
302
303 function delBibToContext( bibnum ) {
304     var bibnums = getContextBiblioNumbers();
305     removeByValue( bibnums, bibnum );
306     setContextBiblioNumbers( $.uniqueArray( bibnums ) );
307 }
308
309 function setContextBiblioNumbers( bibnums ) {
310     $.cookie('bibs_selected', JSON.stringify( bibnums ));
311 }
312
313 function getContextBiblioNumbers() {
314     var r = $.cookie('bibs_selected');
315     if ( r ) {
316         return JSON.parse(r);
317     }
318     r = new Array();
319     return r;
320 }
321
322 function resetSearchContext() {
323     setContextBiblioNumbers( new Array() );
324 }