summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMark Wells <mark@freeside.biz>2016-01-22 16:15:38 -0800
committerMark Wells <mark@freeside.biz>2016-01-22 16:25:26 -0800
commit7a331ae6d41521b99a542741c3c92f5edd4ea336 (patch)
tree06aff1fdc2645b76370f67dd3b8b5d65edec1781
parentd8dca5284d56fc6f93074a17af7026bb0732a280 (diff)
optionally round tax to the nearest cent for each line item, #39487
-rw-r--r--FS/FS/Conf.pm7
-rw-r--r--FS/FS/cust_main_county.pm24
2 files changed, 28 insertions, 3 deletions
diff --git a/FS/FS/Conf.pm b/FS/FS/Conf.pm
index c259b7a62..8fac97806 100644
--- a/FS/FS/Conf.pm
+++ b/FS/FS/Conf.pm
@@ -5393,6 +5393,13 @@ and customer address. Include units.',
},
{
+ 'key' => 'tax-round_per_line_item',
+ 'section' => 'billing',
+ 'description' => 'Calculate tax and round to the nearest cent for each line item, rather than for the whole invoice.',
+ 'type' => 'checkbox',
+ },
+
+ {
'key' => 'cust_main-default_view',
'section' => 'UI',
'description' => 'Default customer view, for users who have not selected a default view in their preferences.',
diff --git a/FS/FS/cust_main_county.pm b/FS/FS/cust_main_county.pm
index 523a810d4..e7597c024 100644
--- a/FS/FS/cust_main_county.pm
+++ b/FS/FS/cust_main_county.pm
@@ -275,6 +275,8 @@ sub taxline {
my $taxable_cents = 0;
my $tax_cents = 0;
+ my $round_per_line_item = $conf->exists('tax-round_per_line_item');
+
my $cust_bill = $taxables->[0]->cust_bill;
my $custnum = $cust_bill ? $cust_bill->custnum : $opt{'custnum'};
my $invoice_time = $cust_bill ? $cust_bill->_date : $opt{'invoice_time'};
@@ -450,7 +452,16 @@ sub taxline {
$taxable_charged = sprintf( "%.2f", $taxable_charged);
next if $taxable_charged == 0;
- my $this_tax_cents = int($taxable_charged * $self->tax);
+ my $this_tax_cents = $taxable_charged * $self->tax;
+ if ( $round_per_line_item ) {
+ # Round the tax to the nearest cent for each line item, instead of
+ # across the whole invoice.
+ $this_tax_cents = sprintf('%.0f', $this_tax_cents);
+ } else {
+ # Otherwise truncate it so that rounding error is always positive.
+ $this_tax_cents = int($this_tax_cents);
+ }
+
my $location = FS::cust_bill_pkg_tax_location->new({
'taxnum' => $self->taxnum,
'taxtype' => ref($self),
@@ -465,12 +476,19 @@ sub taxline {
$taxable_cents += $taxable_charged;
$tax_cents += $this_tax_cents;
} #foreach $cust_bill_pkg
-
- # now round and distribute
+
+ # calculate tax and rounding error for the whole group
my $extra_cents = sprintf('%.2f', $taxable_cents * $self->tax / 100) * 100
- $tax_cents;
# make sure we have an integer
$extra_cents = sprintf('%.0f', $extra_cents);
+
+ # if we're rounding per item, then ignore that and don't distribute any
+ # extra cents.
+ if ( $round_per_line_item ) {
+ $extra_cents = 0;
+ }
+
if ( $extra_cents < 0 ) {
die "nonsense extra_cents value $extra_cents";
}