Bug 9507: prevent submit: refactor some code in a js file
[koha.git] / opac / opac-user.pl
1 #!/usr/bin/perl
2
3 # This file is part of Koha.
4 # parts copyright 2010 BibLibre
5 #
6 # Koha is free software; you can redistribute it and/or modify it under the
7 # terms of the GNU General Public License as published by the Free Software
8 # Foundation; either version 2 of the License, or (at your option) any later
9 # version.
10 #
11 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
12 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
13 # A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
14 #
15 # You should have received a copy of the GNU General Public License along
16 # with Koha; if not, write to the Free Software Foundation, Inc.,
17 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
18
19
20 use strict;
21 #use warnings; FIXME - Bug 2505
22
23 use CGI;
24
25 use C4::Auth;
26 use C4::Koha;
27 use C4::Circulation;
28 use C4::Reserves;
29 use C4::Members;
30 use C4::Members::AttributeTypes;
31 use C4::Members::Attributes qw/GetBorrowerAttributeValue/;
32 use C4::Output;
33 use C4::Overdues qw/CheckBorrowerDebarred/;
34 use C4::Biblio;
35 use C4::Items;
36 use C4::Letters;
37 use C4::Branch; # GetBranches
38 use Koha::DateUtils;
39
40 use constant ATTRIBUTE_SHOW_BARCODE => 'SHOW_BCODE';
41
42 use Date::Calc qw(
43   Today
44   Add_Delta_Days
45   Date_to_Days
46 );
47
48 my $query = new CGI;
49
50 BEGIN {
51     if (C4::Context->preference('BakerTaylorEnabled')) {
52         require C4::External::BakerTaylor;
53         import C4::External::BakerTaylor qw(&image_url &link_url);
54     }
55 }
56
57 my ( $template, $borrowernumber, $cookie ) = get_template_and_user(
58     {
59         template_name   => "opac-user.tmpl",
60         query           => $query,
61         type            => "opac",
62         authnotrequired => 0,
63         flagsrequired   => { borrow => 1 },
64         debug           => 1,
65     }
66 );
67
68 my $show_priority;
69 for ( C4::Context->preference("OPACShowHoldQueueDetails") ) {
70     m/priority/ and $show_priority = 1;
71 }
72 my $patronupdate = $query->param('patronupdate');
73 my $canrenew = 1;
74
75 # get borrower information ....
76 my ( $borr ) = GetMemberDetails( $borrowernumber );
77
78 my (  $today_year,   $today_month,   $today_day) = Today();
79 my ($warning_year, $warning_month, $warning_day) = split /-/, $borr->{'dateexpiry'};
80
81 $borr->{'ethnicity'} = fixEthnicity( $borr->{'ethnicity'} );
82
83 my $debar = CheckBorrowerDebarred($borrowernumber);
84 my $userdebarred;
85
86 if ($debar) {
87     $userdebarred = 1;
88     $template->param( 'userdebarred' => $userdebarred );
89     if ( $debar ne "9999-12-31" ) {
90         $borr->{'userdebarreddate'} = $debar;
91     }
92 }
93
94 if ( $userdebarred || $borr->{'gonenoaddress'} || $borr->{'lost'} ) {
95     $borr->{'flagged'} = 1;
96     $canrenew = 0;
97 }
98
99 if ( $borr->{'amountoutstanding'} > 5 ) {
100     $borr->{'amountoverfive'} = 1;
101 }
102 if ( 5 >= $borr->{'amountoutstanding'} && $borr->{'amountoutstanding'} > 0 ) {
103     $borr->{'amountoverzero'} = 1;
104 }
105 my $no_renewal_amt = C4::Context->preference( 'OPACFineNoRenewals' );
106 $no_renewal_amt ||= 0;
107
108 if ( $borr->{amountoutstanding} > $no_renewal_amt ) {
109     $borr->{'flagged'} = 1;
110     $canrenew = 0;
111     $template->param(
112         renewal_blocked_fines => sprintf( '%.02f', $no_renewal_amt ),
113         renewal_blocked_fines_amountoutstanding => sprintf( '%.02f', $borr->{amountoutstanding} ),
114     );
115 }
116
117 if ( $borr->{'amountoutstanding'} < 0 ) {
118     $borr->{'amountlessthanzero'} = 1;
119     $borr->{'amountoutstanding'} = -1 * ( $borr->{'amountoutstanding'} );
120 }
121
122 $borr->{'amountoutstanding'} = sprintf "%.02f", $borr->{'amountoutstanding'};
123
124 my @bordat;
125 $bordat[0] = $borr;
126
127 # Warningdate is the date that the warning starts appearing
128 if ( $borr->{dateexpiry} && Date_to_Days( $today_year, $today_month, $today_day ) > Date_to_Days( $warning_year, $warning_month, $warning_day ) ) {
129     $borr->{'warnexpired'} = 1;
130 }
131 elsif ( $borr->{dateexpiry} && C4::Context->preference('NotifyBorrowerDeparture') &&
132         Date_to_Days(Add_Delta_Days($warning_year, $warning_month, $warning_day,- C4::Context->preference('NotifyBorrowerDeparture'))) <
133         Date_to_Days( $today_year, $today_month, $today_day ) ) {
134         # borrower card soon to expire, warn the borrower
135         $borr->{'warndeparture'} = $borr->{dateexpiry};
136         if (C4::Context->preference('ReturnBeforeExpiry')){
137             $borr->{'returnbeforeexpiry'} = 1;
138         }
139 }
140
141
142 $template->param(   BORROWER_INFO     => \@bordat,
143                     borrowernumber    => $borrowernumber,
144                     patron_flagged    => $borr->{flagged},
145                     OPACMySummaryHTML => (C4::Context->preference("OPACMySummaryHTML")) ? 1 : 0,
146                     surname           => $borr->{surname},
147                     showname          => $borr->{showname},
148
149                 );
150
151 #get issued items ....
152
153 my $count          = 0;
154 my $overdues_count = 0;
155 my @overdues;
156 my @issuedat;
157 my $itemtypes = GetItemTypes();
158 my $issues = GetPendingIssues($borrowernumber);
159 if ($issues){
160     foreach my $issue ( sort { $b->{date_due}->datetime() cmp $a->{date_due}->datetime() } @{$issues} ) {
161         # check for reserves
162         my ( $restype, $res, undef ) = CheckReserves( $issue->{'itemnumber'} );
163         if ( $restype ) {
164             $issue->{'reserved'} = 1;
165         }
166
167         my ( $total , $accts, $numaccts) = GetMemberAccountRecords( $borrowernumber );
168         my $charges = 0;
169         foreach my $ac (@$accts) {
170             if ( $ac->{'itemnumber'} == $issue->{'itemnumber'} ) {
171                 $charges += $ac->{'amountoutstanding'}
172                   if $ac->{'accounttype'} eq 'F';
173                 $charges += $ac->{'amountoutstanding'}
174                   if $ac->{'accounttype'} eq 'FU';
175                 $charges += $ac->{'amountoutstanding'}
176                   if $ac->{'accounttype'} eq 'L';
177             }
178         }
179         $issue->{'charges'} = $charges;
180
181         # check if item is renewable
182         my ($status,$renewerror) = CanBookBeRenewed( $borrowernumber, $issue->{'itemnumber'} );
183         ($issue->{'renewcount'},$issue->{'renewsallowed'},$issue->{'renewsleft'}) = GetRenewCount($borrowernumber, $issue->{'itemnumber'});
184         if($status && C4::Context->preference("OpacRenewalAllowed")){
185             $issue->{'status'} = $status;
186         }
187         $issue->{'too_many'} = 1 if $renewerror and $renewerror eq 'too_many';
188         $issue->{'on_reserve'} = 1 if $renewerror and $renewerror eq 'on_reserve';
189
190         if ( $issue->{'overdue'} ) {
191             push @overdues, $issue;
192             $overdues_count++;
193             $issue->{'overdue'} = 1;
194         }
195         else {
196             $issue->{'issued'} = 1;
197         }
198         # imageurl:
199         my $itemtype = $issue->{'itemtype'};
200         if ( $itemtype ) {
201             $issue->{'imageurl'}    = getitemtypeimagelocation( 'opac', $itemtypes->{$itemtype}->{'imageurl'} );
202             $issue->{'description'} = $itemtypes->{$itemtype}->{'description'};
203         }
204         push @issuedat, $issue;
205         $count++;
206
207         my $isbn = GetNormalizedISBN($issue->{'isbn'});
208         $issue->{normalized_isbn} = $isbn;
209
210                 # My Summary HTML
211                 if (my $my_summary_html = C4::Context->preference('OPACMySummaryHTML')){
212                     $issue->{author} ? $my_summary_html =~ s/{AUTHOR}/$issue->{author}/g : $my_summary_html =~ s/{AUTHOR}//g;
213                     $issue->{title} =~ s/\/+$//; # remove trailing slash
214                     $issue->{title} =~ s/\s+$//; # remove trailing space
215                     $issue->{title} ? $my_summary_html =~ s/{TITLE}/$issue->{title}/g : $my_summary_html =~ s/{TITLE}//g;
216                     $issue->{isbn} ? $my_summary_html =~ s/{ISBN}/$isbn/g : $my_summary_html =~ s/{ISBN}//g;
217                     $issue->{biblionumber} ? $my_summary_html =~ s/{BIBLIONUMBER}/$issue->{biblionumber}/g : $my_summary_html =~ s/{BIBLIONUMBER}//g;
218                     $issue->{MySummaryHTML} = $my_summary_html;
219                 }
220     }
221 }
222 $template->param( ISSUES       => \@issuedat );
223 $template->param( issues_count => $count );
224 $template->param( canrenew     => $canrenew );
225 $template->param( OVERDUES       => \@overdues );
226 $template->param( overdues_count => $overdues_count );
227
228 my $show_barcode = C4::Members::AttributeTypes::AttributeTypeExists( ATTRIBUTE_SHOW_BARCODE );
229 if ($show_barcode) {
230     my $patron_show_barcode = GetBorrowerAttributeValue($borrowernumber, ATTRIBUTE_SHOW_BARCODE);
231     undef $show_barcode if defined($patron_show_barcode) && !$patron_show_barcode;
232 }
233 $template->param( show_barcode => 1 ) if $show_barcode;
234
235 # load the branches
236 my $branches = GetBranches();
237 my @branch_loop;
238 for my $branch_hash ( sort keys %{$branches} ) {
239     my $selected;
240     if ( C4::Context->preference('SearchMyLibraryFirst') ) {
241         $selected =
242           ( C4::Context->userenv
243               && ( $branch_hash eq C4::Context->userenv->{branch} ) );
244     }
245     push @branch_loop,
246       { value      => "branch: $branch_hash",
247         branchname => $branches->{$branch_hash}->{'branchname'},
248         selected   => $selected,
249       };
250 }
251 $template->param( branchloop => \@branch_loop );
252
253 # now the reserved items....
254 my @reserves  = GetReservesFromBorrowernumber( $borrowernumber );
255 foreach my $res (@reserves) {
256
257     if ( $res->{'expirationdate'} eq '0000-00-00' ) {
258       $res->{'expirationdate'} = '';
259     }
260
261     $res->{'waiting'} = 1 if $res->{'found'} eq 'W';
262     $res->{'branch'} = $branches->{ $res->{'branchcode'} }->{'branchname'};
263     my $biblioData = GetBiblioData($res->{'biblionumber'});
264     $res->{'reserves_title'} = $biblioData->{'title'};
265     if ($show_priority) {
266         $res->{'priority'} ||= '';
267     }
268     $res->{'suspend_until'} = C4::Dates->new( $res->{'suspend_until'}, "iso")->output("syspref") if ( $res->{'suspend_until'} );
269 }
270
271 # use Data::Dumper;
272 # warn Dumper(@reserves);
273
274 $template->param( RESERVES       => \@reserves );
275 $template->param( reserves_count => $#reserves+1 );
276 $template->param( showpriority=>$show_priority );
277
278 my @waiting;
279 my $wcount = 0;
280 foreach my $res (@reserves) {
281     if ( $res->{'itemnumber'} ) {
282         my $item = GetItem( $res->{'itemnumber'});
283         $res->{'holdingbranch'} =
284           $branches->{ $item->{'holdingbranch'} }->{'branchname'};
285         $res->{'branch'} = $branches->{ $res->{'branchcode'} }->{'branchname'};
286         # get document reserve status
287         my $biblioData = GetBiblioData($res->{'biblionumber'});
288         $res->{'waiting_title'} = $biblioData->{'title'};
289         if ( ( $res->{'found'} eq 'W' ) ) {
290             my $item = $res->{'itemnumber'};
291             $item = GetBiblioFromItemNumber($item,undef);
292             $res->{'wait'}= 1;
293             $res->{'holdingbranch'}=$item->{'holdingbranch'};
294             $res->{'biblionumber'}=$item->{'biblionumber'};
295             $res->{'barcode'} = $item->{'barcode'};
296             $res->{'wbrcode'} = $res->{'branchcode'};
297             $res->{'itemnumber'}    = $res->{'itemnumber'};
298             $res->{'wbrname'} = $branches->{$res->{'branchcode'}}->{'branchname'};
299             if($res->{'holdingbranch'} eq $res->{'wbrcode'}){
300                 $res->{'atdestination'} = 1;
301             }
302             # set found to 1 if reserve is waiting for patron pickup
303             $res->{'found'} = 1 if $res->{'found'} eq 'W';
304         } else {
305             my ($transfertwhen, $transfertfrom, $transfertto) = GetTransfers( $res->{'itemnumber'} );
306             if ($transfertwhen) {
307                 $res->{intransit} = 1;
308                 $res->{datesent}   = $transfertwhen;
309                 $res->{frombranch} = GetBranchName($transfertfrom);
310             }
311         }
312         push @waiting, $res;
313         $wcount++;
314     }
315     # can be cancelled
316     #$res->{'cancelable'} = 1 if ($res->{'wait'} && $res->{'atdestination'} && $res->{'found'} ne "1");
317     $res->{'cancelable'} = 1 if    ($res->{wait} and not $res->{found}) or (not $res->{wait} and not $res->{intransit});
318
319 }
320
321 $template->param( WAITING => \@waiting );
322
323 # current alert subscriptions
324 my $alerts = getalert($borrowernumber);
325 foreach ( @$alerts ) {
326     $_->{ $_->{type} } = 1;
327     $_->{relatedto} = findrelatedto( $_->{type}, $_->{externalid} );
328 }
329
330 if (C4::Context->preference('BakerTaylorEnabled')) {
331     $template->param(
332         BakerTaylorEnabled  => 1,
333         BakerTaylorImageURL => &image_url(),
334         BakerTaylorLinkURL  => &link_url(),
335         BakerTaylorBookstoreURL => C4::Context->preference('BakerTaylorBookstoreURL'),
336     );
337 }
338
339 if (C4::Context->preference("OPACAmazonCoverImages") or 
340     C4::Context->preference("GoogleJackets") or
341     C4::Context->preference("BakerTaylorEnabled") or
342     C4::Context->preference("SyndeticsCoverImages")) {
343         $template->param(JacketImages=>1);
344 }
345
346 if ( GetMessagesCount( $borrowernumber, 'B' ) ) {
347     $template->param( bor_messages => 1 );
348 }
349
350 if ( $borr->{'opacnote'} ) {
351   $template->param( 
352     bor_messages => 1,
353     opacnote => $borr->{'opacnote'},
354   );
355 }
356
357 $template->param(
358     bor_messages_loop    => GetMessages( $borrowernumber, 'B', 'NONE' ),
359     waiting_count      => $wcount,
360     patronupdate => $patronupdate,
361     OpacRenewalAllowed => C4::Context->preference("OpacRenewalAllowed"),
362     userview => 1,
363     dateformat    => C4::Context->preference("dateformat"),
364 );
365
366 $template->param( DHTMLcalendar_dateformat  => C4::Dates->DHTMLcalendar() );
367 $template->param(
368     SuspendHoldsOpac => C4::Context->preference('SuspendHoldsOpac'),
369     AutoResumeSuspendedHolds => C4::Context->preference('AutoResumeSuspendedHolds') ,
370 );
371
372 output_html_with_http_headers $query, $cookie, $template->output;
373