comment
[freeside.git] / httemplate / edit / quick-charge.html
1 <& /elements/header-popup.html, mt('One-time charge'), '',
2             ( $cgi->param('error') ? '' : 'onload="addRow()"' ),
3 &>
4
5 <LINK REL="stylesheet" TYPE="text/css" HREF="<%$fsurl%>elements/calendar-win2k-2.css" TITLE="win2k-2">
6 <SCRIPT TYPE="text/javascript" SRC="<%$fsurl%>elements/calendar_stripped.js"></SCRIPT>
7 <SCRIPT TYPE="text/javascript" SRC="<%$fsurl%>elements/calendar-en.js"></SCRIPT>
8 <SCRIPT TYPE="text/javascript" SRC="<%$fsurl%>elements/calendar-setup.js"></SCRIPT>
9
10 <& /elements/error.html &>
11
12 <SCRIPT TYPE="text/javascript">
13
14 function enable_quick_charge (e) {
15
16   if (    document.QuickChargeForm.amount.value
17        && document.QuickChargeForm.pkg.value    ) {
18     document.QuickChargeForm.submit.disabled = false;
19   } else {
20     document.QuickChargeForm.submit.disabled = true;
21   }
22
23 % if ( $curuser->option('disable_enter_submit_onetimecharge') ) {
24
25     var key;
26     if (window.event)
27           key = window.event.keyCode; //IE
28     else
29
30           key = e.which; //firefox, others
31
32     return (key != 13);
33
34 % } else {
35     return true;
36 % }
37
38 }
39
40 function validate_quick_charge () {
41   var pkg = document.QuickChargeForm.pkg.value;
42   var pkg_regex = /^([\w \!\@\#\$\%\&\(\)\-\+\;\:\'\"\,\.\?\/\=\[\]]*)$/ ;
43   var amount = document.QuickChargeForm.amount.value;
44   var amount_regex = /^\s*\$?\s*(\d*(\.?\d{1,2}))\s*$/ ;
45   var rval = true;
46
47   if ( ! amount_regex.test(amount) ) {
48     alert('Illegal amount - enter an amount to charge, for example, "5" or "43" or "21.46".');
49     return false;
50   }
51   if ( String(pkg).length < 1 ) {
52     rval = false;
53   }
54   if ( ! pkg_regex.test(pkg) ) {
55     rval = false;
56   }
57   var i=0;
58   for (i=0; i < rownum; i++) {
59     if (! eval('pkg_regex.test(document.QuickChargeForm.description' + i + '.value)')){
60       rval = false;
61       break;
62     }
63   }
64   if (rval == true) {
65     return true;
66   }
67
68   if ( ! pkg ) {
69     alert('Enter a description for the one-time charge');
70     return false;
71   }
72
73   alert('Illegal description - spaces, letters, numbers, and the following punctuation characters are allowed: . , ! ? @ # $ % & ( ) - + ; : ' + "'" + ' " = [ ]' );
74   return false;
75 }
76
77 function bill_now_changed (what) {
78   var form = what.form;
79   if ( what.checked ) {
80     form.start_date_text.disabled = true;
81     form.start_date.style.backgroundColor = '#dddddd';
82     form.start_date_button.style.display = 'none';
83     form.start_date_button_disabled.style.display = '';
84     form.invoice_terms.disabled = false;
85   } else {
86     form.start_date_text.disabled = false;
87     form.start_date.style.backgroundColor = '#ffffff';
88     form.start_date_button.style.display = '';
89     form.start_date_button_disabled.style.display = 'none';
90     form.invoice_terms.disabled = true;
91   }
92 }
93
94 </SCRIPT>
95
96 <FORM ACTION   = "process/quick-charge.cgi"
97       NAME     = "QuickChargeForm"
98       ID       = "QuickChargeForm"
99       METHOD   = "POST"
100       onSubmit = "document.QuickChargeForm.submit.disabled=true; return validate_quick_charge();"
101 >
102
103 <INPUT TYPE="hidden" NAME="custnum"     VALUE="<% $cust_main ? $cust_main->custnum : '' %>">
104 <INPUT TYPE="hidden" NAME="prospectnum" VALUE="<% $prospect_main ? $prospect_main->prospectnum : '' %>">
105 <INPUT TYPE="hidden" NAME="quotationnum" VALUE="<% $quotationnum %>">
106
107 <TABLE ID="QuickChargeTable" BGCOLOR="#cccccc" BORDER=0 CELLSPACING=0 STYLE="background-color: #cccccc">
108
109 % if ( $cust_pkg ) { #modify one-time charge
110
111 <INPUT TYPE="hidden" NAME="pkgnum" VALUE="<% $cust_pkg->pkgnum %>">
112 % my $field = '/elements/tr-input-text.html';
113 % # don't allow changing these after the fact
114 % $field = '/elements/tr-fixed.html' if $billed;
115 <& $field,
116   label  => mt('Amount to charge'),
117   field  => 'amount',
118   value  => sprintf('%.2f',$part_pkg->option('setup_fee')),
119   size   => 8,
120   prefix => $money_char,
121 &> 
122
123 % if ( $curuser->access_right('Edit package definition costs') ) {
124   <& $field,
125     label  => mt('Cost'),
126     field  => 'setup_cost',
127     value  => sprintf('%.2f',$part_pkg->setup_cost),
128     size   => 8,
129     prefix => $money_char,
130   &> 
131 % }
132
133 %   if ( $conf->exists('invoice-unitprice') ) {
134 <& $field,
135   label => 'Quantity',
136   field => 'quantity',
137   value => $cust_pkg->quantity
138 &>
139 %   }
140
141 <& /elements/tr-select-pkg_class.html, 'curr_value' => $classnum  &>
142
143 % # crudely estimate whether any agent commission credits might exist
144 %   my @events = grep { $_->part_event->action =~ /credit/ }
145 %                $cust_pkg->cust_event;
146 %   if ( scalar @events ) {
147 <TR><TD></TD>
148   <TD><INPUT TYPE="checkbox" NAME="adjust_commission" VALUE="Y" CHECKED>
149 <% emt('Adjust commission credits if necessary') %>
150 </TD>
151 </TR>
152 %   }
153
154 % #display the future or past charge date, but don't allow changes
155 % # XXX we probably _could_ let as-yet unbilled charges be rescheduled, but
156 % # there's no compelling need yet
157 %   if ( $billed ) {
158       <& /elements/tr-fixed-date.html,
159         label => emt('Billed on'),
160         value => $cust_pkg->get('setup')
161       &>
162 %   } else {
163       <& /elements/tr-input-date-field.html,
164         {
165           name    => 'start_date',
166           label   => emt('Will be billed'),
167           value   => $cust_pkg->get('start_date'),
168           format  => $date_format,
169           noinit  => 1,
170         }
171       &>
172 %   }
173
174 % } else { # new one-time charge
175
176     <TR>
177       <TD ALIGN="right"><% mt('Amount to charge') |h %> </TD>
178       <TD>
179         <% $money_char %><INPUT TYPE       = "text"
180                                 NAME       = "amount"
181                                 SIZE       = 8
182                                 VALUE      = "<% $amount %>"
183                                 onChange   = "return enable_quick_charge(event)"
184                                 onKeyPress = "return enable_quick_charge(event)"
185                          >
186       </TD>
187     </TR>
188
189 %   if ( $curuser->access_right('Edit package definition costs') ) {
190       <& /elements/tr-input-text.html,
191            label  => mt('Cost'),
192            field  => 'setup_cost',
193            value  => $setup_cost,
194            size   => 8,
195            prefix => $money_char,
196       &> 
197 %   }
198
199 %   if ( $conf->exists('invoice-unitprice') ) {
200     <TR>
201       <TD ALIGN="right"><% mt('Quantity') |h %> </TD>
202       <TD>
203         <INPUT TYPE       = "text"
204                NAME       = "quantity"
205                SIZE       = 4
206                VALUE      = "<% $quantity %>"
207                onKeyPress = "return enable_quick_charge(event)">
208       </TD>
209     </TR>
210 %   }
211
212 <& /elements/tr-select-pkg_class.html, 'curr_value' => $classnum  &>
213
214 % unless ( $quotationnum ) {
215
216     <TR>
217       <TD ALIGN="right"><% mt('Invoice now') |h %></TD>
218       <TD>
219         <INPUT TYPE  = "checkbox"
220                NAME  = "bill_now"
221                VALUE = "1"
222                <% $cgi->param('bill_now') ? 'CHECKED' : '' %>
223                onClick  = "bill_now_changed(this);"
224                onChange = "bill_now_changed(this);"
225         >
226         <% mt('with terms') |h %> 
227         <& /elements/select-terms.html,
228              'curr_value' => scalar($cgi->param('invoice_terms')),
229              'disabled'   => ( $cgi->param('bill_now') ? 0 : 1 ),
230              'agentnum'   => $cust_or_prospect->agentnum,
231         &>
232       </TD>
233     </TR>
234
235 %   # false laziness w/misc/order_pkg.html
236     <TR>
237       <TD ALIGN="right"><% mt('Charge date') |h %> </TD>
238       <TD>
239         <INPUT TYPE  = "text"
240                NAME  = "start_date"
241                SIZE  = 32
242                ID    = "start_date_text"
243                VALUE = "<% $start_date %>"
244                onKeyPress="return enable_quick_charge(event)"
245                <% $cgi->param('bill_now')
246                     ? 'STYLE = "background-color:#dddddd" DISABLED'
247                     : ''
248                %>
249         >
250         <IMG SRC   = "<%$fsurl%>images/calendar.png"
251              ID    = "start_date_button"
252              TITLE = "<% mt('Select date') |h %>"
253              STYLE = "cursor:pointer<% $cgi->param('bill_now') ? ';display:none' : '' %>"
254         >
255         <IMG SRC   = "<%$fsurl%>images/calendar-disabled.png"
256              ID    = "start_date_button_disabled"
257              <% $cgi->param('bill_now') ? '' : 'STYLE="display:none"' %>
258         >
259         <FONT SIZE=-1>(<% mt('leave blank to charge immediately') |h %>)</FONT>
260       </TD>
261     </TR>
262
263     <SCRIPT TYPE="text/javascript">
264       Calendar.setup({
265         inputField: "start_date_text",
266         ifFormat:   "<% $date_format %>",
267         button:     "start_date_button",
268         align:      "BR"
269       });
270     </SCRIPT>
271
272 % }
273
274 % if ( ! $quotationnum && $cust_main->payby =~ /^(CARD|CHEK)$/ ) {
275 %   my $what = lc(FS::payby->shortname($cust_main->payby));
276     <TR>
277       <TD ALIGN="right"><% mt("Disable automatic $what charge") |h %> </TD>
278       <TD COLSPAN=6><INPUT TYPE="checkbox" NAME="no_auto" VALUE="Y"></TD>
279     </TR>
280 % }
281
282 <TR>
283   <TD ALIGN="right"><% mt('Tax exempt') |h %> </TD>
284   <TD><INPUT TYPE="checkbox" NAME="setuptax" VALUE="Y" <% $cgi->param('setuptax') ? 'CHECKED' : '' %>></TD>
285 </TR>
286
287 <& /elements/tr-select-taxclass.html, 'curr_value' => $cgi->param('taxclass')  &>
288
289 <& /elements/tr-select-taxproduct.html, 'label' => emt('Tax product'), 'onclick' => 'parent.taxproductmagic(this);', 'curr_value' => $cgi->param('taxproductnum')  &>
290
291 <& /elements/tr-select-taxoverride.html, 'onclick' => 'parent.taxoverridemagic(this);', 'curr_value' => $cgi->param('tax_override')  &>
292
293 % } # if !$cust_pkg
294
295 <TR>
296   <TD ALIGN="right"><% mt('Description') |h %> </TD>
297   <TD>
298     <INPUT TYPE       = "text"
299            NAME       = "pkg"
300            SIZE       = "50"
301            MAXLENGTH  = "50"
302            VALUE      = "<% $pkg %>"
303            onChange   = "return enable_quick_charge(event)"
304            onKeyPress = "return enable_quick_charge(event)"
305     >
306   </TD>
307 </TR>
308
309 <TR>
310   <TD></TD>
311   <TD><FONT SIZE="-1"><% mt('Optional additional description (also printed on invoice):') |h %> </FONT></TD>
312 </TR>
313
314 % my $row = 0;
315 % foreach (@description) {
316     <TR>
317       <TD></TD>
318       <TD>
319         <INPUT TYPE       = "text"
320                NAME       = "description<% $row %>"
321                SIZE       = "60"
322                MAXLENGTH  = "65"
323                VALUE      = "<% $_ |h %>"
324                rownum     = "<% $row %>"
325                onKeyPress = "return enable_quick_charge(event)"
326                onKeyUp    = "return possiblyAddRow(event)"
327         >
328       </TD>
329     </TR>
330 % $row++;
331 % } 
332
333
334 </TABLE>
335
336 <BR>
337 % my $label = $cust_pkg
338 %             ? emt('Modify one-time charge')
339 %             : emt('Add one-time charge');
340 <INPUT TYPE="submit" ID="submit" NAME="submit" VALUE="<% $label %>" \
341 <% ($cgi->param('error') || $cust_pkg) ? '' :' DISABLED' %>>
342
343 </FORM>
344
345
346 <SCRIPT TYPE="text/javascript">
347
348   var rownum = <% $row %>;
349
350   function possiblyAddRow(e) {
351
352     if ( ( rownum - this.getAttribute('rownum') ) == 1 ) {
353       addRow();
354     }
355
356 %   if ( $curuser->option('disable_enter_submit_onetimecharge') ) {
357
358       var key;
359       if (window.event)
360             key = window.event.keyCode; //IE
361       else
362             key = e.which; //firefox, others
363
364       return (key != 13);
365
366 %   } else {
367       return true;
368 %   }
369
370   }
371
372   function addRow() {
373
374     var table = document.getElementById('QuickChargeTable');
375     var tablebody = table.getElementsByTagName('tbody').item(0);
376
377     var row = document.createElement('TR');
378
379     var empty_cell = document.createElement('TD');
380     row.appendChild(empty_cell);
381
382     var description_cell = document.createElement('TD');
383
384       //var description_input = document.createElement('INPUT');
385       var di = document.createElement('INPUT');
386       di.setAttribute('name', 'description'+rownum);
387       di.setAttribute('id',   'description'+rownum);
388       di.setAttribute('size', 60);
389       di.setAttribute('maxLength', 65);
390       di.setAttribute('rownum',   rownum);
391       di.onkeyup = possiblyAddRow;
392       di.onkeypress = enable_quick_charge;
393       description_cell.appendChild(di);
394
395     row.appendChild(description_cell);
396
397     tablebody.appendChild(row);
398
399     rownum++;
400
401   }
402
403 </SCRIPT>
404
405 </BODY>
406 </HTML>
407 <%init>
408
409 my $curuser = $FS::CurrentUser::CurrentUser;
410
411 die "access denied"
412   unless $curuser->access_right('One-time charge');
413
414 my $conf = new FS::Conf;
415 my $date_format = $conf->config('date_format') || '%m/%d/%Y';
416 my $money_char = $conf->config('money_char') || '$';
417
418 my( $cust_main, $cust_pkg, $prospect_main, $quotationnum ) = ( '', '', '', '' );
419 if ( $cgi->param('change_pkgnum') ) {
420   # change an existing one-time charge
421   die "access denied"
422     unless $curuser->access_right('Modify one-time charge');
423
424   $cgi->param('change_pkgnum') =~ /^(\d+)$/ or die "illegal pkgnum";
425   $cust_pkg = FS::cust_pkg->by_key($1) or die "pkgnum $1 not found";
426   $cust_main = $cust_pkg->cust_main;
427 } else {
428   if ( $cgi->param('quotationnum') =~ /^(\d+)$/ ) {
429     $quotationnum = $1;
430   }
431   if ( $cgi->param('custnum') =~ /^(\d+)$/ ) {
432     $cust_main = FS::cust_main->by_key($1) or die "custnum $1 not found";
433   }
434   if ( $cgi->param('prospectnum') =~ /^(\d+)$/ ) {
435     $prospect_main = FS::prospect_main->by_key($1) or die "prospectnum $1 not found";
436   }
437   die "custnum or prospectnum must be specified"
438     unless $cust_main || $prospect_main;
439 }
440
441 my $cust_or_prospect = $cust_main || $prospect_main;
442
443 if ( $cust_main ) {
444   my $custnum = $cust_main->custnum;
445   # agent-virt
446   if (!exists($curuser->agentnums_href->{$cust_main->agentnum})) {
447     die "custnum $custnum not found";
448   }
449 } elsif ( $prospect_main ) {
450   my $prospectnum = $prospect_main->prospectnum;
451   # agent-virt
452   if (!exists($curuser->agentnums_href->{$prospect_main->agentnum})) {
453     die "prospectnum $prospectnum not found";
454   }
455 }
456
457 my $format = "%m/%d/%Y %T %z (%Z)"; #false laziness w/REAL_cust_pkg.cgi?
458 my $start_date = $cust_main ? $cust_main->next_bill_date : '';
459 $start_date = $start_date ? time2str($format, $start_date) : '';
460
461 my $amount = '';
462 if ( $cgi->param('amount') =~ /^\s*\$?\s*(\d+(\.\d{1,2})?)\s*$/ ) {
463   $amount = $1;
464 }
465
466 my $setup_cost = '';
467 if ( $cgi->param('setup_cost') =~ /^\s*\$?\s*(\d+(\.\d{1,2})?)\s*$/ ) {
468   $setup_cost = $1;
469 }
470
471 my $quantity = 1;
472 if ( $cgi->param('quantity') =~ /^\s*(\d+)\s*$/ ) {
473   $quantity = $1;
474 }
475
476 $cgi->param('pkg') =~ /^([\w \!\@\#\$\%\&\(\)\-\+\;\:\'\"\,\.\?\/\=\[\]]*)$/ 
477   or die 'illegal description';
478 my $pkg = $1;
479
480 my $default_terms;
481 if ( $cust_main && $cust_main->invoice_terms ) {
482   $default_terms = emt("Customer default ([_1])", $cust_main->invoice_terms);
483 } else {
484   $default_terms =
485     emt( "Default ([_1])",
486          ( $conf->config('invoice_default_terms', $cust_or_prospect->agentnum)
487              || emt('Payable upon receipt')
488          )
489        );
490 }
491
492 my @description;
493 my %param = $cgi->Vars;
494 for (my $i = 0; exists($param{"description$i"}); $i++) {
495   push @description, $param{"description$i"};
496 }
497
498 my $classnum;
499 if ( $cgi->param('classnum') =~ /^(\d+)$/ ) {
500   $classnum = $1;
501 }
502
503 my $part_pkg;
504 my $billed = 0;
505
506 if ( $cust_pkg ) { # set defaults
507   $part_pkg = $cust_pkg->part_pkg;
508   $pkg ||= $part_pkg->pkg;
509   $classnum ||= $part_pkg->classnum;
510   if (!@description) {
511     for (my $i = 0; $i < ($part_pkg->option('additional_count',1) || 0); $i++) 
512     {
513       push @description, $part_pkg->option("additional_info$i",1);
514     }
515   }
516   $billed = $cust_pkg->get('setup') ? 1 : 0;
517 }
518
519 </%init>