1 <% include('/elements/header.html',
4 ' onUnload="myclose()"'
7 <% include('/elements/init_overlib.html') %>
9 <% include('/elements/error.html') %>
11 <FORM NAME="topform" STYLE="margin-bottom: 0">
12 <INPUT TYPE="hidden" NAME="custnum" VALUE="<% $custnum %>">
15 Customer #<B><% $cust_main->display_custnum %></B> -
16 <B><FONT COLOR="#<% $cust_main->statuscolor %>">
17 <% ucfirst($cust_main->status) %>
22 <% &ntable("#cccccc") %>
25 <% include('/elements/tr-select-agent.html',
26 'curr_value' => $cust_main->agentnum,
27 'label' => "<B>${r}Agent</B>",
28 'empty_label' => 'Select agent',
29 'disable_empty' => ( $cust_main->agentnum ? 1 : 0 ),
34 % if ( $conf->exists('cust_main-edit_agent_custid') ) {
37 <TD ALIGN="right">Customer identifier</TD>
38 <TD><INPUT TYPE="text" NAME="agent_custid" VALUE="<% $cust_main->agent_custid %>"></TD>
43 <INPUT TYPE="hidden" NAME="agent_custid" VALUE="<% $cust_main->agent_custid %>">
47 %# referral (advertising source)
48 %my $refnum = $cust_main->refnum || $conf->config('referraldefault') || 0;
49 %if ( $custnum && ! $conf->exists('editreferrals') ) {
51 <INPUT TYPE="hidden" NAME="refnum" VALUE="<% $refnum %>">
55 <% include('/elements/tr-select-part_referral.html',
56 'curr_value' => $refnum
63 %my $referring_cust_main = '';
64 %if ( $cust_main->referral_custnum
65 % and $referring_cust_main =
66 % qsearchs('cust_main', { custnum => $cust_main->referral_custnum } )
70 <TD ALIGN="right">Referring customer</TD>
72 <A HREF="<% popurl(1) %>/cust_main.cgi?<% $cust_main->referral_custnum %>"><% $cust_main->referral_custnum %>: <% $referring_cust_main->name %></A>
75 <INPUT TYPE="hidden" NAME="referral_custnum" VALUE="<% $cust_main->referral_custnum %>">
76 % } elsif ( ! $conf->exists('disable_customer_referrals') ) {
80 <TD ALIGN="right">Referring customer</TD>
82 <!-- <INPUT TYPE="text" NAME="referral_custnum" VALUE=""> -->
83 <% include('/elements/search-cust_main.html',
84 'field_name' => 'referral_custnum',
92 <INPUT TYPE="hidden" NAME="referral_custnum" VALUE="">
100 % if ( $conf->exists('cust_main-enable_birthdate') ) {
103 <% ntable("#cccccc", 2) %>
104 <% include ('/elements/tr-input-date-field.html',
106 $cust_main->birthdate,
108 $conf->config('date_format') || "%m/%d/%Y",
116 <!-- contact info -->
118 % my $same_checked = '';
119 % my $ship_disabled = '';
120 % unless ( $cust_main->ship_last && $same ne 'Y' ) {
121 % $same_checked = 'CHECKED';
122 % $ship_disabled = 'DISABLED STYLE="background-color: #dddddd"';
124 % qw( last first company address1 address2 city county state zip country
125 % daytime night fax )
127 % $cust_main->set("ship_$_", $cust_main->get($_) );
133 <% include('cust_main/contact.html',
134 'cust_main' => $cust_main,
136 'onchange' => 'bill_changed(this)',
139 'stateid' => $stateid,
140 'same_checked' => $same_checked, #for address2 "Unit #" labeling
145 function bill_changed(what) {
146 if ( what.form.same.checked ) {
147 % for (qw( last first company address1 address2 city zip daytime night fax )) {
149 what.form.ship_<%$_%>.value = what.form.<%$_%>.value;
152 what.form.ship_country.selectedIndex = what.form.country.selectedIndex;
154 function fix_ship_county() {
155 what.form.ship_county.selectedIndex = what.form.county.selectedIndex;
158 function fix_ship_state() {
159 what.form.ship_state.selectedIndex = what.form.state.selectedIndex;
160 ship_state_changed(what.form.ship_state, fix_ship_county );
163 ship_country_changed(what.form.ship_country, fix_ship_state );
167 function samechanged(what) {
168 if ( what.checked ) {
171 % for (qw( last first company address1 address2 city county state zip country daytime night fax )) {
172 what.form.ship_<%$_%>.disabled = true;
173 what.form.ship_<%$_%>.style.backgroundColor = '#dddddd';
176 % if ( $conf->exists('cust_main-require_address2') ) {
177 document.getElementById('address2_required').style.visibility = '';
178 document.getElementById('address2_label').style.visibility = '';
179 document.getElementById('ship_address2_required').style.visibility = 'hidden';
180 document.getElementById('ship_address2_label').style.visibility = 'hidden';
185 % for (qw( last first company address1 address2 city county state zip country daytime night fax )) {
186 what.form.ship_<%$_%>.disabled = false;
187 what.form.ship_<%$_%>.style.backgroundColor = '#ffffff';
190 % if ( $conf->exists('cust_main-require_address2') ) {
191 document.getElementById('address2_required').style.visibility = 'hidden';
192 document.getElementById('address2_label').style.visibility = 'hidden';
193 document.getElementById('ship_address2_required').style.visibility = '';
194 document.getElementById('ship_address2_label').style.visibility = '';
203 (<INPUT TYPE="checkbox" NAME="same" VALUE="Y" onClick="samechanged(this)" <%$same_checked%>>same as billing address)
204 <% include('cust_main/contact.html',
205 'cust_main' => $cust_main,
208 'disabled' => $ship_disabled,
213 <!-- billing info -->
215 <% include( 'cust_main/billing.html', $cust_main,
216 'payinfo' => $payinfo,
217 'invoicing_list' => \@invoicing_list,
221 <% include( '/elements/xmlhttp.html',
222 'url' => $p.'misc/xmlhttp-cust_main-address_standardize.html',
223 'subs' => [ 'address_standardize' ],
224 #'method' => 'POST', #could get too long?
229 function bottomfixup(what) {
231 //i don't think we need to copy things between two forms anymore, modern
232 //browsers are fine with DIVs inside FORMs
234 var topvars = new Array(
237 'custnum', 'agentnum', 'agent_custid', 'refnum', 'referral_custnum',
239 'last', 'first', 'ss', 'company',
240 'address1', 'address2', 'city',
241 'county', 'state', 'zip', 'country',
242 'daytime', 'night', 'fax',
243 'stateid', 'stateid_state',
247 'ship_last', 'ship_first', 'ship_company',
248 'ship_address1', 'ship_address2', 'ship_city',
249 'ship_county', 'ship_state', 'ship_zip', 'ship_country',
250 'ship_daytime','ship_night', 'ship_fax',
257 var layervars = new Array(
259 'payinfo', 'payinfo1', 'payinfo2', 'paytype',
260 'payname', 'paystate', 'exp_month', 'exp_year', 'paycvv',
261 'paystart_month', 'paystart_year', 'payissue',
266 var billing_bottomvars = new Array(
268 'invoicing_list', 'invoicing_list_POST', 'invoicing_list_FAX',
274 for ( f=0; f < topvars.length; f++ ) {
275 var field = topvars[f];
276 copyelement( document.topform.elements[field],
277 document.bottomform.elements[field]
281 var layerform = document.topform.select.options[document.topform.select.selectedIndex].value;
282 for ( f=0; f < layervars.length; f++ ) {
283 var field = layervars[f];
284 copyelement( document.forms[layerform].elements[field],
285 document.bottomform.elements[field]
289 for ( f=0; f < billing_bottomvars.length; f++ ) {
290 var field = billing_bottomvars[f];
291 copyelement( document.billing_bottomform.elements[field],
292 document.bottomform.elements[field]
296 //this part does USPS address correction
298 // XXX should this be first and should we update the form fields that are
301 //var state_el = document.bottomform.elements['state'];
303 //address_standardize(
304 var cust_main = new Array(
305 'company', document.bottomform.elements['company'].value,
306 'address1', document.bottomform.elements['address1'].value,
307 'address2', document.bottomform.elements['address2'].value,
308 'city', document.bottomform.elements['city'].value,
309 'state', document.bottomform.elements['state'].value,
310 //'state', state_el.options[ state_el.selectedIndex ].value,
311 'zip', document.bottomform.elements['zip'].value,
313 'ship_company', document.bottomform.elements['ship_company'].value,
314 'ship_address1', document.bottomform.elements['ship_address1'].value,
315 'ship_address2', document.bottomform.elements['ship_address2'].value,
316 'ship_city', document.bottomform.elements['ship_city'].value,
317 'ship_state', document.bottomform.elements['ship_state'].value,
318 //'ship_state', state_el.options[ state_el.selectedIndex ].value,
319 'ship_zip', document.bottomform.elements['ship_zip'].value
322 address_standardize( cust_main, update_address );
326 var standardize_address;
328 function update_address(arg) {
330 var argsHash = eval('(' + arg + ')');
332 var changed = argsHash['address_standardized'];
333 var ship_changed = argsHash['ship_address_standardized'];
334 var error = argsHash['error'];
335 var ship_error = argsHash['ship_error'];
338 standardize_address = function () {
341 document.bottomform.elements['company'].value = argsHash['new_company'];
342 document.bottomform.elements['address1'].value = argsHash['new_address1'];
343 document.bottomform.elements['address2'].value = argsHash['new_address2'];
344 document.bottomform.elements['city'].value = argsHash['new_city'];
345 document.bottomform.elements['state'].value = argsHash['new_state'];
346 //'state', state_el.options[ state_el.selectedIndex ].value,
347 document.bottomform.elements['zip'].value = argsHash['new_zip'];
350 if ( ship_changed ) {
351 document.bottomform.elements['ship_company'].value = argsHash['new_ship_company'];
352 document.bottomform.elements['ship_address1'].value = argsHash['new_ship_address1'];
353 document.bottomform.elements['ship_address2'].value = argsHash['new_ship_address2'];
354 document.bottomform.elements['ship_city'].value = argsHash['new_ship_city'];
355 document.bottomform.elements['ship_state'].value = argsHash['new_ship_state'];
356 //'state', state_el.options[ state_el.selectedIndex ].value,
357 document.bottomform.elements['ship_zip'].value = argsHash['new_ship_zip'];
362 if ( error || ship_error ) {
364 var url = "cust_main/choose_tax_location.html?data_vendor=cch-zip;city="+document.bottomform.elements['city'].value+";state="+document.bottomform.elements['state'].value+";zip="+document.bottomform.elements['zip'].value+";";
366 OLgetAJAX( url, update_geocode, 300 );
369 } else if ( changed || ship_changed ) {
371 % if ( $conf->exists('cust_main-auto_standardize_address') ) {
373 standardize_address();
374 document.bottomform.submit();
378 // popup a confirmation popup
381 '<CENTER><BR><B>Confirm address standardization</B><BR><BR>' +
386 confirm_change = confirm_change +
387 '<TR><TH>Entered billing address</TH>' +
388 '<TH>Standardized billing address</TH></TR>';
389 // + '<TR><TD> </TD><TD> </TD></TR>';
391 if ( argsHash['company'] || argsHash['new_company'] ) {
392 confirm_change = confirm_change +
393 '<TR><TD>' + argsHash['company'] +
394 '</TD><TD>' + argsHash['new_company'] + '</TD></TR>';
397 confirm_change = confirm_change +
398 '<TR><TD>' + argsHash['address1'] +
399 '</TD><TD>' + argsHash['new_address1'] + '</TD></TR>' +
400 '<TR><TD>' + argsHash['address2'] +
401 '</TD><TD>' + argsHash['new_address2'] + '</TD></TR>' +
402 '<TR><TD>' + argsHash['city'] + ', ' + argsHash['state'] + ' ' + argsHash['zip'] +
403 '</TD><TD>' + argsHash['new_city'] + ', ' + argsHash['new_state'] + ' ' + argsHash['new_zip'] + '</TD></TR>' +
404 '<TR><TD> </TD><TD> </TD></TR>';
408 if ( ship_changed ) {
410 confirm_change = confirm_change +
411 '<TR><TH>Entered service address</TH>' +
412 '<TH>Standardized service address</TH></TR>';
413 // + '<TR><TD> </TD><TD> </TD></TR>';
415 if ( argsHash['ship_company'] || argsHash['new_ship_company'] ) {
416 confirm_change = confirm_change +
417 '<TR><TD>' + argsHash['ship_company'] +
418 '</TD><TD>' + argsHash['new_ship_company'] + '</TD></TR>';
421 confirm_change = confirm_change +
422 '<TR><TD>' + argsHash['ship_address1'] +
423 '</TD><TD>' + argsHash['new_ship_address1'] + '</TD></TR>' +
424 '<TR><TD>' + argsHash['ship_address2'] +
425 '</TD><TD>' + argsHash['new_ship_address2'] + '</TD></TR>' +
426 '<TR><TD>' + argsHash['ship_city'] + ', ' + argsHash['ship_state'] + ' ' + argsHash['ship_zip'] +
427 '</TD><TD>' + argsHash['new_ship_city'] + ', ' + argsHash['new_ship_state'] + ' ' + argsHash['new_ship_zip'] + '</TD></TR>' +
428 '<TR><TD> </TD><TD> </TD></TR>';
432 var addresses = 'address';
434 if ( changed && ship_changed ) {
435 addresses = 'addresses';
436 height = 396; // #what
439 confirm_change = confirm_change +
441 '<BUTTON TYPE="button" onClick="document.bottomform.submit();"><IMG SRC="<%$p%>images/error.png" ALT=""> Use entered ' + addresses + '</BUTTON>' +
443 '<BUTTON TYPE="button" onClick="standardize_address(); document.bottomform.submit();"><IMG SRC="<%$p%>images/tick.png" ALT=""> Use standardized ' + addresses + '</BUTTON>' +
445 '<TR><TD COLSPAN=2 ALIGN="center">' +
446 '<BUTTON TYPE="button" onClick="document.bottomform.submitButton.disabled=false; parent.cClick();"><IMG SRC="<%$p%>images/cross.png" ALT=""> Cancel submission</BUTTON></TD></TR>' +
450 overlib( confirm_change, CAPTION, 'Confirm address standardization', STICKY, AUTOSTATUSCAP, CLOSETEXT, '', MIDX, 0, MIDY, 0, DRAGGABLE, WIDTH, 576, HEIGHT, height, BGCOLOR, '#333399', CGCOLOR, '#333399', TEXTSIZE, 3 );
456 document.bottomform.submit();
462 function update_geocode() {
465 set_geocode = function (what) {
467 //alert(what.options[what.selectedIndex].value);
468 var argsHash = eval('(' + what.options[what.selectedIndex].value + ')');
469 document.bottomform.elements['city'].value = argsHash['city'];
470 document.bottomform.elements['state'].value = argsHash['state'];
471 document.bottomform.elements['zip'].value = argsHash['zip'];
472 document.bottomform.elements['geocode'].value = argsHash['geocode'];
478 overlib( OLresponseAJAX, CAPTION, 'Select tax location', STICKY, AUTOSTATUSCAP, CLOSETEXT, '', MIDX, 0, MIDY, 0, DRAGGABLE, WIDTH, 576, HEIGHT, 268, BGCOLOR, '#333399', CGCOLOR, '#333399', TEXTSIZE, 3 );
482 function copyelement(from, to) {
483 if ( from == undefined ) {
485 } else if ( from.type == 'select-one' ) {
486 to.value = from.options[from.selectedIndex].value;
487 //alert(from + " (" + from.type + "): " + to.name + " => (" + from.selectedIndex + ") " + to.value);
488 } else if ( from.type == 'checkbox' ) {
489 if ( from.checked ) {
490 to.value = from.value;
495 if ( from.value == undefined ) {
498 to.value = from.value;
501 //alert(from + " (" + from.type + "): " + to.name + " => " + to.value);
506 <FORM ACTION="<% popurl(1) %>process/cust_main.cgi" METHOD=POST NAME="bottomform" STYLE="margin-top: 0; margin-bottom: 0">
507 % foreach my $hidden (
510 % 'custnum', 'agentnum', 'agent_custid', 'refnum', 'referral_custnum',
511 % 'last', 'first', 'ss', 'company',
512 % 'address1', 'address2', 'city',
513 % 'county', 'state', 'zip', 'country',
514 % 'daytime', 'night', 'fax',
515 % 'stateid', 'stateid_state',
519 % 'ship_last', 'ship_first', 'ship_company',
520 % 'ship_address1', 'ship_address2', 'ship_city',
521 % 'ship_county', 'ship_state', 'ship_zip', 'ship_country',
522 % 'ship_daytime','ship_night', 'ship_fax',
529 % 'payinfo', 'payinfo1', 'payinfo2', 'paytype',
530 % 'payname', 'paystate', 'exp_month', 'exp_year', 'paycvv',
531 % 'paystart_month', 'paystart_year', 'payissue',
536 % 'invoicing_list', 'invoicing_list_POST', 'invoicing_list_FAX',
543 <INPUT TYPE="hidden" NAME="<% $hidden %>" VALUE="">
546 % my $ro_comments = $conf->exists('cust_main-use_comments')?'':'readonly';
547 % if (!$ro_comments || $cust_main->comments) {
550 <% &ntable("#cccccc") %>
553 <TEXTAREA COLS=80 ROWS=5 WRAP="HARD" NAME="comments" <%$ro_comments%>><% $cust_main->comments %></TEXTAREA>
560 %unless ( $custnum ) {
561 % # pry the wrong place for this logic. also pretty expensive
564 % #false laziness, copied from FS::cust_pkg::order
566 % my @agents = $FS::CurrentUser::CurrentUser->agents;
567 % if ( scalar(@agents) == 1 ) {
568 % # $pkgpart->{PKGPART} is true iff $custnum may purchase PKGPART
569 % $pkgpart = $agents[0]->pkgpart_hashref;
571 % #can't know (agent not chosen), so, allow all
573 % foreach my $agent ( @agents ) {
574 % next if $typenum{$agent->typenum}++;
575 % #fixed in 5.004_05 #$pkgpart->{$_}++ foreach keys %{ $agent->pkgpart_hashref }
576 % foreach ( keys %{ $agent->pkgpart_hashref } ) { $pkgpart->{$_}++; } #5.004_04 workaround
581 % my @part_pkg = grep { $_->svcpart('svc_acct') && $pkgpart->{ $_->pkgpart } }
582 % qsearch( 'part_pkg', { 'disabled' => '' }, '', 'ORDER BY pkg' ); # case?
586 % # print "<BR><BR>First package", &itable("#cccccc", "0 ALIGN=LEFT"),
587 % #apiabuse & undesirable wrapping
592 <% ntable("#cccccc") %>
596 <% include('cust_main/select-domain.html',
597 'pkgparts' => \@part_pkg,
598 'saved_pkgpart' => $saved_pkgpart,
599 'saved_domsvc' => $saved_domsvc,
605 % #false laziness: (mostly) copied from edit/svc_acct.cgi
606 % #$ulen = $svc_acct->dbdef_table->column('username')->length;
607 % my $ulen = dbdef->table('svc_acct')->column('username')->length;
608 % my $ulen2 = $ulen+2;
609 % my $passwordmax = $conf->config('passwordmax') || 8;
610 % my $pmax2 = $passwordmax + 2;
615 <TD ALIGN="right">Username</TD>
617 <INPUT TYPE="text" NAME="username" VALUE="<% $username %>" SIZE=<% $ulen2 %> MAXLENGTH=<% $ulen %>>
622 <TD ALIGN="right">Domain</TD>
624 <SELECT NAME="domsvc">
625 <OPTION>(none)</OPTION>
631 <TD ALIGN="right">Password</TD>
633 <INPUT TYPE="text" NAME="_password" VALUE="<% $password %>" SIZE=<% $pmax2 %> MAXLENGTH=<% $passwordmax %>>
639 <TD ALIGN="right">Access number</TD>
640 <TD><% FS::svc_acct_pop::popselector($popnum) %></TD>
647 <INPUT TYPE="hidden" NAME="otaker" VALUE="<% $cust_main->otaker %>">
649 <INPUT TYPE="button" NAME="submitButton" ID="submitButton" VALUE="<% $custnum ? "Apply Changes" : "Add Customer" %>" onClick="document.bottomform.submitButton.disabled=true; bottomfixup(this.form);">
653 <% include('/elements/footer.html') %>
658 unless $FS::CurrentUser::CurrentUser->access_right('Edit customer');
660 #for misplaced logic below
663 #for false laziness below (now more properly lazy)
664 #use FS::svc_acct_pop;
666 #for (other) false laziness below
670 my $conf = new FS::Conf;
674 my($custnum, $username, $password, $popnum, $cust_main, $saved_pkgpart, $saved_domsvc);
676 my ($ss,$stateid,$payinfo);
678 if ( $cgi->param('error') ) {
679 $cust_main = new FS::cust_main ( {
680 map { $_, scalar($cgi->param($_)) } fields('cust_main')
682 $custnum = $cust_main->custnum;
683 $saved_domsvc = $cgi->param('domsvc') || '';
684 if ( $saved_domsvc =~ /^(\d+)$/ ) {
689 $saved_pkgpart = $cgi->param('pkgpart_svcpart') || '';
690 if ( $saved_pkgpart =~ /^(\d+)_/ ) {
695 $username = $cgi->param('username');
696 $password = $cgi->param('_password');
697 $popnum = $cgi->param('popnum');
698 @invoicing_list = split( /\s*,\s*/, $cgi->param('invoicing_list') );
699 $same = $cgi->param('same');
700 $cust_main->setfield('paid' => $cgi->param('paid')) if $cgi->param('paid');
701 $ss = $cust_main->ss; # don't mask an entered value on errors
702 $stateid = $cust_main->stateid; # don't mask an entered value on errors
703 $payinfo = $cust_main->payinfo; # don't mask an entered value on errors
704 } elsif ( $cgi->keywords ) { #editing
705 my( $query ) = $cgi->keywords;
708 $cust_main = qsearchs('cust_main', { 'custnum' => $custnum } );
709 if ( $cust_main->dbdef_table->column('paycvv')
710 && length($cust_main->paycvv) ) {
711 my $paycvv = $cust_main->paycvv;
713 $cust_main->paycvv($paycvv);
720 @invoicing_list = $cust_main->invoicing_list;
721 $ss = $cust_main->masked('ss');
722 $stateid = $cust_main->masked('stateid');
723 $payinfo = $cust_main->paymask;
726 $cust_main = new FS::cust_main ( {} );
727 $cust_main->otaker( &getotaker );
728 $cust_main->referral_custnum( $cgi->param('referral_custnum') );
734 @invoicing_list = ();
735 push @invoicing_list, 'POST'
736 unless $conf->exists('disablepostalinvoicedefault');
742 my $error = $cgi->param('error');
744 $cgi->param('error', $error);
746 my $action = $custnum ? 'Edit' : 'Add';
747 $action .= ": ". $cust_main->name if $custnum;
749 my $r = qq!<font color="#ff0000">*</font> !;