Bug 18137: List Mojolicious::Plugin::OpenAPI and JSON::Validator as dependencies
[koha.git] / acqui / basketgroup.pl
blob0e8a9a197ab85cfebb16580577b954d175a53ded
1 #!/usr/bin/perl
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
12 # under the terms of the GNU General Public License as published by
13 # the Free Software Foundation; either version 3 of the License, or
14 # (at your option) any later version.
16 # Koha is distributed in the hope that it will be useful, but
17 # WITHOUT ANY WARRANTY; without even the implied warranty of
18 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 # GNU General Public License for more details.
21 # You should have received a copy of the GNU General Public License
22 # along with Koha; if not, see <http://www.gnu.org/licenses>.
25 =head1 NAME
27 basketgroup.pl
29 =head1 DESCRIPTION
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.
34 =head1 CGI PARAMETERS
36 =over 4
38 =item $booksellerid
40 The bookseller who we want to display the baskets (and basketgroups) of.
42 =back
44 =cut
46 use strict;
47 use warnings;
48 use Carp;
50 use C4::Auth;
51 use C4::Output;
52 use CGI qw ( -utf8 );
54 use C4::Acquisition qw/CloseBasketgroup ReOpenBasketgroup GetOrders GetBasketsByBasketgroup GetBasketsByBookseller ModBasketgroup NewBasketgroup DelBasketgroup GetBasketgroups ModBasket GetBasketgroup GetBasket GetBasketGroupAsCSV/;
55 use Koha::EDI qw/create_edi_order get_edifact_ean/;
57 use Koha::Biblioitems;
58 use Koha::Acquisition::Booksellers;
59 use Koha::ItemTypes;
60 use Koha::Patrons;
62 our $input=new CGI;
64 our ($template, $loggedinuser, $cookie)
65 = get_template_and_user({template_name => "acqui/basketgroup.tt",
66 query => $input,
67 type => "intranet",
68 authnotrequired => 0,
69 flagsrequired => {acquisition => 'group_manage'},
70 debug => 1,
71 });
73 sub BasketTotal {
74 my $basketno = shift;
75 my $bookseller = shift;
76 my $total = 0;
77 my @orders = GetOrders($basketno);
78 for my $order (@orders){
79 # FIXME The following is wrong
80 if ( $bookseller->listincgst ) {
81 $total = $total + ( $order->{ecost_tax_included} * $order->{quantity} );
82 } else {
83 $total = $total + ( $order->{ecost_tax_excluded} * $order->{quantity} );
86 $total .= " " . ($bookseller->invoiceprice // 0);
87 return $total;
90 #displays all basketgroups and all closed baskets (in their respective groups)
91 sub displaybasketgroups {
92 my $basketgroups = shift;
93 my $bookseller = shift;
94 my $baskets = shift;
95 if (scalar @$basketgroups != 0) {
96 foreach my $basketgroup (@$basketgroups){
97 my $i = 0;
98 my $basketsqty = 0;
99 while($i < scalar(@$baskets)){
100 my $basket = @$baskets[$i];
101 if($basket->{'basketgroupid'} && $basket->{'basketgroupid'} == $basketgroup->{'id'}){
102 $basket->{total} = BasketTotal($basket->{basketno}, $bookseller);
103 push(@{$basketgroup->{'baskets'}}, $basket);
104 splice(@$baskets, $i, 1);
105 ++$basketsqty;
106 --$i;
108 ++$i;
110 $basketgroup -> {'basketsqty'} = $basketsqty;
112 $template->param(basketgroups => $basketgroups);
114 for(my $i=0; $i < scalar @$baskets; ++$i) {
115 if( ! @$baskets[$i]->{'closedate'} ) {
116 splice(@$baskets, $i, 1);
117 --$i;
118 }else{
119 @$baskets[$i]->{total} = BasketTotal(@$baskets[$i]->{basketno}, $bookseller);
122 $template->param(baskets => $baskets);
123 $template->param( booksellername => $bookseller->name);
126 sub printbasketgrouppdf{
127 my ($basketgroupid) = @_;
129 my $pdfformat = C4::Context->preference("OrderPdfFormat");
130 if ($pdfformat eq 'pdfformat::layout3pages' || $pdfformat eq 'pdfformat::layout2pages' || $pdfformat eq 'pdfformat::layout3pagesfr'
131 || $pdfformat eq 'pdfformat::layout2pagesde'){
132 eval {
133 eval "require $pdfformat";
134 import $pdfformat;
136 if ($@){
139 else {
140 print $input->header;
141 print $input->start_html; # FIXME Should do a nicer page
142 print "<h1>Invalid PDF Format set</h1>";
143 print "Please go to the systempreferences and set a valid pdfformat";
144 exit;
147 my $basketgroup = GetBasketgroup($basketgroupid);
148 my $bookseller = Koha::Acquisition::Booksellers->find( $basketgroup->{booksellerid} );
149 my $baskets = GetBasketsByBasketgroup($basketgroupid);
151 my %orders;
152 for my $basket (@$baskets) {
153 my @ba_orders;
154 my @ords = &GetOrders($basket->{basketno});
155 for my $ord (@ords) {
157 next unless ( $ord->{biblionumber} or $ord->{quantity}> 0 );
158 eval {
159 require C4::Biblio;
160 import C4::Biblio;
162 if ($@){
163 croak $@;
165 eval {
166 require C4::Koha;
167 import C4::Koha;
169 if ($@){
170 croak $@;
173 $ord->{tax_value} = $ord->{tax_value_on_ordering};
174 $ord->{tax_rate} = $ord->{tax_rate_on_ordering};
175 $ord->{total_tax_included} = $ord->{ecost_tax_included} * $ord->{quantity};
176 $ord->{total_tax_excluded} = $ord->{ecost_tax_excluded} * $ord->{quantity};
178 my $biblioitem = Koha::Biblioitems->search({ biblionumber => $ord->{biblionumber} })->next;
180 #FIXME DELETE ME
181 # 0 1 2 3 4 5 6 7 8 9
182 #isbn, itemtype, author, title, publishercode, quantity, listprice ecost discount tax_rate
184 # Editor Number
185 my $en;
186 my $edition;
187 $ord->{marcxml} = C4::Biblio::GetXmlBiblio( $ord->{biblionumber} );
188 my $marcrecord=eval{MARC::Record::new_from_xml( $ord->{marcxml},'UTF-8' )};
189 if ($marcrecord){
190 if ( C4::Context->preference("marcflavour") eq 'UNIMARC' ) {
191 $en = $marcrecord->subfield( '345', "b" );
192 $edition = $marcrecord->subfield( '205', 'a' );
193 } elsif ( C4::Context->preference("marcflavour") eq 'MARC21' ) {
194 $en = $marcrecord->subfield( '037', "a" );
195 $edition = $marcrecord->subfield( '250', 'a' );
199 $ord->{itemtype} = ( $ord->{itemtype} and $biblioitem->itemtype ) ? Koha::ItemTypes->find( $biblioitem->itemtype )->description : undef;
200 $ord->{en} = $en ? $en : undef;
201 $ord->{edition} = $edition ? $edition : undef;
203 push(@ba_orders, $ord);
205 $orders{$basket->{basketno}} = \@ba_orders;
207 print $input->header(
208 -type => 'application/pdf',
209 -attachment => ( $basketgroup->{name} || $basketgroupid ) . '.pdf'
211 my $pdf = printpdf($basketgroup, $bookseller, $baskets, \%orders, $bookseller->tax_rate // C4::Context->preference("gist")) || die "pdf generation failed";
212 print $pdf;
216 sub generate_edifact_orders {
217 my $basketgroupid = shift;
218 my $baskets = GetBasketsByBasketgroup($basketgroupid);
219 my $ean = get_edifact_ean();
221 for my $basket ( @{$baskets} ) {
222 create_edi_order( { ean => $ean, basketno => $basket->{basketno}, } );
224 return;
227 my $op = $input->param('op') || 'display';
228 # possible values of $op :
229 # - add : adds a new basketgroup, or edit an open basketgroup, or display a closed basketgroup
230 # - mod_basket : modify an individual basket of the basketgroup
231 # - closeandprint : close and print an closed basketgroup in pdf. called by clicking on "Close and print" button in closed basketgroups list
232 # - print : print a closed basketgroup. called by clicking on "Print" button in closed basketgroups list
233 # - ediprint : generate edi order messages for the baskets in the group
234 # - export : export in CSV a closed basketgroup. called by clicking on "Export" button in closed basketgroups list
235 # - delete : delete an open basketgroup. called by clicking on "Delete" button in open basketgroups list
236 # - reopen : reopen a closed basketgroup. called by clicking on "Reopen" button in closed basketgroup list
237 # - attachbasket : save a modified basketgroup, or creates a new basketgroup when a basket is closed. called from basket page
238 # - display : display the list of all basketgroups for a vendor
239 my $booksellerid = $input->param('booksellerid');
240 $template->param(booksellerid => $booksellerid);
242 if ( $op eq "add" ) {
244 # if no param('basketgroupid') is not defined, adds a new basketgroup
245 # else, edit (if it is open) or display (if it is close) the basketgroup basketgroupid
246 # the template will know if basketgroup must be displayed or edited, depending on the value of closed key
248 my $bookseller = Koha::Acquisition::Booksellers->find( $booksellerid );
249 my $basketgroupid = $input->param('basketgroupid');
250 my $billingplace;
251 my $deliveryplace;
252 my $freedeliveryplace;
253 if ( $basketgroupid ) {
254 # Get the selected baskets in the basketgroup to display them
255 my $selecteds = GetBasketsByBasketgroup($basketgroupid);
256 foreach my $basket(@{$selecteds}){
257 $basket->{total} = BasketTotal($basket->{basketno}, $bookseller);
259 $template->param(basketgroupid => $basketgroupid,
260 selectedbaskets => $selecteds);
262 # Get general informations about the basket group to prefill the form
263 my $basketgroup = GetBasketgroup($basketgroupid);
264 $template->param(
265 name => $basketgroup->{name},
266 deliverycomment => $basketgroup->{deliverycomment},
267 freedeliveryplace => $basketgroup->{freedeliveryplace},
269 $billingplace = $basketgroup->{billingplace};
270 $deliveryplace = $basketgroup->{deliveryplace};
271 $freedeliveryplace = $basketgroup->{freedeliveryplace};
272 $template->param( closedbg => ($basketgroup ->{'closed'}) ? 1 : 0);
273 } else {
274 $template->param( closedbg => 0);
276 # determine default billing and delivery places depending on librarian homebranch and existing basketgroup data
277 my $patron = Koha::Patrons->find( $loggedinuser ); # FIXME Not needed if billingplace and deliveryplace are set
278 $billingplace = $billingplace || $patron->branchcode;
279 $deliveryplace = $deliveryplace || $patron->branchcode;
281 $template->param( billingplace => $billingplace );
282 $template->param( deliveryplace => $deliveryplace );
283 $template->param( booksellerid => $booksellerid );
285 # the template will display a unique basketgroup
286 $template->param(grouping => 1);
287 my $basketgroups = &GetBasketgroups($booksellerid);
288 my $baskets = &GetBasketsByBookseller($booksellerid);
289 displaybasketgroups($basketgroups, $bookseller, $baskets);
290 } elsif ($op eq 'mod_basket') {
292 # edit an individual basket contained in this basketgroup
294 my $basketno=$input->param('basketno');
295 my $basketgroupid=$input->param('basketgroupid');
296 ModBasket( { basketno => $basketno,
297 basketgroupid => $basketgroupid } );
298 print $input->redirect("basket.pl?basketno=" . $basketno);
299 } elsif ( $op eq 'closeandprint') {
301 # close an open basketgroup and generates a pdf
303 my $basketgroupid = $input->param('basketgroupid');
304 CloseBasketgroup($basketgroupid);
305 printbasketgrouppdf($basketgroupid);
306 exit;
307 }elsif ($op eq 'print'){
309 # print a closed basketgroup
311 my $basketgroupid = $input->param('basketgroupid');
312 printbasketgrouppdf($basketgroupid);
313 exit;
314 }elsif ( $op eq "export" ) {
316 # export a closed basketgroup in csv
318 my $basketgroupid = $input->param('basketgroupid');
319 print $input->header(
320 -type => 'text/csv',
321 -attachment => 'basketgroup' . $basketgroupid . '.csv',
323 print GetBasketGroupAsCSV( $basketgroupid, $input );
324 exit;
325 }elsif( $op eq "delete"){
327 # delete an closed basketgroup
329 my $basketgroupid = $input->param('basketgroupid');
330 DelBasketgroup($basketgroupid);
331 print $input->redirect('/cgi-bin/koha/acqui/basketgroup.pl?booksellerid=' . $booksellerid.'&amp;listclosed=1');
332 }elsif ( $op eq 'reopen'){
334 # reopen a closed basketgroup
336 my $basketgroupid = $input->param('basketgroupid');
337 my $booksellerid = $input->param('booksellerid');
338 ReOpenBasketgroup($basketgroupid);
339 my $redirectpath = ((defined $input->param('mode'))&& ($input->param('mode') eq 'singlebg')) ?'/cgi-bin/koha/acqui/basketgroup.pl?op=add&amp;basketgroupid='.$basketgroupid.'&amp;booksellerid='.$booksellerid : '/cgi-bin/koha/acqui/basketgroup.pl?booksellerid=' .$booksellerid.'&amp;listclosed=1';
340 print $input->redirect($redirectpath);
341 } elsif ( $op eq 'attachbasket') {
343 # save a modified basketgroup, or creates a new basketgroup when a basket is closed. called from basket page
345 # Getting parameters
346 my $basketgroup = {};
347 my @baskets = $input->multi_param('basket');
348 my $basketgroupid = $input->param('basketgroupid');
349 my $basketgroupname = $input->param('basketgroupname');
350 my $booksellerid = $input->param('booksellerid');
351 my $billingplace = $input->param('billingplace');
352 my $deliveryplace = $input->param('deliveryplace');
353 my $freedeliveryplace = $input->param('freedeliveryplace');
354 my $deliverycomment = $input->param('deliverycomment');
355 my $closedbg = $input->param('closedbg') ? 1 : 0;
356 if ($basketgroupid) {
357 # If we have a basketgroupid we edit the basketgroup
358 $basketgroup = {
359 name => $basketgroupname,
360 id => $basketgroupid,
361 basketlist => \@baskets,
362 billingplace => $billingplace,
363 deliveryplace => $deliveryplace,
364 freedeliveryplace => $freedeliveryplace,
365 deliverycomment => $deliverycomment,
366 closed => $closedbg,
368 ModBasketgroup($basketgroup);
369 if($closedbg){
370 # FIXME
372 }else{
373 # we create a new basketgroup (with a closed basket)
374 $basketgroup = {
375 name => $basketgroupname,
376 booksellerid => $booksellerid,
377 basketlist => \@baskets,
378 billingplace => $billingplace,
379 deliveryplace => $deliveryplace,
380 freedeliveryplace => $freedeliveryplace,
381 deliverycomment => $deliverycomment,
382 closed => $closedbg,
384 $basketgroupid = NewBasketgroup($basketgroup);
386 my $redirectpath = ((defined $input->param('mode')) && ($input->param('mode') eq 'singlebg')) ?'/cgi-bin/koha/acqui/basketgroup.pl?op=add&amp;basketgroupid='.$basketgroupid.'&amp;booksellerid='.$booksellerid : '/cgi-bin/koha/acqui/basketgroup.pl?booksellerid=' . $booksellerid;
387 $redirectpath .= "&amp;listclosed=1" if $closedbg ;
388 print $input->redirect($redirectpath );
390 } elsif ( $op eq 'ediprint') {
391 my $basketgroupid = $input->param('basketgroupid');
392 generate_edifact_orders( $basketgroupid );
393 exit;
394 }else{
395 # no param : display the list of all basketgroups for a given vendor
396 my $basketgroups = &GetBasketgroups($booksellerid);
397 my $bookseller = Koha::Acquisition::Booksellers->find( $booksellerid );
398 my $baskets = &GetBasketsByBookseller($booksellerid);
400 displaybasketgroups($basketgroups, $bookseller, $baskets);
402 $template->param(listclosed => ((defined $input->param('listclosed')) && ($input->param('listclosed') eq '1'))? 1:0 );
403 #prolly won't use all these, maybe just use print, the rest can be done inside validate
404 output_html_with_http_headers $input, $cookie, $template->output;