Bug 22359: Prevent undercollection at the server
[koha.git] / koha-tmpl / intranet-tmpl / prog / en / modules / members / paycollect.tt
1 [% USE raw %]
2 [% USE Asset %]
3 [% USE Koha %]
4 [% USE Branches %]
5 [% USE Price %]
6 [% SET footerjs = 1 %]
7 [% PROCESS 'payments.inc' %]
8 [% INCLUDE 'doc-head-open.inc' %]
9 <title>Koha &rsaquo; Patrons &rsaquo;
10     [% IF type == 'WRITEOFF' %]
11         Write off an amount for [% patron.firstname | html %] [% patron.surname | html %]
12     [% ELSE %]
13         Collect fine payment for [% patron.firstname | html %] [% patron.surname | html %]
14     [% END %]
15 </title>
16 [% INCLUDE 'doc-head-close.inc' %]
17 </head>
18
19 <body id="pat_paycollect" class="pat">
20 [% INCLUDE 'header.inc' %]
21 [% INCLUDE 'patron-search.inc' %]
22 <div id="breadcrumbs">
23     <a href="/cgi-bin/koha/mainpage.pl">Home</a> &rsaquo;
24     <a href="/cgi-bin/koha/members/members-home.pl">Patrons</a> &rsaquo;
25     <a href="/cgi-bin/koha/members/pay.pl?borrowernumber=[% patron.borrowernumber | uri %]">Pay fines for [% patron.firstname | html %] [% patron.surname | html %]</a> &rsaquo;
26     [% IF    ( pay_individual )      %]Pay an individual fine
27     [% ELSIF ( writeoff_individual ) %]Write off an individual fine
28     [% ELSE %]
29         [% IF ( selected_accts ) %]
30             [% IF type == 'writeoff' %]Write off an amount toward selected fines
31             [% ELSE                  %]Pay an amount toward selected fines
32             [% END %]
33         [% ELSE                      %]Pay an amount toward all fines
34         [% END %]
35     [% END %]
36 </div>
37
38 <div class="main container-fluid">
39     <div class="row">
40         <div class="col-sm-10 col-sm-push-2">
41             <main>
42
43 [% INCLUDE 'members-toolbar.inc' borrowernumber=patron.borrowernumber %]
44
45
46 <!-- The manual invoice and credit buttons -->
47 <div class="statictabs">
48 <ul>
49     <li>
50     <a href="/cgi-bin/koha/members/boraccount.pl?borrowernumber=[% patron.borrowernumber | uri %]">Transactions</a>
51     </li>
52     <li class="active">
53     <a href="/cgi-bin/koha/members/pay.pl?borrowernumber=[% patron.borrowernumber | uri %]" >Make a payment</a>
54     </li>
55     <li>
56     <a href="/cgi-bin/koha/members/maninvoice.pl?borrowernumber=[% patron.borrowernumber | uri %]" >Create manual invoice</a>
57     </li>
58     <li>
59     <a href="/cgi-bin/koha/members/mancredit.pl?borrowernumber=[% patron.borrowernumber | uri %]" >Create manual credit</a>
60     </li>
61 </ul>
62 <div class="tabs-container">
63 [% IF ( error_over ) %]
64     <div id="error_message" class="dialog alert">
65     You must pay a value less than or equal to [% total_due | format('%.2f') %].
66     </div>
67 [% END %]
68 [% IF ( error_under ) %]
69     <div id="error_message" class="dialog alert">
70     You must collect a value greater than or equal to [% total_paid | format('%.2f') %].
71     </div>
72 [% END %]
73
74 [% IF ( pay_individual ) %]
75     [% IF ( error_registers ) %]
76     <div id="error_message" class="dialog alert">
77         You must have at least one cash register associated with this branch before you can record payments.
78     </div>
79     [% ELSE %]
80
81     <form name="payindivfine" id="payindivfine" method="post" action="/cgi-bin/koha/members/paycollect.pl">
82     <input type="hidden" name="csrf_token" value="[% csrf_token | html %]" />
83     <input type="hidden" name="borrowernumber" id="borrowernumber" value="[% patron.borrowernumber | html %]" />
84     <input type="hidden" name="pay_individual" id="pay_individual" value="[% pay_individual | html %]" />
85     <input type="hidden" name="itemnumber" id="itemnumber" value="[% itemnumber | html %]" />
86     <input type="hidden" name="description" id="description" value="[% description | html %]" />
87     <input type="hidden" name="debit_type_code" id="debit_type_code" value="[% debit_type_code | html %]" />
88     <input type="hidden" name="amount" id="amount" value="[% amount | html %]" />
89     <input type="hidden" name="amountoutstanding" id="amountoutstanding" value="[% amountoutstanding | html %]" />
90     <input type="hidden" name="accountlines_id" id="accountlines_id" value="[% accountlines_id | html %]" />
91     <input type="hidden" name="title" id="title" value="[% title | html %]" />
92     <input type="hidden" name="change_given" id="change_given" />
93
94 <fieldset class="rows">
95     <legend>Pay an individual fine</legend>
96     <input type="hidden" name="payment_note" id="payment_note" value="[% payment_note | html %]" />
97     <table>
98     <thead><tr>
99             <th>Description</th>
100             <th>Account type</th>
101             <th>Amount</th>
102             <th>Amount outstanding</th>
103         </tr></thead>
104     <tfoot>
105         <tr><td colspan="3">Total amount payable:</td><td>[% amountoutstanding | format('%.2f') %]</td></tr>
106     </tfoot>
107     <tbody><tr>
108             <td>
109                 [% individual_description | html %]
110             </td>
111             <td>[% debit_type_code | html %]</td>
112             <td class="debit">[% amount | format('%.2f') %]</td>
113             <td class="debit">[% amountoutstanding | format('%.2f') %]</td>
114         </tr></tbody>
115 </table>
116
117 <ol>
118
119     <li>
120         <label for="paid">Amount being paid: </label>
121         <input name="paid" id="paid" type="text" step="0.01" value="[% amountoutstanding | $Price on_editing => 1 %]"/>
122     </li>
123     <li>
124         <label for="collected">Collected from patron: </label>
125         <input name="collected" id="collected" type="text" step="0.01" value="[% amountoutstanding | $Price on_editing => 1 %]"/>
126     </li>
127     <li>
128         <label>Change to give: </label>
129         <span id="change">0.00</span>
130     </li>
131
132     [% PROCESS account_payment_types %]
133
134     [% IF Koha.Preference('UseCashRegisters') %]
135     <li>
136         <label for="cash_register">Cash register: </label>
137         <select name="cash_register" id="cash_register">
138             [% FOREACH register IN registers %]
139               [% IF register.id == registerid %]
140             <option value="[% register.id | html %]" selected="selected">[% register.name | html %]</option>
141               [% ELSE %]
142             <option value="[% register.id | html %]">[% register.name | html %]</option>
143               [% END %]
144             [% END %]
145         </select>
146     </li>
147     [% END %]
148 </ol>
149 </fieldset>
150
151         <div class="action">
152             <input type="submit" name="submitbutton" value="Confirm" />
153             <a class="cancel" href="/cgi-bin/koha/members/pay.pl?borrowernumber=[% patron.borrowernumber | html %]">Cancel</a>
154         </div>
155     </form>
156     [% END %]
157 [% ELSIF ( writeoff_individual ) %]
158     <form name="woindivfine" id="woindivfine" action="/cgi-bin/koha/members/pay.pl" method="post" >
159     <input type="hidden" name="csrf_token" value="[% csrf_token | html %]" />
160     <fieldset class="rows">
161     <legend>Write off an individual fine</legend>
162     <input type="hidden" name="borrowernumber" id="borrowernumber" value="[% patron.borrowernumber | html %]" />
163     <input type="hidden" name="pay_individual" id="pay_individual" value="[% pay_individual | html %]" />
164     <input type="hidden" name="itemnumber" id="itemnumber" value="[% itemnumber | html %]" />
165     <input type="hidden" name="description" id="description" value="[% description | html %]" />
166     <input type="hidden" name="debit_type_code" id="debit_type_code" value="[% debit_type_code | html %]" />
167     <input type="hidden" name="amount" id="amount" value="[% amount | html %]" />
168     <input type="hidden" name="accountlines_id" id="accountlines_id" value="[% accountlines_id | html %]" />
169     <input type="hidden" name="title" id="title" value="[% title | html %]" />
170     <input type="hidden" name="payment_note" id="payment_note" value="[% payment_note | html %]" />
171     <input type="hidden" name="amountoutstanding" id="amountoutstanding" value="[% amountoutstanding | html %]" />
172     <input type="hidden" name="confirm_writeoff" id="confirm_writeoff" value="1" />
173     <input type="hidden" name="change_given" id="change_given" />
174     <table>
175     <thead><tr>
176             <th>Description</th>
177             <th>Account type</th>
178             <th>Amount</th>
179             <th>Amount outstanding</th>
180         </tr></thead>
181     <tfoot><tr><td colspan="3">Total amount outstanding:</td><td>[% amountoutstanding | format('%.2f') %]</td></tr></tfoot>
182     <tbody><tr>
183             <td>[% individual_description | html %]</td>
184             <td>[% debit_type_code | html %]</td>
185             <td class="debit">[% amount | format('%.2f') %]</td>
186             <td class="debit">[% amountoutstanding | format('%.2f') %]</td>
187         </tr></tbody>
188     </table>
189
190             <ol>
191                 <li>
192                     <label for="paid">Writeoff amount: </label>
193                     <!-- default to writing off all -->
194                     <input name="amountwrittenoff" id="amountwrittenoff" value="[% amountoutstanding | $Price on_editing => 1 %]" type="text" />
195                 </li>
196             </ol>
197         </fieldset>
198         <div class="action">
199             <input type="submit" value="Write off this charge" />
200             <a class="cancel" href="/cgi-bin/koha/members/pay.pl?borrowernumber=[% patron.borrowernumber | html %]">Cancel</a>
201         </div>
202     </form>
203 [% ELSE %]
204     [% IF ( error_registers && type != 'writeoff' ) %]
205     <div id="error_message" class="dialog alert">
206         You must have at least one cash register associated with this branch before you can record payments.
207     </div>
208     [% ELSE %]
209
210     <form name="payfine" id="payfine" method="post" action="/cgi-bin/koha/members/paycollect.pl">
211     <input type="hidden" name="csrf_token" value="[% csrf_token | html %]" />
212     <input type="hidden" name="borrowernumber" id="borrowernumber" value="[% patron.borrowernumber | html %]" />
213     <input type="hidden" name="selected_accts" id="selected_accts" value="[% selected_accts | html %]" />
214     <input type="hidden" name="total" id="total" value="[% total | html %]" />
215     <input type="hidden" name="type" value="[% type | html %]" />
216     <input type="hidden" name="change_given" id="change_given" />
217
218     <fieldset class="rows">
219     [% IF ( selected_accts ) %]
220         [% IF type == 'WRITEOFF' %]
221             <legend>Write off an amount toward selected fines</legend>
222         [% ELSE %]
223             <legend>Pay an amount toward selected fines</legend>
224         [% END %]
225     [% ELSE %]
226         <legend>Pay an amount toward all fines</legend>
227     [% END %]
228
229     <ol>
230         <li>
231             <span class="label">Total amount outstanding: </span>
232             <span class="debit">[% total | format('%.2f') %]</span>
233         </li>
234     <li>
235         [% IF type == 'WRITEOFF' %]
236             <label for="paid">Writeoff amount: </label>
237         [% ELSE %]
238             <label for="paid">Amount being paid: </label>
239         [% END %]
240         <input name="paid" id="paid" type="text" step="0.01" value="[% total | $Price on_editing => 1 %]"/>
241     </li>
242     [% IF type != 'WRITEOFF' %]
243         <li>
244             <label for="collected">Collected from patron: </label>
245             <input name="collected" id="collected" type="text" step="0.01" value="[% total | $Price on_editing => 1 %]"/>
246         </li>
247         <li>
248             <label>Change to give: </label>
249             <span id="change">0.00</span>
250         </li>
251
252     [% PROCESS account_payment_types %]
253
254     [% IF Koha.Preference('UseCashRegisters') %]
255     <li>
256         <label for="cash_register">Cash register: </label>
257         <select name="cash_register" id="cash_register">
258             [% FOREACH register IN registers %]
259               [% IF register.id == registerid %]
260             <option value="[% register.id | html %]" selected="selected">[% register.name | html %]</option>
261               [% ELSE %]
262             <option value="[% register.id | html %]">[% register.name | html %]</option>
263               [% END %]
264             [% END %]
265         </select>
266     </li>
267     [% END %]
268     [% END %]
269
270     <li>
271         <label for="selected_accts_notes">Note: </label>
272         <textarea name="selected_accts_notes" id="selected_accts_notes">[% selected_accts_notes | html %]</textarea>
273     </li>
274     </ol>
275     </fieldset>
276     <div class="action">
277         <input type="submit" name="submitbutton" value="Confirm" />
278         <a class="cancel" href="/cgi-bin/koha/members/pay.pl?borrowernumber=[% patron.borrowernumber | html %]">Cancel</a>
279     </div>
280     </form>
281     [% END %]
282 [% END %]
283 </div></div>
284
285             </main>
286         </div> <!-- /.col-sm-10.col-sm-push-2 -->
287
288         <div class="col-sm-2 col-sm-pull-10">
289             <aside>
290                 [% INCLUDE 'circ-menu.inc' %]
291             </aside>
292         </div> <!-- /.col-sm-2.col-sm-pull-10 -->
293      </div> <!-- /.row -->
294
295 <!-- Modal -->
296 <div id="confirm_change_form" class="modal" tabindex="-1" role="dialog" aria-hidden="true">
297     <div class="modal-dialog">
298         <div class="modal-content">
299             <div class="modal-header">
300                 <h3>The amount collected is more than the outstanding charge</h3>
301             </div>
302             <div class="modal-body">
303                 <p>The amount collected from the patron is higher than the amount to be paid.</p>
304                 <p>The change to give is <b><span id="modal_change">0.00</span></b>.</p>
305                 <p>Confirm this payment?</p>
306             </div>
307             <div class="modal-footer">
308                 <button class="btn btn-default approve" id="modal_submit" type="button"><i class="fa fa-check"></i> Yes</button>
309                 <button class="btn btn-default deny cancel" href="#" data-dismiss="modal" aria-hidden="true"><i class="fa fa-times"></i> No</button>
310             </div>
311         </div>
312     </div>
313 </div>
314
315 [% MACRO jsinclude BLOCK %]
316     [% INCLUDE 'str/members-menu.inc' %]
317     [% Asset.js("js/members-menu.js") | $raw %]
318     <script>
319         $(document).ready(function() {
320             [% IF payment_id && Koha.Preference('FinePaymentAutoPopup') %]
321                 window.open('/cgi-bin/koha/members/printfeercpt.pl?action=print&accountlines_id=[% payment_id | html %]&change_given=[% change_given | html %]&borrowernumber=[% patron.borrowernumber | html %]', '_blank');
322             [% END %]
323
324             var forms = $('#payindivfine, #payfine');
325             var change = $('#change')[0];
326
327             $('#payindivfine, #payfine').preventDoubleFormSubmit();
328             $("#paid, #collected").on("change",function() {
329                 moneyFormat( this );
330                 if (change != undefined) {
331                     updateChangeValues();
332                 }
333             });
334
335             if (change != undefined) {
336                 forms.on("submit", function(e) {
337                     if (change.innerHTML > 0.00) {
338                         e.preventDefault();
339                         $("#confirm_change_form").modal("show");
340                     } else {
341                         return true;
342                     }
343                 });
344             }
345
346             $("#confirm_change_form").on("hidden.bs.modal", function(){
347                 // remove class added by preventDoubleFormSubmit if necessary
348                 $("body, form input[type='submit'], form button[type='submit'], form a").removeClass('waiting');
349             });
350
351             $('#modal_submit').click(function() {
352                 forms[0].submit();
353             });
354
355             $( "#payindivfine, #payfine" ).validate({
356                 rules: {
357                     paid: { required: true },
358                     collected: {
359                         required: true
360                     }
361                 }
362             });
363         });
364
365         prevent_default = 1;
366         $('#woindivfine').on('submit', function(e){
367             if ( prevent_default ) {
368                 e.preventDefault();
369
370                 let amount_outstanding = parseFloat( $('#amountoutstanding').attr('value') );
371                 let amount_writeoff = parseFloat( $('#amountwrittenoff').attr('value') );
372                 if ( amount_writeoff > amount_outstanding ) {
373                     alert(_("You are attemping to writeoff more than the value of the fee."));
374                     $('#woindivfine').beenSubmitted = false;
375                 } else {
376                     prevent_default = 0;
377                     $('#woindivfine').preventDoubleFormSubmit();
378                     $('#woindivfine').submit();
379                 }
380             }
381         });
382
383         function moneyFormat(textObj) {
384             var newValue = textObj.value;
385             var decAmount = "";
386             var dolAmount = "";
387             var decFlag   = false;
388             var aChar     = "";
389
390             for(i=0; i < newValue.length; i++) {
391                 aChar = newValue.substring(i, i+1);
392                 if (aChar >= "0" && aChar <= "9") {
393                     if(decFlag) {
394                         decAmount = "" + decAmount + aChar;
395                     }
396                     else {
397                         dolAmount = "" + dolAmount + aChar;
398                     }
399                 }
400                 if (aChar == ".") {
401                     if (decFlag) {
402                         dolAmount = "";
403                         break;
404                     }
405                     decFlag = true;
406                 }
407             }
408
409             if (dolAmount == "") {
410                 dolAmount = "0";
411             }
412         // Strip leading 0s
413             if (dolAmount.length > 1) {
414                 while(dolAmount.length > 1 && dolAmount.substring(0,1) == "0") {
415                     dolAmount = dolAmount.substring(1,dolAmount.length);
416                 }
417             }
418             if (decAmount.length > 2) {
419                 decAmount = decAmount.substring(0,2);
420             }
421         // Pad right side
422             if (decAmount.length == 1) {
423                decAmount = decAmount + "0";
424             }
425             if (decAmount.length == 0) {
426                decAmount = decAmount + "00";
427             }
428
429             textObj.value = dolAmount + "." + decAmount;
430         }
431
432         function updateChangeValues() {
433             var change = $('#change')[0];
434             change.innerHTML = Math.round(($('#collected')[0].value - $('#paid')[0].value) * 100) / 100;
435             if (change.innerHTML <= 0) {
436                 var paid = $('#paid')[0];
437                 moneyFormat(paid);
438                 $('#collected').rules( "add", { min: Number(paid.value) });
439                 $( "#payindivfine, #payfine" ).valid();
440                 change.innerHTML = "0.00";
441                 $('input[name="change_given"]').val('0.00');
442             } else {
443                 change.value = change.innerHTML;
444                 moneyFormat(change);
445                 change.innerHTML = change.value;
446                 $('input[name="change_given"]').val(change.value);
447             }
448
449             $('#modal_change').html(change.innerHTML);
450         }
451     </script>
452 [% END %]
453
454 [% INCLUDE 'intranet-bottom.inc' %]