3 #script to group (closed) baskets into basket groups for easier order management
4 #written by john.soros@biblibre.com 01/10/2008
6 # Copyright 2008 - 2009 BibLibre SARL
7 # Parts Copyright Catalyst 2010
9 # This file is part of Koha.
11 # Koha is free software; you can redistribute it and/or modify it under the
12 # terms of the GNU General Public License as published by the Free Software
13 # Foundation; either version 2 of the License, or (at your option) any later
16 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
17 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
18 # A PARTICULAR PURPOSE. See the GNU General Public License for more details.
20 # You should have received a copy of the GNU General Public License along
21 # with Koha; if not, write to the Free Software Foundation, Inc.,
22 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
31 This script lets the user group (closed) baskets into basket groups for easier order management. Note that the grouped baskets have to be from the same bookseller and
32 have to be closed to be printed or exported.
40 The bookseller who we want to display the baskets (and basketgroups) of.
55 use C4
::Bookseller qw
/GetBookSellerFromId/;
56 use C4
::Budgets qw
/ConvertCurrency/;
57 use C4
::Acquisition qw
/CloseBasketgroup ReOpenBasketgroup GetOrders GetBasketsByBasketgroup GetBasketsByBookseller ModBasketgroup NewBasketgroup DelBasketgroup GetBasketgroups ModBasket GetBasketgroup GetBasket GetBasketGroupAsCSV/;
58 use C4
::Bookseller qw
/GetBookSellerFromId/;
59 use C4
::Branch qw
/GetBranches/;
60 use C4
::Members qw
/GetMember/;
64 our ($template, $loggedinuser, $cookie)
65 = get_template_and_user
({template_name
=> "acqui/basketgroup.tmpl",
69 flagsrequired
=> {acquisition
=> 'group_manage'},
75 my $bookseller = shift;
77 my @orders = GetOrders
($basketno);
78 for my $order (@orders){
79 $total = $total + ( $order->{ecost
} * $order->{quantity
} );
80 if ($bookseller->{invoiceincgst
} && ! $bookseller->{listincgst
} && ( $bookseller->{gstrate
} // C4
::Context
->preference("gist") )) {
81 my $gst = $bookseller->{gstrate
} // C4
::Context
->preference("gist");
82 $total = $total * ( $gst / 100 +1);
85 $total .= " " . ($bookseller->{invoiceprice
} // 0);
89 #displays all basketgroups and all closed baskets (in their respective groups)
90 sub displaybasketgroups
{
91 my $basketgroups = shift;
92 my $bookseller = shift;
94 if (scalar @
$basketgroups != 0) {
95 foreach my $basketgroup (@
$basketgroups){
98 while($i < scalar(@
$baskets)){
99 my $basket = @
$baskets[$i];
100 if($basket->{'basketgroupid'} && $basket->{'basketgroupid'} == $basketgroup->{'id'}){
101 $basket->{total
} = BasketTotal
($basket->{basketno
}, $bookseller);
102 push(@
{$basketgroup->{'baskets'}}, $basket);
103 splice(@
$baskets, $i, 1);
109 $basketgroup -> {'basketsqty'} = $basketsqty;
111 $template->param(basketgroups
=> $basketgroups);
113 for(my $i=0; $i < scalar @
$baskets; ++$i) {
114 if( ! @
$baskets[$i]->{'closedate'} ) {
115 splice(@
$baskets, $i, 1);
118 @
$baskets[$i]->{total
} = BasketTotal
(@
$baskets[$i]->{basketno
}, $bookseller);
121 $template->param(baskets
=> $baskets);
122 $template->param( booksellername
=> $bookseller ->{'name'});
125 sub printbasketgrouppdf
{
126 my ($basketgroupid) = @_;
128 my $pdfformat = C4
::Context
->preference("OrderPdfFormat");
129 if ($pdfformat eq 'pdfformat::layout3pages' || $pdfformat eq 'pdfformat::layout2pages' || $pdfformat eq 'pdfformat::layout3pagesfr'){
131 eval "require $pdfformat";
138 print $input->header;
139 print $input->start_html; # FIXME Should do a nicer page
140 print "<h1>Invalid PDF Format set</h1>";
141 print "Please go to the systempreferences and set a valid pdfformat";
145 my $basketgroup = GetBasketgroup
($basketgroupid);
146 my $bookseller = GetBookSellerFromId
($basketgroup->{'booksellerid'});
147 my $baskets = GetBasketsByBasketgroup
($basketgroupid);
150 for my $basket (@
$baskets) {
152 my @ords = &GetOrders
($basket->{basketno
});
153 for my $ord (@ords) {
155 next unless ( $ord->{biblionumber
} or $ord->{quantity
}> 0 );
171 $ord->{rrp
} = ConvertCurrency
( $ord->{'currency'}, $ord->{rrp
} );
172 if ( $bookseller->{'listincgst'} ) {
173 $ord->{rrpgsti
} = sprintf( "%.2f", $ord->{rrp
} );
174 $ord->{gstgsti
} = sprintf( "%.2f", $ord->{gstrate
} * 100 );
175 $ord->{rrpgste
} = sprintf( "%.2f", $ord->{rrp
} / ( 1 + ( $ord->{gstgsti} / 100 ) ) );
176 $ord->{gstgste
} = sprintf( "%.2f", $ord->{gstgsti
} / ( 1 + ( $ord->{gstgsti} / 100 ) ) );
177 $ord->{ecostgsti
} = sprintf( "%.2f", $ord->{ecost
} );
178 $ord->{ecostgste
} = sprintf( "%.2f", $ord->{ecost
} / ( 1 + ( $ord->{gstgsti} / 100 ) ) );
179 $ord->{gstvalue
} = sprintf( "%.2f", ( $ord->{ecostgsti
} - $ord->{ecostgste
} ) * $ord->{quantity
});
180 $ord->{totalgste
} = sprintf( "%.2f", $ord->{quantity
} * $ord->{ecostgste
} );
181 $ord->{totalgsti
} = sprintf( "%.2f", $ord->{quantity
} * $ord->{ecostgsti
} );
183 $ord->{rrpgsti
} = sprintf( "%.2f", $ord->{rrp
} * ( 1 + ( $ord->{gstrate
} ) ) );
184 $ord->{rrpgste
} = sprintf( "%.2f", $ord->{rrp
} );
185 $ord->{gstgsti
} = sprintf( "%.2f", $ord->{gstrate
} * 100 );
186 $ord->{gstgste
} = sprintf( "%.2f", $ord->{gstrate
} * 100 );
187 $ord->{ecostgsti
} = sprintf( "%.2f", $ord->{ecost
} * ( 1 + ( $ord->{gstrate
} ) ) );
188 $ord->{ecostgste
} = sprintf( "%.2f", $ord->{ecost
} );
189 $ord->{gstvalue
} = sprintf( "%.2f", ( $ord->{ecostgsti
} - $ord->{ecostgste
} ) * $ord->{quantity
});
190 $ord->{totalgste
} = sprintf( "%.2f", $ord->{quantity
} * $ord->{ecostgste
} );
191 $ord->{totalgsti
} = sprintf( "%.2f", $ord->{quantity
} * $ord->{ecostgsti
} );
193 my $bib = GetBiblioData
($ord->{biblionumber
});
194 my $itemtypes = GetItemTypes
();
197 # 0 1 2 3 4 5 6 7 8 9
198 #isbn, itemtype, author, title, publishercode, quantity, listprice ecost discount gstrate
202 my $marcrecord=eval{MARC
::Record
::new_from_xml
( $ord->{marcxml
},'UTF-8' )};
204 if ( C4
::Context
->preference("marcflavour") eq 'UNIMARC' ) {
205 $en = $marcrecord->subfield( '345', "b" );
206 } elsif ( C4
::Context
->preference("marcflavour") eq 'MARC21' ) {
207 $en = $marcrecord->subfield( '037', "a" );
212 isbn
=> ($ord->{isbn
} ?
$ord->{isbn
} : undef),
213 itemtype
=> ( $ord->{itemtype
} and $bib->{itemtype
} ?
$itemtypes->{$bib->{itemtype
}}->{description
} : undef ),
214 en
=> ( $en ?
$en : undef ),
216 for my $key ( qw
/ gstrate author title itemtype publishercode discount quantity rrpgsti rrpgste gstgsti gstgste ecostgsti ecostgste gstvalue totalgste totalgsti / ) {
217 $ba_order->{$key} = $ord->{$key};
220 push(@ba_orders, $ba_order);
222 $orders{$basket->{basketno
}} = \
@ba_orders;
224 print $input->header(
225 -type
=> 'application/pdf',
226 -attachment
=> ( $basketgroup->{name
} || $basketgroupid ) . '.pdf'
228 my $pdf = printpdf
($basketgroup, $bookseller, $baskets, \
%orders, $bookseller->{gstrate
} // C4
::Context
->preference("gist")) || die "pdf generation failed";
233 my $op = $input->param('op') || 'display';
234 # possible values of $op :
235 # - add : adds a new basketgroup, or edit an open basketgroup, or display a closed basketgroup
236 # - mod_basket : modify an individual basket of the basketgroup
237 # - closeandprint : close and print an closed basketgroup in pdf. called by clicking on "Close and print" button in closed basketgroups list
238 # - print : print a closed basketgroup. called by clicking on "Print" button in closed basketgroups list
239 # - export : export in CSV a closed basketgroup. called by clicking on "Export" button in closed basketgroups list
240 # - delete : delete an open basketgroup. called by clicking on "Delete" button in open basketgroups list
241 # - reopen : reopen a closed basketgroup. called by clicking on "Reopen" button in closed basketgroup list
242 # - attachbasket : save a modified basketgroup, or creates a new basketgroup when a basket is closed. called from basket page
243 # - display : display the list of all basketgroups for a vendor
244 my $booksellerid = $input->param('booksellerid');
245 $template->param(booksellerid
=> $booksellerid);
247 if ( $op eq "add" ) {
249 # if no param('basketgroupid') is not defined, adds a new basketgroup
250 # else, edit (if it is open) or display (if it is close) the basketgroup basketgroupid
251 # the template will know if basketgroup must be displayed or edited, depending on the value of closed key
253 my $bookseller = &GetBookSellerFromId
($booksellerid);
254 my $basketgroupid = $input->param('basketgroupid');
257 my $freedeliveryplace;
258 if ( $basketgroupid ) {
259 # Get the selected baskets in the basketgroup to display them
260 my $selecteds = GetBasketsByBasketgroup
($basketgroupid);
261 foreach my $basket(@
{$selecteds}){
262 $basket->{total
} = BasketTotal
($basket->{basketno
}, $bookseller);
264 $template->param(basketgroupid
=> $basketgroupid,
265 selectedbaskets
=> $selecteds);
267 # Get general informations about the basket group to prefill the form
268 my $basketgroup = GetBasketgroup
($basketgroupid);
270 name
=> $basketgroup->{name
},
271 deliverycomment
=> $basketgroup->{deliverycomment
},
272 freedeliveryplace
=> $basketgroup->{freedeliveryplace
},
274 $billingplace = $basketgroup->{billingplace
};
275 $deliveryplace = $basketgroup->{deliveryplace
};
276 $freedeliveryplace = $basketgroup->{freedeliveryplace
};
277 $template->param( closedbg
=> ($basketgroup ->{'closed'}) ?
1 : 0);
279 $template->param( closedbg
=> 0);
281 # determine default billing and delivery places depending on librarian homebranch and existing basketgroup data
282 my $borrower = GetMember
( ( 'borrowernumber' => $loggedinuser ) );
283 $billingplace = $billingplace || $borrower->{'branchcode'};
284 $deliveryplace = $deliveryplace || $borrower->{'branchcode'};
286 my $branches = C4
::Branch
::GetBranchesLoop
( $billingplace );
287 $template->param( billingplaceloop
=> $branches );
288 $branches = C4
::Branch
::GetBranchesLoop
( $deliveryplace );
289 $template->param( deliveryplaceloop
=> $branches );
290 $template->param( booksellerid
=> $booksellerid );
292 # the template will display a unique basketgroup
293 $template->param(grouping
=> 1);
294 my $basketgroups = &GetBasketgroups
($booksellerid);
295 my $baskets = &GetBasketsByBookseller
($booksellerid);
296 displaybasketgroups
($basketgroups, $bookseller, $baskets);
297 } elsif ($op eq 'mod_basket') {
299 # edit an individual basket contained in this basketgroup
301 my $basketno=$input->param('basketno');
302 my $basketgroupid=$input->param('basketgroupid');
303 ModBasket
( { basketno
=> $basketno,
304 basketgroupid
=> $basketgroupid } );
305 print $input->redirect("basket.pl?basketno=" . $basketno);
306 } elsif ( $op eq 'closeandprint') {
308 # close an open basketgroup and generates a pdf
310 my $basketgroupid = $input->param('basketgroupid');
311 CloseBasketgroup
($basketgroupid);
312 printbasketgrouppdf
($basketgroupid);
314 }elsif ($op eq 'print'){
316 # print a closed basketgroup
318 my $basketgroupid = $input->param('basketgroupid');
319 printbasketgrouppdf
($basketgroupid);
321 }elsif ( $op eq "export" ) {
323 # export a closed basketgroup in csv
325 my $basketgroupid = $input->param('basketgroupid');
326 print $input->header(
328 -attachment
=> 'basketgroup' . $basketgroupid . '.csv',
330 print GetBasketGroupAsCSV
( $basketgroupid, $input );
332 }elsif( $op eq "delete"){
334 # delete an closed basketgroup
336 my $basketgroupid = $input->param('basketgroupid');
337 DelBasketgroup
($basketgroupid);
338 print $input->redirect('/cgi-bin/koha/acqui/basketgroup.pl?booksellerid=' . $booksellerid.'&listclosed=1');
339 }elsif ( $op eq 'reopen'){
341 # reopen a closed basketgroup
343 my $basketgroupid = $input->param('basketgroupid');
344 my $booksellerid = $input->param('booksellerid');
345 ReOpenBasketgroup
($basketgroupid);
346 my $redirectpath = ((defined $input->param('mode'))&& ($input->param('mode') eq 'singlebg')) ?
'/cgi-bin/koha/acqui/basketgroup.pl?op=add&basketgroupid='.$basketgroupid.'&booksellerid='.$booksellerid : '/cgi-bin/koha/acqui/basketgroup.pl?booksellerid=' .$booksellerid.'&listclosed=1';
347 print $input->redirect($redirectpath);
348 } elsif ( $op eq 'attachbasket') {
350 # save a modified basketgroup, or creates a new basketgroup when a basket is closed. called from basket page
353 my $basketgroup = {};
354 my @baskets = $input->param('basket');
355 my $basketgroupid = $input->param('basketgroupid');
356 my $basketgroupname = $input->param('basketgroupname');
357 my $booksellerid = $input->param('booksellerid');
358 my $billingplace = $input->param('billingplace');
359 my $deliveryplace = $input->param('deliveryplace');
360 my $freedeliveryplace = $input->param('freedeliveryplace');
361 my $deliverycomment = $input->param('deliverycomment');
362 my $closedbg = $input->param('closedbg') ?
1 : 0;
363 if ($basketgroupid) {
364 # If we have a basketgroupid we edit the basketgroup
366 name
=> $basketgroupname,
367 id
=> $basketgroupid,
368 basketlist
=> \
@baskets,
369 billingplace
=> $billingplace,
370 deliveryplace
=> $deliveryplace,
371 freedeliveryplace
=> $freedeliveryplace,
372 deliverycomment
=> $deliverycomment,
375 ModBasketgroup
($basketgroup);
380 # we create a new basketgroup (whith a closed basket)
382 name
=> $basketgroupname,
383 booksellerid
=> $booksellerid,
384 basketlist
=> \
@baskets,
385 billingplace
=> $billingplace,
386 deliveryplace
=> $deliveryplace,
387 freedeliveryplace
=> $freedeliveryplace,
388 deliverycomment
=> $deliverycomment,
391 $basketgroupid = NewBasketgroup
($basketgroup);
393 my $redirectpath = ((defined $input->param('mode')) && ($input->param('mode') eq 'singlebg')) ?
'/cgi-bin/koha/acqui/basketgroup.pl?op=add&basketgroupid='.$basketgroupid.'&booksellerid='.$booksellerid : '/cgi-bin/koha/acqui/basketgroup.pl?booksellerid=' . $booksellerid;
394 $redirectpath .= "&listclosed=1" if $closedbg ;
395 print $input->redirect($redirectpath );
398 # no param : display the list of all basketgroups for a given vendor
399 my $basketgroups = &GetBasketgroups
($booksellerid);
400 my $bookseller = &GetBookSellerFromId
($booksellerid);
401 my $baskets = &GetBasketsByBookseller
($booksellerid);
403 displaybasketgroups
($basketgroups, $bookseller, $baskets);
405 $template->param(listclosed
=> ((defined $input->param('listclosed')) && ($input->param('listclosed') eq '1'))?
1:0 );
406 #prolly won't use all these, maybe just use print, the rest can be done inside validate
407 output_html_with_http_headers
$input, $cookie, $template->output;