X-Git-Url: http://git.freeside.biz/gitweb/?a=blobdiff_plain;f=FS%2FFS%2Fquotation.pm;h=19be00670bf0a53c8a94ab7076e04e3cb6b60212;hb=4386d67b6b49d2915be38ccfefae474e95849611;hp=ccaa1c34b5eda80deb8eb7451e673743e8aa6417;hpb=493be9f0635db1e24783e5c3945b8bef39624674;p=freeside.git diff --git a/FS/FS/quotation.pm b/FS/FS/quotation.pm index ccaa1c34b..19be00670 100644 --- a/FS/FS/quotation.pm +++ b/FS/FS/quotation.pm @@ -2,9 +2,12 @@ package FS::quotation; use base qw( FS::Template_Mixin FS::cust_main_Mixin FS::otaker_Mixin FS::Record ); use strict; +use Tie::RefHash; +use FS::UID qw( dbh ); use FS::Record qw( qsearch qsearchs ); use FS::CurrentUser; use FS::cust_main; +use FS::cust_pkg; use FS::prospect_main; use FS::quotation_pkg; @@ -117,6 +120,10 @@ sub check { $self->usernum($FS::CurrentUser::CurrentUser->usernum) unless $self->usernum; + return 'prospectnum or custnum must be specified' + if ! $self->prospectnum + && ! $self->custnum; + $self->SUPER::check; } @@ -142,13 +149,69 @@ sub cust_main { =cut -sub cust_bill_pkg { +sub cust_bill_pkg { #actually quotation_pkg objects my $self = shift; - #actually quotation_pkg objects qsearch('quotation_pkg', { quotationnum=>$self->quotationnum }); } -=back +=item total_setup + +=cut + +sub total_setup { + my $self = shift; + $self->_total('setup'); +} + +=item total_recur [ FREQ ] + +=cut + +sub total_recur { + my $self = shift; +#=item total_recur [ FREQ ] + #my $freq = @_ ? shift : ''; + $self->_total('recur'); +} + +sub _total { + my( $self, $method ) = @_; + + my $total = 0; + $total += $_->$method() for $self->cust_bill_pkg; + sprintf('%.2f', $total); + +} + +#prevent things from falsely showing up as taxes, at least until we support +# quoting tax amounts.. +sub _items_tax { + return (); +} +sub _items_nontax { + shift->cust_bill_pkg; +} + +sub _items_total { + my( $self, $total_items ) = @_; + + if ( $self->total_setup > 0 ) { + push @$total_items, { + 'total_item' => $self->mt( $self->total_recur > 0 ? 'Total Setup' : 'Total' ), + 'total_amount' => $self->total_setup, + }; + } + + #could/should add up the different recurring frequencies on lines of their own + # but this will cover the 95% cases for now + if ( $self->total_recur > 0 ) { + push @$total_items, { + 'total_item' => $self->mt('Total Recurring'), + 'total_amount' => $self->total_recur, + }; + } + +} =item enable_previous @@ -156,6 +219,208 @@ sub cust_bill_pkg { sub enable_previous { 0 } +=item convert_cust_main + +If this quotation already belongs to a customer, then returns that customer, as +an FS::cust_main object. + +Otherwise, creates a new customer (FS::cust_main object and record, and +associated) based on this quotation's prospect, then orders this quotation's +packages as real packages for the customer. + +If there is an error, returns an error message, otherwise, returns the +newly-created FS::cust_main object. + +=cut + +sub convert_cust_main { + my $self = shift; + + my $cust_main = $self->cust_main; + return $cust_main if $cust_main; #already converted, don't again + + my $oldAutoCommit = $FS::UID::AutoCommit; + local $FS::UID::AutoCommit = 0; + my $dbh = dbh; + + $cust_main = $self->prospect_main->convert_cust_main; + unless ( ref($cust_main) ) { # eq 'FS::cust_main' ) { + $dbh->rollback if $oldAutoCommit; + return $cust_main; + } + + $self->prospectnum(''); + $self->custnum( $cust_main->custnum ); + my $error = $self->replace || $self->order; + if ( $error ) { + $dbh->rollback if $oldAutoCommit; + return $error; + } + + $dbh->commit or die $dbh->errstr if $oldAutoCommit; + + $cust_main; + +} + +=item order + +This method is for use with quotations which are already associated with a customer. + +Orders this quotation's packages as real packages for the customer. + +If there is an error, returns an error message, otherwise returns false. + +=cut + +sub order { + my $self = shift; + + tie my %cust_pkg, 'Tie::RefHash', + map { FS::cust_pkg->new({ pkgpart => $_->pkgpart, + quantity => $_->quantity, + }) + => [] #services + } + $self->quotation_pkg ; + + $self->cust_main->order_pkgs( \%cust_pkg ); + +} + +=item quotation_pkg + +=cut + +sub quotation_pkg { + my $self = shift; + qsearch('quotation_pkg', { 'quotationnum' => $self->quotationnum } ); +} + +=back + +=head1 CLASS METHODS + +=over 4 + + +=item search_sql_where HASHREF + +Class method which returns an SQL WHERE fragment to search for parameters +specified in HASHREF. Valid parameters are + +=over 4 + +=item _date + +List reference of start date, end date, as UNIX timestamps. + +=item invnum_min + +=item invnum_max + +=item agentnum + +=item charged + +List reference of charged limits (exclusive). + +=item owed + +List reference of charged limits (exclusive). + +=item open + +flag, return open invoices only + +=item net + +flag, return net invoices only + +=item days + +=item newest_percust + +=back + +Note: validates all passed-in data; i.e. safe to use with unchecked CGI params. + +=cut + +sub search_sql_where { + my($class, $param) = @_; + #if ( $DEBUG ) { + # warn "$me search_sql_where called with params: \n". + # join("\n", map { " $_: ". $param->{$_} } keys %$param ). "\n"; + #} + + my @search = (); + + #agentnum + if ( $param->{'agentnum'} =~ /^(\d+)$/ ) { + push @search, "( prospect_main.agentnum = $1 OR cust_main.agentnum = $1 )"; + } + +# #refnum +# if ( $param->{'refnum'} =~ /^(\d+)$/ ) { +# push @search, "cust_main.refnum = $1"; +# } + + #prospectnum + if ( $param->{'prospectnum'} =~ /^(\d+)$/ ) { + push @search, "quotation.prospectnum = $1"; + } + + #custnum + if ( $param->{'custnum'} =~ /^(\d+)$/ ) { + push @search, "cust_bill.custnum = $1"; + } + + #_date + if ( $param->{_date} ) { + my($beginning, $ending) = @{$param->{_date}}; + + push @search, "quotation._date >= $beginning", + "quotation._date < $ending"; + } + + #quotationnum + if ( $param->{'quotationnum_min'} =~ /^(\d+)$/ ) { + push @search, "quotation.quotationnum >= $1"; + } + if ( $param->{'quotationnum_max'} =~ /^(\d+)$/ ) { + push @search, "quotation.quotationnum <= $1"; + } + +# #charged +# if ( $param->{charged} ) { +# my @charged = ref($param->{charged}) +# ? @{ $param->{charged} } +# : ($param->{charged}); +# +# push @search, map { s/^charged/cust_bill.charged/; $_; } +# @charged; +# } + + my $owed_sql = FS::cust_bill->owed_sql; + + #days + push @search, "quotation._date < ". (time-86400*$param->{'days'}) + if $param->{'days'}; + + #agent virtualization + my $curuser = $FS::CurrentUser::CurrentUser; + #false laziness w/search/quotation.html + push @search,' ( '. $curuser->agentnums_sql( table=>'prospect_main' ). + ' OR '. $curuser->agentnums_sql( table=>'cust_main' ). + ' ) '; + + join(' AND ', @search ); + +} + +=back + =head1 BUGS =head1 SEE ALSO