3 # This file is part of Koha.
5 # Koha is free software; you can redistribute it and/or modify it under the
6 # terms of the GNU General Public License as published by the Free Software
9 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
10 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
11 # A PARTICULAR PURPOSE. See the GNU General Public License for more details.
13 # You should have received a copy of the GNU General Public License along
14 # with Koha; if not, write to the Free Software Foundation, Inc.,
15 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
27 use Text::CSV::Encoded;
30 my $dbh = C4::Context->dbh;
32 my ($template, $borrowernumber, $cookie) = get_template_and_user({
33 template_name => "reports/cash_register_stats.tt",
37 flagsrequired => {reports => '*'},
41 my $do_it = $input->param('do_it');
42 my $output = $input->param("output");
43 my $basename = $input->param("basename");
44 my $transaction_type = $input->param("transaction_type") || 'ACT';
45 my $manager_branchcode = $input->param("branch") || C4::Context->userenv->{'branch'};
49 CGIsepChoice => GetDelimiterChoices,
52 #Initialize date pickers to today
53 my $fromDate = dt_from_string;
54 my $toDate = dt_from_string;
56 my $query_manualinv = "SELECT id, authorised_value FROM authorised_values WHERE category = 'MANUAL_INV'";
57 my $sth_manualinv = $dbh->prepare($query_manualinv) or die "Unable to prepare query" . $dbh->errstr;
58 $sth_manualinv->execute() or die "Unable to execute query " . $sth_manualinv->errstr;
59 my $manualinv_types = $sth_manualinv->fetchall_arrayref({});
64 $fromDate = output_pref({ dt => eval { dt_from_string($input->param("from")) } || dt_from_string,
65 dateformat => 'sql', dateonly => 1 }); #for sql query
66 $toDate = output_pref({ dt => eval { dt_from_string($input->param("to")) } || dt_from_string,
67 dateformat => 'sql', dateonly => 1 }); #for sql query
71 if ($transaction_type eq 'ALL') { #All Transactons
73 } elsif ($transaction_type eq 'ACT') { #Active
74 $whereTType = " accounttype NOT IN ('F', 'FU', 'FOR', 'M', 'L') AND ";
75 } else { #Single transac type
76 if ($transaction_type eq 'FORW') {
77 $whereTType = " accounttype = 'FOR' OR accounttype = 'W' AND ";
79 $whereTType = " accounttype = '$transaction_type' AND ";
83 my $whereBranchCode = '';
84 if ($manager_branchcode ne 'ALL') {
85 $whereBranchCode = "AND m.branchcode = '$manager_branchcode'";
90 SELECT round(amount,2) AS amount, description,
91 bo.surname AS bsurname, bo.firstname AS bfirstname, m.surname AS msurname, m.firstname AS mfirstname,
92 bo.cardnumber, br.branchname, bo.borrowernumber,
93 al.borrowernumber, DATE(al.date) as date, al.accounttype, al.amountoutstanding, al.note,
94 bi.title, bi.biblionumber, i.barcode, i.itype
96 LEFT JOIN borrowers bo ON (al.borrowernumber = bo.borrowernumber)
97 LEFT JOIN borrowers m ON (al.manager_id = m.borrowernumber)
98 LEFT JOIN branches br ON (br.branchcode = m.branchcode )
99 LEFT JOIN items i ON (i.itemnumber = al.itemnumber)
100 LEFT JOIN biblio bi ON (bi.biblionumber = i.biblionumber)
102 CAST(al.date AS DATE) BETWEEN ? AND ?
106 my $sth_stats = $dbh->prepare($query) or die "Unable to prepare query" . $dbh->errstr;
107 $sth_stats->execute($fromDate, $toDate) or die "Unable to execute query " . $sth_stats->errstr;
111 while ( my $row = $sth_stats->fetchrow_hashref()) {
112 $row->{amountoutstanding} = 0 if (!$row->{amountoutstanding});
113 #if ((abs($row->{amount}) - $row->{amountoutstanding}) > 0) {
114 $row->{amount} = sprintf("%.2f", abs ($row->{amount}));
115 $row->{date} = dt_from_string($row->{date}, 'sql');
117 push (@loopresult, $row);
118 if($transaction_type eq 'ACT' && ($row->{accounttype} !~ /^C$|^CR$|^LR$|^Pay$/)){
122 if($row->{accounttype} =~ /^C$|^CR$|^LR$/){
123 $grantotal -= abs($row->{amount});
124 $row->{amount} = '-' . $row->{amount};
125 }elsif($row->{accounttype} eq 'FORW' || $row->{accounttype} eq 'W'){
127 $grantotal += abs($row->{amount});
132 $grantotal = sprintf("%.2f", $grantotal);
134 if($output eq 'screen'){
136 loopresult => \@loopresult,
140 binmode STDOUT, ':encoding(UTF-8)';
143 my $reportname = $input->param('basename');
144 my $reportfilename = $reportname ? "$reportname.$format" : "reportresults.$format" ;
145 #my $reportfilename = "$reportname.html" ;
146 my $delimiter = C4::Context->preference('delimiter') || ',';
148 if ( $format eq 'csv' ) {
149 my $csv = Text::CSV::Encoded->new({ encoding_out => 'UTF-8', sep_char => $delimiter});
150 $csv or die "Text::CSV::Encoded->new({binary => 1}) FAILED: " . Text::CSV::Encoded->error_diag();
152 push @headers, "mfirstname",
162 if ($csv->combine(@headers)) {
163 $content .= Encode::decode('UTF-8', $csv->string()) . "\n";
165 push @$q_errors, { combine => 'HEADER ROW: ' . $csv->error_diag() } ;
167 foreach my $row (@loopresult) {
169 push @rowValues, $row->{mfirstname},
179 if ($csv->combine(@rowValues)) {
180 $content .= Encode::decode('UTF-8',$csv->string()) . "\n";
182 push @$q_errors, { combine => $csv->error_diag() } ;
186 print $input->header(
188 -attachment=> $reportfilename
192 print $delimiter x 6;
193 print $grantotal."\n";
194 foreach my $err (@$q_errors) {
195 print "# ERROR: " . (map {$_ . ": " . $err->{$_}} keys %$err) . "\n";
196 } # here we print all the non-fatal errors at the end. Not super smooth, but better than nothing.
203 beginDate => $fromDate,
205 transaction_type => $transaction_type,
206 branchloop => C4::Branch::GetBranchesLoop($manager_branchcode),
207 manualinv_types => $manualinv_types,
208 CGIsepChoice => GetDelimiterChoices,
211 output_html_with_http_headers $input, $cookie, $template->output;