Reverting to previous version, kohastructure.sql file didn't work for me, I got an...
[koha.git] / reserve / request.pl
1 #!/usr/bin/perl
2
3 # $Id$
4
5 #writen 2/1/00 by chris@katipo.oc.nz
6 # Copyright 2000-2002 Katipo Communications
7 #
8 # This file is part of Koha.
9 #
10 # Koha is free software; you can redistribute it and/or modify it under the
11 # terms of the GNU General Public License as published by the Free Software
12 # Foundation; either version 2 of the License, or (at your option) any later
13 # version.
14 #
15 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
16 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
17 # A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
18 #
19 # You should have received a copy of the GNU General Public License along with
20 # Koha; if not, write to the Free Software Foundation, Inc., 59 Temple Place,
21 # Suite 330, Boston, MA  02111-1307 USA
22
23 =head1 request.pl
24
25 script to place reserves/requests
26
27 =cut
28
29 use strict;
30 use C4::Branch; # GetBranches get_branchinfos_of
31 use CGI;
32 use List::MoreUtils qw/uniq/;
33 use Date::Calc qw/Today Date_to_Days/;
34 use C4::Output;
35 use C4::Auth;
36 use C4::Reserves;
37 use C4::Biblio;
38 use C4::Koha;
39 use C4::Circulation;
40 use C4::Date;
41 use C4::Members;
42
43 my $dbh = C4::Context->dbh;
44 my $sth;
45 my $input = new CGI;
46 my ( $template, $borrowernumber, $cookie ) = get_template_and_user(
47     {
48         template_name   => "reserve/request.tmpl",
49         query           => $input,
50         type            => "intranet",
51         authnotrequired => 0,
52         flagsrequired   => { reserveforothers => 1 },
53     }
54 );
55
56 # get biblio information....
57 my $biblionumber = $input->param('biblionumber');
58 my $dat          = GetBiblioData($biblionumber);
59
60 # Select borrowers infos
61 my $findborrower = $input->param('findborrower');
62 $findborrower =~ s|,| |g;
63 my $cardnumber = $input->param('cardnumber');
64 my $borrowerslist;
65 my $messageborrower;
66
67 my $date = sprintf( '%04d-%02d-%02d', Today() );
68
69 if ($findborrower) {
70     my ( $count, $borrowers ) =
71       SearchMember($findborrower, 'cardnumber', 'web' );
72
73     my @borrowers = @$borrowers;
74
75     if ( $#borrowers == -1 ) {
76         $input->param( 'findborrower', '' );
77         $messageborrower = "'$findborrower'";
78     }
79     elsif ( $#borrowers == 0 ) {
80         $input->param( 'cardnumber', $borrowers[0]->{'cardnumber'} );
81         $cardnumber = $borrowers[0]->{'cardnumber'};
82     }
83     else {
84         $borrowerslist = \@borrowers;
85     }
86 }
87
88 if ($cardnumber) {
89     my $borrowerinfo = GetMemberDetails( 0, $cardnumber );
90     my $expiry;
91     my $diffbranch;
92     my @getreservloop;
93     my $count_reserv = 0;
94     my $maxreserves;
95
96 #       we check the reserves of the borrower, and if he can reserv a document
97 # FIXME At this time we have a simple count of reservs, but, later, we could improve the infos "title" ...
98
99     my $number_reserves =
100       GetReserveCount( $borrowerinfo->{'borrowernumber'} );
101
102     if ( $number_reserves > C4::Context->preference('maxreserves') ) {
103         $maxreserves = 1;
104     }
105
106     # we check the date expiricy of the borrower
107     my $warning = (Date_to_Days(split /-/,$date) > Date_to_Days( split /-/,$borrowerinfo->{'dateexpiry'}));
108      
109     if ( $warning > 0 ) {
110         $expiry = 1;
111     }
112
113     # check if the borrower make the reserv in a different branch
114     if ( $borrowerinfo->{'branchcode'} ne C4::Context->userenv->{'branch'} ) {
115         $diffbranch = 1;
116     }
117
118     $template->param(
119         borrowersurname   => $borrowerinfo->{'surname'},
120         borrowerfirstname => $borrowerinfo->{'firstname'},
121         borrowerreservs   => $count_reserv,
122         maxreserves       => $maxreserves,
123         expiry            => $expiry,
124         diffbranch        => $diffbranch
125     );
126 }
127
128 $template->param( messageborrower => $messageborrower );
129
130 my $CGIselectborrower;
131 if ($borrowerslist) {
132     my @values;
133     my %labels;
134
135     foreach my $borrower (
136         sort {
137                 $a->{surname}
138               . $a->{firstname} cmp $b->{surname}
139               . $b->{firstname}
140         } @{$borrowerslist}
141       )
142     {
143         push @values, $borrower->{cardnumber};
144
145         $labels{ $borrower->{cardnumber} } = sprintf(
146             '%s, %s ... (%s - %s) ... %s',
147             $borrower->{surname},    $borrower->{firstname},
148             $borrower->{cardnumber}, $borrower->{categorycode},
149             $borrower->{streetaddress},
150         );
151     }
152
153     $CGIselectborrower = CGI::scrolling_list(
154         -name     => 'cardnumber',
155         -values   => \@values,
156         -labels   => \%labels,
157         -size     => 7,
158         -multiple => 0,
159     );
160 }
161
162 # get existing reserves .....
163 my ( $count, $reserves ) = GetReservesFromBiblionumber($biblionumber);
164 my $totalcount = $count;
165 my $alreadyreserved;
166
167 # FIXME launch another time GetMemberDetails perhaps until
168 my $borrowerinfo = GetMemberDetails( 0, $cardnumber );
169
170 foreach my $res (@$reserves) {
171     if ( ( $res->{found} eq 'W' ) or ( $res->{priority} == 0 ) ) {
172         $count--;
173     }
174
175     if ( $borrowerinfo->{borrowernumber} eq $res->{borrowernumber} ) {
176         $alreadyreserved = 1;
177     }
178 }
179
180 $template->param( alreadyreserved => $alreadyreserved );
181
182 # FIXME think @optionloop, is maybe obsolete, or  must be switchable by a systeme preference fixed rank or not
183 # make priorities options
184
185 my @optionloop;
186 for ( 1 .. $count + 1 ) {
187     push(
188         @optionloop,
189         {
190             num      => $_,
191             selected => ( $_ == $count + 1 ),
192         }
193     );
194 }
195 # adding a fixed value for priority options
196 my $fixedRank = $count+1;
197
198 my @branchcodes;
199 my %itemnumbers_of_biblioitem;
200 my @itemnumbers  = @{ get_itemnumbers_of($biblionumber)->{$biblionumber} };
201 my $iteminfos_of = GetItemInfosOf(@itemnumbers);
202
203 foreach my $itemnumber (@itemnumbers) {
204     push( @branchcodes,
205         $iteminfos_of->{$itemnumber}->{homebranch},
206         $iteminfos_of->{$itemnumber}->{holdingbranch} );
207
208     my $biblioitemnumber = $iteminfos_of->{$itemnumber}->{biblioitemnumber};
209     push( @{ $itemnumbers_of_biblioitem{$biblioitemnumber} }, $itemnumber );
210 }
211
212 @branchcodes = uniq @branchcodes;
213
214 my @biblioitemnumbers = keys %itemnumbers_of_biblioitem;
215
216 my $branchinfos_of      = get_branchinfos_of(@branchcodes);
217 my $notforloan_label_of = get_notforloan_label_of();
218 my $biblioiteminfos_of  = GetBiblioItemInfosOf(@biblioitemnumbers);
219
220 my @itemtypes;
221 foreach my $biblioitemnumber (@biblioitemnumbers) {
222     push @itemtypes, $biblioiteminfos_of->{$biblioitemnumber}{itemtype};
223 }
224
225 my $itemtypeinfos_of = get_itemtypeinfos_of(@itemtypes);
226
227 my @bibitemloop;
228
229 foreach my $biblioitemnumber (@biblioitemnumbers) {
230     my $biblioitem = $biblioiteminfos_of->{$biblioitemnumber};
231
232     $biblioitem->{description} =
233       $itemtypeinfos_of->{ $biblioitem->{itemtype} }{description};
234
235     foreach
236       my $itemnumber ( @{ $itemnumbers_of_biblioitem{$biblioitemnumber} } )
237     {
238         my $item = $iteminfos_of->{$itemnumber};
239
240         $item->{homebranchname} =
241           $branchinfos_of->{ $item->{homebranch} }{branchname};
242
243         # if the holdingbranch is different than the homebranch, we show the
244         # holdingbranch of the document too
245         if ( $item->{homebranch} ne $item->{holdingbranch} ) {
246             $item->{holdingbranchname} =
247               $branchinfos_of->{ $item->{holdingbranch} }{branchname};
248         }
249         
250 #       add information
251         $item->{itemcallnumber} = $item->{itemcallnumber};
252         
253         # if the item is currently on loan, we display its return date and
254         # change the background color
255         my $issues= GetItemIssue($itemnumber);
256         if ( $issues->{'date_due'} ) {
257             $item->{date_due} = format_date($issues->{'date_due'});
258             $item->{backgroundcolor} = 'onloan';
259         }
260
261         # checking reserve
262         my ($reservedate,$reservedfor,$expectedAt) = GetReservesFromItemnumber($itemnumber);
263         my $ItemBorrowerReserveInfo = GetMemberDetails( $reservedfor, 0);
264
265         if ( defined $reservedate ) {
266             $item->{backgroundcolor} = 'reserved';
267             $item->{reservedate}     = format_date($reservedate);
268             $item->{ReservedForBorrowernumber}     = $reservedfor;
269             $item->{ReservedForSurname}     = $ItemBorrowerReserveInfo->{'surname'};
270             $item->{ReservedForFirstname}     = $ItemBorrowerReserveInfo->{'firstname'};
271             $item->{ExpectedAtLibrary}     = $expectedAt;
272             
273         }
274
275         # Management of the notforloan document
276         if ( $item->{notforloan} ) {
277             $item->{backgroundcolor} = 'other';
278             $item->{notforloanvalue} =
279               $notforloan_label_of->{ $item->{notforloan} };
280         }
281
282         # Management of lost or long overdue items
283         if ( $item->{itemlost} ) {
284
285             # FIXME localized strings should never be in Perl code
286             $item->{message} =
287                 $item->{itemlost} == 1 ? "(lost)"
288               : $item->{itemlost} == 2 ? "(long overdue)"
289               : "";
290             $item->{backgroundcolor} = 'other';
291         }
292
293         # Check of the transfered documents
294         my ( $transfertwhen, $transfertfrom, $transfertto ) =
295           GetTransfers($itemnumber);
296
297         if ( $transfertwhen ne '' ) {
298             $item->{transfertwhen} = format_date($transfertwhen);
299             $item->{transfertfrom} =
300               $branchinfos_of->{$transfertfrom}{branchname};
301             $item->{transfertto} = $branchinfos_of->{$transfertto}{branchname};
302                 $item->{nocancel} = 1;
303         }
304
305         # If there is no loan, return and transfer, we show a checkbox.
306         $item->{notforloan} = $item->{notforloan} || 0;
307
308         # An item is available only if:
309         if (
310             not defined $reservedate    # not reserved yet
311             and $issues->{'date_due'} eq ''         # not currently on loan
312             and not $item->{itemlost}   # not lost
313             and not $item->{notforloan} # not forbidden to loan
314             and $transfertwhen eq ''    # not currently on transfert
315           )
316         {
317             $item->{available} = 1;
318         }
319
320         push @{ $biblioitem->{itemloop} }, $item;
321     }
322
323     push @bibitemloop, $biblioitem;
324 }
325
326 # existingreserves building
327 my @reserveloop;
328 my $branches = GetBranches();
329 my ( $count, $reserves ) = GetReservesFromBiblionumber($biblionumber);
330 foreach my $res ( sort { $a->{found} cmp $b->{found} } @$reserves ) {
331     my %reserve;
332     my @optionloop;
333     for ( my $i = 1 ; $i <= $totalcount ; $i++ ) {
334         push(
335             @optionloop,
336             {
337                 num      => $i,
338                 selected => ( $i == $res->{priority} ),
339             }
340         );
341     }
342     my @branchloop;
343     foreach my $br ( keys %$branches ) {
344         my %abranch;
345         $abranch{'selected'}   = ( $br eq $res->{'branchcode'} );
346         $abranch{'branch'}     = $br;
347         $abranch{'branchname'} = $branches->{$br}->{'branchname'};
348         push( @branchloop, \%abranch );
349     }
350
351     if ( ( $res->{'found'} eq 'W' ) or ( $res->{'priority'} eq '0' ) ) {
352         my $item = $res->{'itemnumber'};
353         $item = GetBiblioFromItemNumber($item,undef);
354         $reserve{'wait'}= 1; 
355         $reserve{'holdingbranch'}=$item->{'holdingbranch'};
356         $reserve{'biblionumber'}=$item->{'biblionumber'};
357         $reserve{'barcodenumber'}       = $item->{'barcode'};
358         $reserve{'wbrcode'} = $res->{'branchcode'};
359         $reserve{'itemnumber'}  = $res->{'itemnumber'};
360         $reserve{'wbrname'} = $branches->{$res->{'branchcode'}}->{'branchname'};
361         if($reserve{'holdingbranch'} eq $reserve{'wbrcode'}){
362             $reserve{'atdestination'} = 1;
363         }
364     }
365     
366 #     get borrowers reserve info
367 my $reserveborrowerinfo = GetMemberDetails( $res->{'borrowernumber'}, 0);
368
369     $reserve{'date'}           = format_date( $res->{'reservedate'} );
370     $reserve{'borrowernumber'} = $res->{'borrowernumber'};
371     $reserve{'biblionumber'}   = $res->{'biblionumber'};
372     $reserve{'borrowernumber'} = $res->{'borrowernumber'};
373     $reserve{'firstname'}      = $reserveborrowerinfo->{'firstname'};
374     $reserve{'surname'}        = $reserveborrowerinfo->{'surname'};
375     $reserve{'notes'}          = $res->{'reservenotes'};
376     $reserve{'wait'}           =
377       ( ( $res->{'found'} eq 'W' ) or ( $res->{'priority'} eq '0' ) );
378     $reserve{'constrainttypea'} = ( $res->{'constrainttype'} eq 'a' );
379     $reserve{'constrainttypeo'} = ( $res->{'constrainttype'} eq 'o' );
380     $reserve{'voldesc'}         = $res->{'volumeddesc'};
381     $reserve{'ccode'}           = $res->{'ccode'};
382     $reserve{'barcode'}         = $res->{'barcode'};
383     $reserve{'priority'}        = $res->{'priority'};
384     $reserve{'branchloop'} = \@branchloop;
385     $reserve{'optionloop'} = \@optionloop;
386
387     push( @reserveloop, \%reserve );
388 }
389
390 my $default = C4::Context->userenv->{branch};
391 my @values;
392 my %label_of;
393
394 foreach my $branchcode ( keys %{$branches} ) {
395     push @values, $branchcode;
396     $label_of{$branchcode} = $branches->{$branchcode}->{branchname};
397 }
398 my $CGIbranch = CGI::scrolling_list(
399     -name     => 'pickup',
400     -values   => \@values,
401     -default  => $default,
402     -labels   => \%label_of,
403     -size     => 1,
404     -multiple => 0,
405 );
406
407 # get the time for the form name...
408 my $time = time();
409
410 $template->param(
411     CGIbranch   => $CGIbranch,
412     reserveloop => \@reserveloop,
413     time        => $time,
414     fixedRank   => $fixedRank,
415 );
416
417 # display infos
418 $template->param(
419     optionloop        => \@optionloop,
420     bibitemloop       => \@bibitemloop,
421     date              => $date,
422     biblionumber      => $biblionumber,
423     findborrower      => $findborrower,
424     cardnumber        => $cardnumber,
425     CGIselectborrower => $CGIselectborrower,
426     title             => $dat->{title},
427     author            => $dat->{author}
428 );
429
430 # printout the page
431 output_html_with_http_headers $input, $cookie, $template->output;