4 [% USE TablesSettings %]
8 [% PROCESS 'patronfields.inc' %]
9 [% INCLUDE 'doc-head-open.inc' %]
10 <title>Patrons[% IF ( searching ) %] › Search results[% END %] › Koha</title>
11 [% INCLUDE 'doc-head-close.inc' %]
14 <body id="pat_member" class="pat">
15 [% INCLUDE 'header.inc' %]
16 [% INCLUDE 'patron-search.inc' %]
18 <nav id="breadcrumbs" aria-label="Breadcrumb" class="breadcrumb">
21 <a href="/cgi-bin/koha/mainpage.pl">Home</a>
23 [% IF ( searching ) %]
25 <a href="/cgi-bin/koha/members/members-home.pl">Patrons</a>
28 <a href="#" aria-current="page">
34 <a href="#" aria-current="page">
42 <div class="main container-fluid">
44 <div class="col-sm-10 col-sm-push-2">
47 [% IF CAN_user_tools_manage_patron_lists %]
48 <div id="patron_list_dialog" class="dialog alert">
49 Added <span class="patrons-length"></span> patrons to <a></a>.
53 [% INCLUDE 'patron-toolbar.inc' %]
54 [% INCLUDE 'noadd-warnings.inc' %]
58 [% FOREACH letter IN alphabet.split(' ') %]
59 <a href="#" class="filterByLetter">[% letter | html %]</a>
63 [% IF CAN_user_borrowers_edit_borrowers && pending_borrower_modifications %]
64 <div class="pending-info" id="patron_updates_pending">
65 <a href="/cgi-bin/koha/members/members-update.pl">Patrons requesting modifications</a>:
66 <span class="number_box"><a href="/cgi-bin/koha/members/members-update.pl">[% pending_borrower_modifications | html %]</a></span>
70 <div id="searchresults">
71 <h3>Patrons found for: <span id="searchpattern">[% IF searchmember %] for '[% searchmember | html %]'[% END %]</span></h3>
72 [% IF CAN_user_tools_manage_patron_lists || CAN_user_borrowers_edit_borrowers %]
73 <div class="searchheader fh-fixedHeader" id="searchheader">
75 <a href="#" id="select_all"><i class="fa fa-check"></i> Select all</a>
77 <a href="#" id="clear_all"><i class="fa fa-remove"></i> Clear all</a>
78 [% IF CAN_user_tools_manage_patron_lists %]
81 <label for="add_to_patron_list">Add selected patrons to:</label>
82 <select id="add_to_patron_list" name="add_to_patron_list">
83 <option value=""></option>
85 <optgroup label="Patron lists:">
86 [% FOREACH pl IN patron_lists %]
87 <option value="[% pl.patron_list_id | html %]">[% pl.name | html %]</option>
92 <option value="new">[ New list ]</option>
95 <input type="text" id="new_patron_list" name="new_patron_list" id="new_patron_list" />
97 <input id="add_to_patron_list_submit" type="submit" class="submit" value="Save">
101 [% IF CAN_user_tools_manage_patron_lists && CAN_user_borrowers_edit_borrowers %]
105 [% IF CAN_user_borrowers_edit_borrowers %]
106 <button id="merge-patrons" type="submit">Merge selected patrons</button>
112 <table id="memberresultst">
115 [% IF CAN_user_borrowers_edit_borrowers || CAN_user_tools_manage_patron_lists %]
120 <th>Date of birth</th>
124 <th>OD/Checkouts</th>
135 </div> <!-- /.col-sm-10.col-sm-push-2 -->
137 <div class="col-sm-2 col-sm-pull-10">
139 <form method="get" id="searchform">
140 <input type="hidden" id="firstletter_filter" value="" />
141 <fieldset class="brief">
142 <h3>Search patrons</h3>
145 <label for="searchmember_filter">Search for:</label>
146 <input type="text" id="searchmember_filter" value="[% searchmember | html %]"/>
149 <label for="searchfieldstype_filter">Search field:</label>
150 <select name="searchfieldstype" id="searchfieldstype_filter">
151 [% pref_fields = Koha.Preference('DefaultPatronSearchFields').split(',') %]
152 [% default_fields = [ 'standard', 'surname', 'cardnumber', 'email', 'borrowernumber', 'userid', 'phone', 'address', 'dateofbirth', 'sort1', 'sort2' ] %]
153 [% search_options = default_fields.merge(pref_fields).unique %]
154 [% FOREACH s_o IN search_options %]
155 [% display_name = PROCESS patron_fields name=s_o %]
156 [% NEXT IF !display_name %]
157 [% IF searchfieldstype == s_o %]
158 <option selected="selected" value=[% s_o | html %]>[% display_name | $raw %]</option>
160 <option value=[% s_o | html %]>[% display_name | $raw %]</option>
166 <label for="searchtype_filter">Search type:</label>
167 <select name="searchtype" id="searchtype_filter">
168 [% IF searchtype == "start_with" %]
169 <option value='start_with' selected="selected">Starts with</option>
170 <option value="contain">Contains</option>
172 <option value='start_with'>Starts with</option>
173 <option value="contain" selected="selected">Contains</option>
178 <label for="categorycode_filter">Patron category:</label>
179 [% SET categories = Categories.all() %]
180 <select id="categorycode_filter">
181 <option value="">Any</option>
182 [% FOREACH cat IN categories %]
183 [% IF cat.categorycode == categorycode_filter %]
184 <option selected="selected" value="[% cat.categorycode | html %]">[% cat.description | html %]</option>
186 <option value="[% cat.categorycode | html %]">[% cat.description | html %]</option>
193 <label for="branchcode_filter">Library:</label>
194 [% SET branches = Branches.all( selected => branchcode_filter, only_from_group => 1 ) %]
195 <select id="branchcode_filter">
196 [% IF branches.size != 1 %]
197 <option value="">Any</option>
199 [% PROCESS options_for_libraries libraries => branches %]
203 <fieldset class="action">
204 <input type="submit" value="Search" />
205 <input type="button" value="Clear" id="clear_search" />
210 </div> <!-- /.col-sm-2.col-sm-pull-10 -->
211 </div> <!-- /.row -->
213 [% MACRO jsinclude BLOCK %]
214 [% INCLUDE 'datatables.inc' %]
215 [% INCLUDE 'columns_settings.inc' %]
216 [% INCLUDE 'str/members-menu.inc' %]
217 [% Asset.js("lib/hc-sticky.js") | $raw %]
218 [% Asset.js("js/members-menu.js") | $raw %]
220 var singleBranchMode = '[% singleBranchMode | html %]';
221 $(document).ready(function() {
222 $('#merge-patrons').prop('disabled', true);
223 $('#memberresultst').on('change', 'input.selection', function() {
224 if ( $('.selection:checked').length > 1 ) {
225 $('#merge-patrons').prop('disabled', false);
227 $('#merge-patrons').prop('disabled', true);
230 $('#merge-patrons').on('click', function() {
231 var merge_patrons_url = 'merge-patrons.pl?' + $('.selection:checked')
233 return "id=" + $(this).val()
236 window.location.href = merge_patrons_url;
239 $('#add_to_patron_list_submit').prop('disabled', true);
240 $('#new_patron_list').hide();
242 $('#add_to_patron_list').change(function() {
243 var value = $('#add_to_patron_list').val();
244 if ( value == 'new' ) {
245 $('#new_patron_list').val('')
246 $('#new_patron_list').show();
247 $('#new_patron_list').focus();
248 } else if ( value ) {
249 $('#new_patron_list').hide();
250 $('#add_to_patron_list_submit').prop('disabled', false);
252 $('#new_patron_list').hide();
253 $('#add_to_patron_list_submit').prop('disabled', true);
257 $('#new_patron_list').on('input', function() {
258 if ( $('#new_patron_list').val() ) {
259 $('#add_to_patron_list_submit').prop('disabled', false);
261 $('#add_to_patron_list_submit').prop('disabled', true);
265 $("#add_to_patron_list_submit").on('click', function(e){
266 if ( $('#add_to_patron_list').val() == 'new' ) {
267 if ( $('#new_patron_list').val() ) {
268 $("#add_to_patron_list option").each(function() {
269 if ( $(this).text() == $('#new_patron_list').val() ) {
270 alert( _("You already have a list with that name!") );
275 alert( _("You must give your new patron list a name!") );
280 if ( $("#memberresultst input:checkbox:checked").length == 0 ) {
281 alert( _("You have not selected any patrons to add to a list!") );
285 var borrowernumbers = [];
286 $("#memberresultst").find("input:checkbox:checked").each(function(){
287 borrowernumbers.push($(this).val());
290 add_to_patron_list: $("#add_to_patron_list").val(),
291 new_patron_list: $("#new_patron_list").val(),
292 borrowernumbers: borrowernumbers
297 url: '/cgi-bin/koha/svc/members/add_to_list',
298 success: function(data) {
299 $("#patron_list_dialog").show();
300 $("#patron_list_dialog > span.patrons-length").html(data.patrons_added_to_list);
301 $("#patron_list_dialog > a").attr("href", "/cgi-bin/koha/patron_lists/list.pl?patron_list_id=" + data.patron_list.patron_list_id);
302 $("#patron_list_dialog > a").html(data.patron_list.name);
303 if ( $('#add_to_patron_list').val() == 'new' ) {
304 var new_patron_list_added = $("<option>", {
305 value: data.patron_list.patron_list_id,
306 text: data.patron_list.name
308 $("#add_to_patron_list optgroup").append(new_patron_list_added);
309 $("#add_to_patron_list").val(data.patron_list.patron_list_id);
310 $("#new_patron_list").val('');
311 $('#add_to_patron_list').change();
315 alert("an error occurred");
320 $(".filterByLetter").on("click",function(e){
322 filterByFirstLetterSurname($(this).text());
324 $("#select_all").on("click",function(e){
326 $(".selection").prop("checked", true).change();
328 $("#clear_all").on("click",function(e){
330 $(".selection").prop("checked", false).change();
332 $("#clear_search").on("click",function(e){
336 $("#searchform").on("submit", filter);
342 $(document).ready(function() {
343 [% IF searchmember %]
344 $("#searchmember_filter").val("[% searchmember | html %]");
346 [% IF searchfieldstype %]
347 $("searchfieldstype_filter").val("[% searchfieldstype | html %]");
350 $("#searchtype_filter").val("[% searchtype | html %]");
352 [% IF categorycode %]
353 $("#categorycode_filter").val("[% categorycode_filter | html %]");
356 $("#branchcode_filter").val("[% branchcode_filter | html %]");
359 [% IF view != "show_results" %]
362 $("#searchresults").show();
365 // Build the aLengthMenu
367 [% PatronsPerPage | html %], 10, 20, 50, 100, -1
369 jQuery.unique(aLengthMenu);
370 aLengthMenu.sort(function( a, b ){
371 // Put "All" at the end
374 } else if ( b == -1 ) {
377 return parseInt(a) < parseInt(b) ? -1 : 1;}
379 var aLengthMenuLabel = [];
380 $(aLengthMenu).each(function(){
382 // Label for -1 is "All"
383 aLengthMenuLabel.push(_("All"));
385 aLengthMenuLabel.push(this);
389 // Apply DataTables on the results table
390 var columns_settings = [% TablesSettings.GetColumns( 'members', 'member', 'memberresultst', 'json' ) | $raw %];
391 [% UNLESS CAN_user_borrowers_edit_borrowers OR CAN_user_tools_manage_patron_lists %]
392 [%# Remove the first column if we do not display the checkbox %]
393 columns_settings.splice(0, 1);
395 dtMemberResults = KohaTable("memberresultst", {
397 'sAjaxSource': "/cgi-bin/koha/svc/members/search",
398 'fnServerData': function(sSource, aoData, fnCallback) {
403 'name': 'searchmember',
404 'value': $("#searchmember_filter").val()
406 'name': 'firstletter',
407 'value': $("#firstletter_filter").val()
409 'name': 'searchfieldstype',
410 'value': $("#searchfieldstype_filter").val()
412 'name': 'searchtype',
413 'value': $("#searchtype_filter").val()
415 'name': 'categorycode',
416 'value': $("#categorycode_filter").val()
418 'name': 'branchcode',
419 'value': $("#branchcode_filter").val()
421 'name': 'name_sorton',
422 'value': 'borrowers.surname borrowers.firstname'
424 'name': 'cardnumber_sorton',
425 'value': 'borrowers.cardnumber',
427 'name': 'dateofbirth_sorton',
428 'value': 'borrowers.dateofbirth',
430 'name': 'dateexpiry_sorton',
431 'value': 'borrowers.dateexpiry',
433 'name': 'category_sorton',
434 'value': 'categories.description',
436 'name': 'branch_sorton',
437 'value': 'branches.branchname'
439 'name': 'borrowernotes_sorton',
440 'value': 'borrowers.borrowernotes'
442 'name': 'template_path',
443 'value': 'members/tables/members_results.tt',
450 'success': function(json){
451 // redirect if there is only 1 result.
452 if ( json.aaData.length == 1 ) {
453 var borrowernumber = json.aaData[0].borrowernumber;
454 document.location.href="/cgi-bin/koha/members/moremember.pl?borrowernumber="+borrowernumber;
462 [% IF CAN_user_borrowers_edit_borrowers || CAN_user_tools_manage_patron_lists %]
463 { 'mDataProp': 'dt_borrowernumber', 'bSortable': false },
465 { 'mDataProp': 'dt_cardnumber' },
466 { 'mDataProp': 'dt_name' },
467 { 'mDataProp': 'dt_dateofbirth' },
468 { 'mDataProp': 'dt_category' },
470 'mDataProp': function ( oObj ) {
471 if( !singleBranchMode && oObj.dt_branch == "[% Branches.GetLoggedInBranchname | html %]" ){
472 return "<span class=\"currentlibrary\">" + oObj.dt_branch + "</span>";
474 return oObj.dt_branch;
478 { 'mDataProp': 'dt_dateexpiry' },
479 { 'mDataProp': 'dt_od_checkouts', 'bSortable': false },
480 { 'mDataProp': 'dt_fines', 'bSortable': false },
481 { 'mDataProp': 'dt_borrowernotes' },
482 { 'mDataProp': 'dt_action', 'bSortable': false, 'sClass': 'actions' }
486 [% IF CAN_user_borrowers_edit_borrowers || CAN_user_tools_manage_patron_lists %]
487 'aaSorting': [[2, 'asc']],
489 'aaSorting': [[1, 'asc']],
491 "aLengthMenu": [aLengthMenu, aLengthMenuLabel],
492 'sPaginationType': 'full_numbers',
493 "iDisplayLength": [% PatronsPerPage | html %],
495 "initComplete": function(settings, json) {
496 Sticky = $("#searchheader");
498 stickTo: "#searchresults",
499 stickyClass: "floating"
502 }, columns_settings);
506 // Update the string "Results found ..."
507 function update_searched(){
508 var searched = $("#searchfieldstype_filter").find("option:selected").text();
509 if ( $("#searchmember_filter").val() ) {
510 if ( $("#searchtype_filter").val() == 'start_with' ) {
511 searched += _(" starting with ");
513 searched += _(" containing ");
515 searched += "'" + $("#searchmember_filter").val() + "'";
517 if ( $("#firstletter_filter").val() ) {
518 searched += _(" begins with ") + "'" + $("#firstletter_filter").val() +"'";
520 if ( $("#categorycode_filter").val() ) {
521 searched += _(" with category ") + "'" + $("#categorycode_filter").find("option:selected").text() + "'";
523 if ( $("#branchcode_filter").val() ) {
524 searched += _(" in library ") + $("#branchcode_filter").find("option:selected").text();
526 $("#searchpattern").text(searched);
531 $("#firstletter_filter").val('');
534 $("#searchresults").show();
535 dtMemberResults.fnDraw();
539 // User has clicked on the Clear button
540 function clearFilters(redraw) {
541 $("#searchform select").val('');
542 $("#firstletter_filter").val('');
543 $("#searchmember_filter").val('');
546 $("#searchresults").show();
547 dtMemberResults.fnDraw();
551 // User has clicked on a letter
552 function filterByFirstLetterSurname(letter) {
554 $("#firstletter_filter").val(letter);
557 $("#searchresults").show();
558 dtMemberResults.fnDraw();
563 [% INCLUDE 'intranet-bottom.inc' %]