Bug 14816: Fix multiple selection in item search
[koha-ffzg.git] / koha-tmpl / intranet-tmpl / prog / en / modules / catalogue / itemsearch.tt
1 [% USE CGI %]
2 [% USE JSON.Escape %]
3
4 [% BLOCK form_label %]
5   [% SWITCH label %]
6     [% CASE 'barcode' %]<span>Barcode</span>
7     [% CASE 'itemcallnumber' %]<span>Call number</span>
8     [% CASE 'stocknumber' %]<span>Inventory number</span>
9     [% CASE 'title' %]<span>Title</span>
10     [% CASE 'author' %]<span>Author</span>
11     [% CASE 'publishercode' %]<span>Publisher</span>
12     [% CASE 'publicationyear' %]<span>Publication date</span>
13     [% CASE 'collectiontitle' %]<span>Collection</span>
14     [% CASE 'isbn' %]<span>ISBN</span>
15     [% CASE 'issn' %]<span>ISSN</span>
16     [% CASE 'homebranch' %]<span>Home library</span>
17     [% CASE 'All libraries' %]<span>All libraries</span>
18     [% CASE 'location' %]<span>Shelving location</span>
19     [% CASE 'All locations' %]<span>All locations</span>
20     [% CASE 'itype' %]<span>Item type</span>
21     [% CASE 'All item types' %]<span>All item types</span>
22     [% CASE 'ccode' %]<span>Collection code</span>
23     [% CASE 'All collection codes' %]<span>All collection codes</span>
24     [% CASE 'notforloan' %]<span>Status</span>
25     [% CASE 'All statuses' %]<span>All statuses</span>
26     [% CASE 'damaged' %]<span>Damaged</span>
27     [% CASE 'itemlost' %]<span>Lost</span>
28   [% END %]
29 [% END %]
30
31 [% BLOCK form_field_select %]
32   <div class="form-field form-field-select">
33     <label class="form-field-label" for="[% name %]">[% INCLUDE form_label label=name %]</label>
34     <select id="[% name %]_op" name="[% name %]_op">
35       <option value="=">is</option>
36       [% IF CGI.param(name _ '_op') == '!=' %]
37         <option value="!=" selected="selected">is not</option>
38       [% ELSE %]
39         <option value="!=" >is not</option>
40       [% END %]
41     </select>
42     [% values = CGI.param(name) %]
43     <select id="[% name %]" name="[% name %]" multiple="multiple" size="[% options.size < 4 ? options.size + 1 : 4 %]">
44       [% IF (values == '') %]
45         <option value="" selected="selected">
46       [% ELSE %]
47         <option value="">
48       [% END %]
49         [% IF (empty_option) %][% INCLUDE form_label label=empty_option %][% ELSE %]<span>All</span>[% END %]
50       </option>
51       [% FOREACH option IN options %]
52         [% IF values != '' && values.grep(option.value).size %]
53           <option value="[% option.value %]" selected="selected">[% option.label %]</option>
54         [% ELSE %]
55           <option value="[% option.value %]">[% option.label %]</option>
56         [% END %]
57       [% END %]
58     </select>
59   </div>
60 [% END %]
61
62 [% BLOCK form_field_select_option %]
63   [% IF params.f == value %]
64     <option value="[% value %]" selected="selected">[% INCLUDE form_label label=value %]</option>
65   [% ELSE %]
66     <option value="[% value %]">[% INCLUDE form_label label=value %]</option>
67   [% END %]
68 [% END %]
69
70 [% BLOCK form_field_select_text %]
71   <div class="form-field form-field-select-text">
72     [% IF params.exists('c') %]
73       <select name="c" class="form-field-conjunction">
74         <option value="and">AND</option>
75         [% IF params.c == 'or' %]
76           <option value="or" selected="selected">OR</option>
77         [% ELSE %]
78           <option value="or">OR</option>
79         [% END %]
80       </select>
81     [% ELSE %]
82       <select name="c" class="form-field-conjunction" disabled="disabled">
83         <option value="and">AND</option>
84         <option value="or">OR</option>
85       </select>
86     [% END %]
87     <select name="f" class="form-field-column">
88       [% INCLUDE form_field_select_option value='barcode' %]
89       [% INCLUDE form_field_select_option value='itemcallnumber' %]
90       [% INCLUDE form_field_select_option value='stocknumber' %]
91       [% INCLUDE form_field_select_option value='title' %]
92       [% INCLUDE form_field_select_option value='author' %]
93       [% INCLUDE form_field_select_option value='publishercode' %]
94       [% INCLUDE form_field_select_option value='publicationyear' %]
95       [% INCLUDE form_field_select_option value='collectiontitle' %]
96       [% INCLUDE form_field_select_option value='isbn' %]
97       [% INCLUDE form_field_select_option value='issn' %]
98       [% IF items_search_fields.size %]
99         <optgroup label="Custom search fields">
100           [% FOREACH field IN items_search_fields %]
101             [% marcfield = field.tagfield %]
102             [% IF field.tagsubfield %]
103               [% marcfield = marcfield _ '$' _ field.tagsubfield %]
104             [% END %]
105             [% IF params.f == "marc:$marcfield" %]
106               <option value="marc:[% marcfield %]" data-authorised-values-category="[% field.authorised_values_category %]" selected="selected">[% field.label %] ([% marcfield %])</option>
107             [% ELSE %]
108               <option value="marc:[% marcfield %]" data-authorised-values-category="[% field.authorised_values_category %]">[% field.label %] ([% marcfield %])</option>
109             [% END %]
110           [% END %]
111         </optgroup>
112       [% END %]
113     </select>
114     <input type="text" name="q" class="form-field-value" value="[% params.q %]" />
115     <input type="hidden" name="op" value="like" />
116   </div>
117 [% END %]
118
119 [% BLOCK form_field_select_text_block %]
120   [% c = CGI.param('c').list %]
121   [% f = CGI.param('f').list %]
122   [% q = CGI.param('q').list %]
123   [% op = CGI.param('op').list %]
124   [% IF q.size %]
125     [% size = q.size - 1 %]
126     [% FOREACH i IN [0 .. size] %]
127       [%
128         params = {
129           f => f.$i
130           q = q.$i
131           op = op.$i
132         }
133       %]
134       [% IF i > 0 %]
135         [% j = i - 1 %]
136         [% params.c = c.$j %]
137       [% END %]
138       [% INCLUDE form_field_select_text params=params %]
139     [% END %]
140   [% ELSE %]
141     [% INCLUDE form_field_select_text %]
142   [% END %]
143 [% END %]
144
145 [% BLOCK form_field_radio_yes_no %]
146   <div class="form-field">
147     <label class="form-field-label">[% INCLUDE form_label label=name %]:</label>
148     <input type="radio" name="[% name %]" id="[% name %]_indifferent" value="" checked="checked"/>
149     <label for="[% name %]_indifferent">Ignore</label>
150     <input type="radio" name="[% name %]" id="[% name %]_yes" value="yes" />
151     <label for="[% name %]_yes">Yes</label>
152     <input type="radio" name="[% name %]" id="[% name %]_no" value="no" />
153     <label for="[% name %]_no">No</label>
154   </div>
155 [% END %]
156
157 [%# Page starts here %]
158
159 [% INCLUDE 'doc-head-open.inc' %]
160   <title>Koha &rsaquo; Catalog &rsaquo; Item search</title>
161   [% INCLUDE 'doc-head-close.inc' %]
162   <link rel="stylesheet" type="text/css" href="[% themelang %]/css/datatables.css" />
163   [% INCLUDE 'datatables.inc' %]
164   <script type="text/javascript" src="[% interface %]/lib/jquery/plugins/jquery.dataTables.columnFilter.js"></script>
165   <link rel="stylesheet" type="text/css" href="[% themelang %]/css/itemsearchform.css" />
166   <script type="text/javascript">
167     //<![CDATA[
168     var authorised_values = [% authorised_values_json %];
169
170     function loadAuthorisedValuesSelect(select) {
171       var selected = select.find('option:selected');
172       var category = selected.data('authorised-values-category');
173       var form_field_value = select.siblings('.form-field-value');
174       if (category && category in authorised_values) {
175         var values = authorised_values[category];
176         var html = '<select name="q" class="form-field-value">\n';
177         for (i in values) {
178           var value = values[i];
179           html += '<option value="' + value.authorised_value + '">' + value.lib + '</option>\n';
180         }
181         html += '</select>\n';
182         var new_form_field_value = $(html);
183         new_form_field_value.val(form_field_value.val());
184         form_field_value.replaceWith(new_form_field_value);
185       } else {
186         if (form_field_value.prop('tagName').toLowerCase() == 'select') {
187           html = '<input name="q" type="text" class="form-field-value" />';
188           var new_form_field_value = $(html);
189           form_field_value.replaceWith(new_form_field_value);
190         }
191       }
192     }
193
194     function addNewField() {
195       var form_field = $('div.form-field-select-text').last();
196       var copy = form_field.clone(true);
197       copy.find('input,select').not('[type="hidden"]').each(function() {
198         $(this).val('');
199       });
200       copy.find('.form-field-conjunction').removeAttr('disabled');
201       form_field.after(copy);
202       copy.find('select.form-field-column').change();
203     }
204
205     function submitForm($form) {
206       var tr = ''
207         + '    <tr>'
208         + '      <th>' + _("Title") + '</th>'
209         + '      <th>' + _("Publication date") + '</th>'
210         + '      <th>' + _("Publisher") + '</th>'
211         + '      <th>' + _("Collection") + '</th>'
212         + '      <th>' + _("Barcode") + '</th>'
213         + '      <th>' + _("Call number") + '</th>'
214         + '      <th>' + _("Home library") + '</th>'
215         + '      <th>' + _("Current location") + '</th>'
216         + '      <th>' + _("Shelving location") + '</th>'
217         + '      <th>' + _("Inventory number") + '</th>'
218         + '      <th>' + _("Status") + '</th>'
219         + '      <th>' + _("Checkouts") + '</th>'
220         + '      <th></th>'
221         + '    </tr>'
222       var table = ''
223         + '<table id="results">'
224         + '  <thead>' + tr + tr + '</thead>'
225         + '  <tbody></tbody>'
226         + '</table>';
227       var results_heading = "<h1>" + _("Item search results") + "</h1>";
228       results_heading += "<p><a href=\"/cgi-bin/koha/catalogue/search.pl\">" + _("Go to advanced search") + "</a></p>";
229       results_heading += "<p><a class=\"editsearchlink\" href=\"#\">" + _("Edit search") + "</a></p>";
230       $('#results-wrapper').empty().html(results_heading + table);
231
232       var params = [];
233       $form.find('select').not('[disabled]').find('option:selected').each(function () {
234         var name = $(this).parent('select').attr('name');
235         var value = $(this).val();
236         params.push({ 'name': name, 'value': value });
237       });
238       $form.find('input[type="text"],input[type="hidden"]').not('[disabled]').each(function () {
239         params.push({ 'name': $(this).attr('name'), 'value': $(this).val() });
240       });
241       $form.find('input[type="radio"]:checked').each(function() {
242         params.push({ 'name': $(this).attr('name'), 'value': $(this).val() });
243       });
244
245       $('#results').dataTable($.extend(true, {}, dataTablesDefaults, {
246         'bDestroy': true,
247         'bServerSide': true,
248         'bProcessing': true,
249         'sAjaxSource': '/cgi-bin/koha/catalogue/itemsearch.pl',
250         'fnServerData': function(sSource, aoData, fnCallback) {
251           aoData.push( { 'name': 'format', 'value': 'json' } );
252           for (i in params) {
253             aoData.push(params[i]);
254           }
255           $.ajax({
256               'dataType': 'json',
257               'type': 'POST',
258               'url': sSource,
259               'data': aoData,
260               'success': function(json){
261                   fnCallback(json);
262               }
263           });
264         },
265         'sDom': '<"top pager"ilp>t<"bottom pager"ip>r',
266         'aoColumns': [
267           { 'sName': 'title' },
268           { 'sName': 'publicationyear' },
269           { 'sName': 'publishercode' },
270           { 'sName': 'collectiontitle' },
271           { 'sName': 'barcode' },
272           { 'sName': 'itemcallnumber' },
273           { 'sName': 'homebranch' },
274           { 'sName': 'holdingbranch' },
275           { 'sName': 'location' },
276           { 'sName': 'stocknumber' },
277           { 'sName': 'notforloan' },
278           { 'sName': 'issues' },
279           { 'sName': 'checkbox', 'bSortable': false }
280         ],
281         "sPaginationType": "full_numbers"
282       })).columnFilter({
283         'sPlaceHolder': 'head:after',
284         'aoColumns': [
285           { 'type': 'text' },
286           { 'type': 'text' },
287           { 'type': 'text' },
288           { 'type': 'text' },
289           { 'type': 'text' },
290           { 'type': 'text' },
291           { 'type': 'select', 'values': [% branches.json %] },
292           { 'type': 'select', 'values': [% branches.json %] },
293           { 'type': 'select', 'values': [% locations.json %] },
294           { 'type': 'text' },
295           { 'type': 'select', 'values': [% notforloans.json %] },
296           { 'type': 'text' },
297           null
298         ]
299       });
300     }
301
302     function hideForm() {
303       $("#item-search-block").hide();
304       $('.editsearchlink').show();
305     }
306
307     $(document).ready(function () {
308       // Add the "New field" link.
309       var form_field = $('div.form-field-select-text').last()
310       var NEW_FIELD = _("New field");
311       var button_field_new = $('<a href="#" class="button-field-new" title="Add a new field">' + NEW_FIELD + '</a>');
312       button_field_new.click(function() {
313         addNewField();
314         return false;
315       });
316       form_field.after(button_field_new);
317
318       // If a field is linked to an authorised values list, display the list.
319       $('div.form-field-select-text select').change(function() {
320         loadAuthorisedValuesSelect($(this));
321       }).change();
322
323       // Prevent user to select the 'All ...' option with other options.
324       $('div.form-field-select').each(function() {
325         $(this).find('select').filter(':last').change(function() {
326           values = $(this).val();
327           if (values.length > 1) {
328             var idx = $.inArray('', values);
329             if (idx != -1) {
330               values.splice(idx, 1);
331               $(this).val(values);
332             }
333           }
334         });
335       });
336
337       $('#itemsearchform').submit(function() {
338         var searchform = $(this);
339         var format = searchform.find('input[name="format"]:checked').val();
340         if (format == 'html') {
341           submitForm(searchform);
342           hideForm();
343           return false;
344         }
345       });
346
347       $("body").on("click",".editsearchlink",function(e) {
348         e.preventDefault();
349         $('#item-search-block').show();
350         $(this).hide();
351         return false;
352       });
353     });
354     //]]>
355   </script>
356 </head>
357 <body id="catalog_itemsearch" class="catalog">
358   [% INCLUDE 'header.inc' %]
359   <div id="breadcrumbs">
360     <a href="/cgi-bin/koha/mainpage.pl">Home</a> &rsaquo; <a href="/cgi-bin/koha/catalogue/search.pl">Catalog</a> &rsaquo; Item search
361   </div>
362
363   <div id="doc" class="yui-t7">
364     <div id="item-search-block">
365       <h1>Item search</h1>
366       <p><a href="/cgi-bin/koha/catalogue/search.pl">Go to advanced search</a></p>
367       <form action="/cgi-bin/koha/catalogue/itemsearch.pl" method="get" id="itemsearchform">
368           <fieldset>
369             [% INCLUDE form_field_select
370               name="homebranch"
371               options = branches
372               empty_option = "All libraries"
373             %]
374             [% INCLUDE form_field_select
375               name="location"
376               options = locations
377               empty_option = "All locations"
378             %]
379           </fieldset>
380           <fieldset>
381             [% INCLUDE form_field_select
382               name="itype"
383               options = itemtypes
384               empty_option = "All item types"
385             %]
386             [% INCLUDE form_field_select
387               name="ccode"
388               options = ccodes
389               empty_option = "All collection codes"
390             %]
391             [% INCLUDE form_field_select
392               name="notforloan"
393               options = notforloans
394               empty_option = "All statuses"
395             %]
396           </fieldset>
397           <fieldset>
398             [% INCLUDE form_field_select_text_block %]
399             <p class="hint">You can use the following wildcard characters: % _</p>
400             <p class="hint">% matches any number of characters</p>
401             <p class="hint">_ matches only a single character</p>
402           </fieldset>
403           <fieldset>
404             <div class="form-field">
405               <label class="form-field-label" for="itemcallnumber_from">From call number:</label>
406               [% value = CGI.param('itemcallnumber_from') %]
407               <input type="text" id="itemcallnumber_from" name="itemcallnumber_from" value="[% value %]" />
408               <span class="hint">(inclusive)</span>
409             </div>
410             <div class="form-field">
411               [% value = CGI.param('itemcallnumber_to') %]
412               <label class="form-field-label" for="itemcallnumber_to">To call number:</label>
413               <input type="text" id="itemcallnumber_to" name="itemcallnumber_to" value="[% value %]" />
414               <span class="hint">(inclusive)</span>
415             </div>
416             [% INCLUDE form_field_radio_yes_no name="damaged" %]
417             [% INCLUDE form_field_radio_yes_no name="itemlost" %]
418             <div class="form-field">
419               <label class="form-field-label" for="issues_op">Checkout count:</label>
420               <select id="issues_op" name="issues_op">
421                 <option value=">">&gt;</option>
422                 <option value="<">&lt;</option>
423                 <option value="=">=</option>
424                 <option value="!=">!=</option>
425               </select>
426               <input type="text" name="issues" />
427             </div>
428             <div class="form-field">
429               <label class="form-field-label" for="datelastborrowed_op">Last checkout date:</label>
430               <select id="datelastborrowed_op" name="datelastborrowed_op">
431                 <option value=">">After</option>
432                 <option value="<">Before</option>
433                 <option value="=">On</option>
434               </select>
435               <input type="text" name="datelastborrowed" />
436               <span class="hint">ISO Format (YYYY-MM-DD)</span>
437             </div>
438           </fieldset>
439           <fieldset>
440             <div class="form-field-radio">
441               <label>Output:</label>
442               <input type="radio" id="format-html" name="format" value="html" checked="checked" /> <label for="format-html">Screen</label>
443               <input type="radio" id="format-csv" name="format" value="csv" /> <label for="format-csv">CSV</label>
444             </div>
445             <div class="form-actions">
446               <input type="submit" value="Search" />
447             </div>
448           </fieldset>
449       </form>
450
451       <p><a id="editsearchlink" href="#" style="display:none">Edit search</a></p>
452     </div>
453   </div>
454   <div id="doc3" class="yui-t7">
455       <div id="results-wrapper">
456         [% IF search_done %]
457           [% IF total_rows > 0 %]
458             <p>Found [% total_rows %] results.</p>
459           [% ELSE %]
460             <p>No results found.</p>
461           [% END %]
462
463           [% IF results %]
464             [% INCLUDE 'catalogue/itemsearch_items.inc' items = results %]
465           [% END %]
466
467           <div class="pages">
468             [% pagination_bar %]
469           </div>
470
471         [% END %]
472       </div>
473
474     [% INCLUDE 'intranet-bottom.inc' %]