3 # Copyright 2000-2002 Katipo Communications
4 # Copyright 2010 BibLibre
5 # Copyright 2011 KohaAloha, NZ
7 # This file is part of Koha.
9 # Koha is free software; you can redistribute it and/or modify it
10 # under the terms of the GNU General Public License as published by
11 # the Free Software Foundation; either version 3 of the License, or
12 # (at your option) any later version.
14 # Koha is distributed in the hope that it will be useful, but
15 # WITHOUT ANY WARRANTY; without even the implied warranty of
16 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 # GNU General Public License for more details.
19 # You should have received a copy of the GNU General Public License
20 # along with Koha; if not, see <http://www.gnu.org/licenses>.
26 use C4
::Acquisition
qw( SearchOrders );
27 use C4
::Auth
qw(:DEFAULT get_session);
29 use C4
::Serials
; #uses getsubscriptionfrom biblionumber
34 use C4
::Tags
qw(get_tags);
35 use C4
::XISBN
qw(get_xisbns);
36 use C4
::External
::Amazon
;
37 use C4
::External
::Syndetics
qw(get_syndetics_index get_syndetics_summary get_syndetics_toc get_syndetics_excerpt get_syndetics_reviews get_syndetics_anotes );
46 use List
::MoreUtils qw
/any none/;
50 use C4
::CourseReserves
qw(GetItemCourseReservesInfo);
53 use Koha
::RecordProcessor
;
54 use Koha
::AuthorisedValues
;
55 use Koha
::IssuingRules
;
58 use Koha
::Acquisition
::Orders
;
59 use Koha
::Virtualshelves
;
65 if (C4
::Context
->preference('BakerTaylorEnabled')) {
66 require C4
::External
::BakerTaylor
;
67 import C4
::External
::BakerTaylor
qw(&image_url &link_url);
71 my $query = CGI
->new();
73 my $biblionumber = $query->param('biblionumber') || $query->param('bib') || 0;
74 $biblionumber = int($biblionumber);
76 my ( $template, $borrowernumber, $cookie ) = get_template_and_user
(
78 template_name
=> "opac-detail.tt",
81 authnotrequired
=> ( C4
::Context
->preference("OpacPublic") ?
1 : 0 ),
85 my @all_items = GetItemsInfo
($biblionumber);
87 my $patron = Koha
::Patrons
->find( $borrowernumber );
89 if ( C4
::Context
->preference('OpacHiddenItemsExceptions') ) {
90 $borcat = $patron ?
$patron->categorycode : q{};
93 my $record = GetMarcBiblio
({
94 biblionumber
=> $biblionumber,
97 print $query->redirect("/cgi-bin/koha/errors/404.pl"); # escape early
101 if ( scalar @all_items >= 1 ) {
103 GetHiddenItemnumbers
( { items
=> \
@all_items, borcat
=> $borcat } );
105 if (scalar @hiddenitems == scalar @all_items ) {
106 print $query->redirect("/cgi-bin/koha/errors/404.pl"); # escape early
111 my $biblio = Koha
::Biblios
->find( $biblionumber );
112 my $framework = $biblio ?
$biblio->frameworkcode : q{};
113 my $record_processor = Koha
::RecordProcessor
->new({
114 filters
=> 'ViewPolicy',
117 frameworkcode
=> $framework
120 $record_processor->process($record);
122 # redirect if opacsuppression is enabled and biblio is suppressed
123 if (C4
::Context
->preference('OpacSuppression')) {
124 # FIXME hardcoded; the suppression flag ought to be materialized
125 # as a column on biblio or the like
126 my $opacsuppressionfield = '942';
127 my $opacsuppressionfieldvalue = $record->field($opacsuppressionfield);
128 # redirect to opac-blocked info page or 404?
129 my $opacsuppressionredirect;
130 if ( C4
::Context
->preference("OpacSuppressionRedirect") ) {
131 $opacsuppressionredirect = "/cgi-bin/koha/opac-blocked.pl";
133 $opacsuppressionredirect = "/cgi-bin/koha/errors/404.pl";
135 if ( $opacsuppressionfieldvalue &&
136 $opacsuppressionfieldvalue->subfield("n") &&
137 $opacsuppressionfieldvalue->subfield("n") == 1) {
138 # if OPAC suppression by IP address
139 if (C4
::Context
->preference('OpacSuppressionByIPRange')) {
140 my $IPAddress = $ENV{'REMOTE_ADDR'};
141 my $IPRange = C4
::Context
->preference('OpacSuppressionByIPRange');
142 if ($IPAddress !~ /^$IPRange/) {
143 print $query->redirect($opacsuppressionredirect);
147 print $query->redirect($opacsuppressionredirect);
153 $template->param( biblio
=> $biblio );
155 # get biblionumbers stored in the cart
158 if($query->cookie("bib_list")){
159 my $cart_list = $query->cookie("bib_list");
160 @cart_list = split(/\//, $cart_list);
161 if ( grep {$_ eq $biblionumber} @cart_list) {
162 $template->param( incart
=> 1 );
167 SetUTF8Flag
($record);
168 my $marcflavour = C4
::Context
->preference("marcflavour");
169 my $ean = GetNormalizedEAN
( $record, $marcflavour );
171 # XSLT processing of some stuff
172 my $xslfile = C4
::Context
->preference('OPACXSLTDetailsDisplay');
173 my $lang = $xslfile ? C4
::Languages
::getlanguage
() : undef;
174 my $sysxml = $xslfile ? C4
::XSLT
::get_xslt_sysprefs
() : undef;
178 XSLTBloc
=> XSLTParse4Display
(
179 $biblionumber, $record, "OPACXSLTDetailsDisplay",
180 1, undef, $sysxml, $xslfile, $lang
185 my $OpacBrowseResults = C4
::Context
->preference("OpacBrowseResults");
187 # We look for the busc param to build the simple paging from the search
188 if ($OpacBrowseResults) {
189 my $session = get_session
($query->cookie("CGISESSID"));
190 my %paging = (previous
=> {}, next => {});
191 if ($session->param('busc')) {
195 # Rebuild the string to store on session
196 # param value is URI encoded and params separator is HTML encode (&)
199 my $arrParamsBusc = shift;
201 my $pasarParams = '';
203 for (keys %$arrParamsBusc) {
204 if ($_ =~ /^(?:query|listBiblios|newlistBiblios|query_type|simple_query|total|offset|offsetSearch|next|previous|count|expand|scan)/) {
205 if (defined($arrParamsBusc->{$_})) {
206 $pasarParams .= '&' if ($j);
207 $pasarParams .= $_ . '=' . Encode
::decode
('UTF-8', uri_escape_utf8
( $arrParamsBusc->{$_} ));
211 for my $value (@
{$arrParamsBusc->{$_}}) {
212 next if !defined($value);
213 $pasarParams .= '&' if ($j);
214 $pasarParams .= $_ . '=' . Encode
::decode
('UTF-8', uri_escape_utf8
($value));
222 # Search given the current values from the busc param
225 my ($arrParamsBusc, $offset, $results_per_page) = @_;
227 my $itemtypes = { map { $_->{itemtype
} => $_ } @
{ Koha
::ItemTypes
->search_with_localization->unblessed } };
229 @servers = @
{$arrParamsBusc->{'server'}} if $arrParamsBusc->{'server'};
230 @servers = ("biblioserver") unless (@servers);
232 my ($default_sort_by, @sort_by);
233 $default_sort_by = C4
::Context
->preference('OPACdefaultSortField')."_".C4
::Context
->preference('OPACdefaultSortOrder') if (C4
::Context
->preference('OPACdefaultSortField') && C4
::Context
->preference('OPACdefaultSortOrder'));
234 @sort_by = @
{$arrParamsBusc->{'sort_by'}} if $arrParamsBusc->{'sort_by'};
235 $sort_by[0] = $default_sort_by if !$sort_by[0] && defined($default_sort_by);
236 my ($error, $results_hashref, $facets);
238 ($error, $results_hashref, $facets) = getRecords
($arrParamsBusc->{'query'},$arrParamsBusc->{'simple_query'},\
@sort_by,\
@servers,$results_per_page,$offset,undef,$itemtypes,$arrParamsBusc->{'query_type'},$arrParamsBusc->{'scan'});
242 my $search_context = {
243 'interface' => 'opac',
244 'category' => $borcat
246 for (my $i=0;$i<@servers;$i++) {
247 my $server = $servers[$i];
248 $hits = $results_hashref->{$server}->{"hits"};
249 @newresults = searchResults
( $search_context, '', $hits, $results_per_page, $offset, $arrParamsBusc->{'scan'}, $results_hashref->{$server}->{"RECORDS"});
254 # Build the current list of biblionumbers in this search
257 my ($newresultsRef, $results_per_page) = @_;
259 my $listBiblios = '';
261 foreach (@
$newresultsRef) {
262 my $bibnum = ($_->{biblionumber
})?
$_->{biblionumber
}:0;
263 $listBiblios .= $bibnum . ',';
265 last if ($j == $results_per_page);
267 chop $listBiblios if ($listBiblios =~ /,$/);
271 my $busc = $session->param("busc");
272 my @arrBusc = split(/\&(?:amp;)?/, $busc);
274 my %arrParamsBusc = ();
276 ($key, $value) = split(/=/, $_, 2);
277 if ($key =~ /^(?:query|listBiblios|newlistBiblios|query_type|simple_query|next|previous|total|offset|offsetSearch|count|expand|scan)/) {
278 $arrParamsBusc{$key} = uri_unescape
($value);
280 unless (exists($arrParamsBusc{$key})) {
281 $arrParamsBusc{$key} = [];
283 push @
{$arrParamsBusc{$key}}, uri_unescape
($value);
287 my $count = C4
::Context
->preference('OPACnumSearchResults') || 20;
288 my $results_per_page = ($arrParamsBusc{'count'} && $arrParamsBusc{'count'} =~ /^[0-9]+?/)?
$arrParamsBusc{'count'}:$count;
289 $arrParamsBusc{'count'} = $results_per_page;
290 my $offset = ($arrParamsBusc{'offset'} && $arrParamsBusc{'offset'} =~ /^[0-9]+?/)?
$arrParamsBusc{'offset'}:0;
291 # The value OPACnumSearchResults has changed and the search has to be rebuild
292 if ($count != $results_per_page) {
293 if (exists($arrParamsBusc{'listBiblios'}) && $arrParamsBusc{'listBiblios'} =~ /^[0-9]+(?:,[0-9]+)*$/) {
295 my @arrBibliosAux = split(',', $arrParamsBusc{'listBiblios'});
296 for (@arrBibliosAux) {
297 last if ($_ == $biblionumber);
300 $indexBiblio += $offset;
301 $offset = int($indexBiblio / $count) * $count;
302 $arrParamsBusc{'offset'} = $offset;
304 $arrParamsBusc{'count'} = $count;
305 $results_per_page = $count;
306 my $newresultsRef = searchAgain
(\
%arrParamsBusc, $offset, $results_per_page);
307 $arrParamsBusc{'listBiblios'} = buildListBiblios
($newresultsRef, $results_per_page);
308 delete $arrParamsBusc{'previous'} if (exists($arrParamsBusc{'previous'}));
309 delete $arrParamsBusc{'next'} if (exists($arrParamsBusc{'next'}));
310 delete $arrParamsBusc{'offsetSearch'} if (exists($arrParamsBusc{'offsetSearch'}));
311 delete $arrParamsBusc{'newlistBiblios'} if (exists($arrParamsBusc{'newlistBiblios'}));
312 my $newbusc = rebuildBuscParam
(\
%arrParamsBusc);
313 $session->param("busc" => $newbusc);
314 @arrBusc = split(/\&(?:amp;)?/, $newbusc);
316 my $modifyListBiblios = 0;
317 # We come from a previous click
318 if (exists($arrParamsBusc{'previous'})) {
319 $modifyListBiblios = 1 if ($biblionumber == $arrParamsBusc{'previous'});
320 delete $arrParamsBusc{'previous'};
321 } elsif (exists($arrParamsBusc{'next'})) { # We come from a next click
322 $modifyListBiblios = 2 if ($biblionumber == $arrParamsBusc{'next'});
323 delete $arrParamsBusc{'next'};
325 if ($modifyListBiblios) {
326 if (exists($arrParamsBusc{'newlistBiblios'})) {
327 my $listBibliosAux = $arrParamsBusc{'listBiblios'};
328 $arrParamsBusc{'listBiblios'} = $arrParamsBusc{'newlistBiblios'};
329 my @arrAux = split(',', $listBibliosAux);
330 $arrParamsBusc{'newlistBiblios'} = $listBibliosAux;
331 if ($modifyListBiblios == 1) {
332 $arrParamsBusc{'next'} = $arrAux[0];
333 $paging{'next'}->{biblionumber
} = $arrAux[0];
335 $arrParamsBusc{'previous'} = $arrAux[$#arrAux];
336 $paging{'previous'}->{biblionumber
} = $arrAux[$#arrAux];
339 delete $arrParamsBusc{'listBiblios'};
341 my $offsetAux = $arrParamsBusc{'offset'};
342 $arrParamsBusc{'offset'} = $arrParamsBusc{'offsetSearch'};
343 $arrParamsBusc{'offsetSearch'} = $offsetAux;
344 $offset = $arrParamsBusc{'offset'};
345 my $newbusc = rebuildBuscParam
(\
%arrParamsBusc);
346 $session->param("busc" => $newbusc);
347 @arrBusc = split(/\&(?:amp;)?/, $newbusc);
352 # Rebuild the query for the button "back to results"
354 unless ($_ =~ /^(?:query|listBiblios|newlistBiblios|query_type|simple_query|next|previous|total|count|offsetSearch)/) {
355 $buscParam .= '&' unless ($j == 0);
356 $buscParam .= $_; # string already URI encoded
360 $template->param('busc' => $buscParam);
363 # We are inside the list of biblios and we don't have to search
364 if (exists($arrParamsBusc{'listBiblios'}) && $arrParamsBusc{'listBiblios'} =~ /^[0-9]+(?:,[0-9]+)*$/) {
365 @arrBiblios = split(',', $arrParamsBusc{'listBiblios'});
367 # We are at the first item of the list
368 if ($arrBiblios[0] == $biblionumber) {
369 if (@arrBiblios > 1) {
370 for (my $j = 1; $j < @arrBiblios; $j++) {
371 next unless ($arrBiblios[$j]);
372 $paging{'next'}->{biblionumber
} = $arrBiblios[$j];
376 # search again if we are not at the first searching list
377 if ($offset && !$arrParamsBusc{'previous'}) {
379 $offsetSearch = $offset - $results_per_page;
381 # we are at the last item of the list
382 } elsif ($arrBiblios[$#arrBiblios] == $biblionumber) {
383 for (my $j = $#arrBiblios - 1; $j >= 0; $j--) {
384 next unless ($arrBiblios[$j]);
385 $paging{'previous'}->{biblionumber
} = $arrBiblios[$j];
389 # search again if we are at the first list and there is more results
390 $searchAgain = 1 if (!$arrParamsBusc{'next'} && $arrParamsBusc{'total'} != @arrBiblios);
392 # search again if we aren't at the first list and there is more results
393 $searchAgain = 1 if (!$arrParamsBusc{'next'} && $arrParamsBusc{'total'} > ($offset + @arrBiblios));
395 $offsetSearch = $offset + $results_per_page if ($searchAgain);
397 for (my $j = 1; $j < $#arrBiblios; $j++) {
398 if ($arrBiblios[$j] == $biblionumber) {
399 for (my $z = $j - 1; $z >= 0; $z--) {
400 next unless ($arrBiblios[$z]);
401 $paging{'previous'}->{biblionumber
} = $arrBiblios[$z];
404 for (my $z = $j + 1; $z < @arrBiblios; $z++) {
405 next unless ($arrBiblios[$z]);
406 $paging{'next'}->{biblionumber
} = $arrBiblios[$z];
414 $offsetSearch = 0 if (defined($offsetSearch) && $offsetSearch < 0);
417 my $newresultsRef = searchAgain
(\
%arrParamsBusc, $offsetSearch, $results_per_page);
418 my @newresults = @
$newresultsRef;
419 # build the new listBiblios
420 my $listBiblios = buildListBiblios
(\
@newresults, $results_per_page);
421 unless (exists($arrParamsBusc{'listBiblios'})) {
422 $arrParamsBusc{'listBiblios'} = $listBiblios;
423 @arrBiblios = split(',', $arrParamsBusc{'listBiblios'});
425 $arrParamsBusc{'newlistBiblios'} = $listBiblios;
427 # From the new list we build again the next and previous result
429 if ($arrBiblios[0] == $biblionumber) {
430 for (my $j = $#newresults; $j >= 0; $j--) {
431 next unless ($newresults[$j]);
432 $paging{'previous'}->{biblionumber
} = $newresults[$j]->{biblionumber
};
433 $arrParamsBusc{'previous'} = $paging{'previous'}->{biblionumber
};
434 $arrParamsBusc{'offsetSearch'} = $offsetSearch;
437 } elsif ($arrBiblios[$#arrBiblios] == $biblionumber) {
438 for (my $j = 0; $j < @newresults; $j++) {
439 next unless ($newresults[$j]);
440 $paging{'next'}->{biblionumber
} = $newresults[$j]->{biblionumber
};
441 $arrParamsBusc{'next'} = $paging{'next'}->{biblionumber
};
442 $arrParamsBusc{'offsetSearch'} = $offsetSearch;
447 # build new busc param
448 my $newbusc = rebuildBuscParam
(\
%arrParamsBusc);
449 $session->param("busc" => $newbusc);
451 my ($numberBiblioPaging, $dataBiblioPaging);
453 $numberBiblioPaging = $paging{'previous'}->{biblionumber
};
454 if ($numberBiblioPaging) {
455 $template->param( 'previousBiblionumber' => $numberBiblioPaging );
456 $dataBiblioPaging = Koha
::Biblios
->find( $numberBiblioPaging );
457 $template->param('previousTitle' => $dataBiblioPaging->title) if $dataBiblioPaging;
460 $numberBiblioPaging = $paging{'next'}->{biblionumber
};
461 if ($numberBiblioPaging) {
462 $template->param( 'nextBiblionumber' => $numberBiblioPaging );
463 $dataBiblioPaging = Koha
::Biblios
->find( $numberBiblioPaging );
464 $template->param('nextTitle' => $dataBiblioPaging->title) if $dataBiblioPaging;
466 # Partial list of biblio results
468 for (my $j = 0; $j < @arrBiblios; $j++) {
469 next unless ($arrBiblios[$j]);
470 $dataBiblioPaging = Koha
::Biblios
->find( $arrBiblios[$j] ) if ($arrBiblios[$j] != $biblionumber);
471 push @listResults, {index => $j + 1 + $offset, biblionumber
=> $arrBiblios[$j], title
=> ($arrBiblios[$j] == $biblionumber)?
'':$dataBiblioPaging->title, author
=> ($arrBiblios[$j] != $biblionumber && $dataBiblioPaging->author)?
$dataBiblioPaging->author:'', url
=> ($arrBiblios[$j] == $biblionumber)?
'':'opac-detail.pl?biblionumber=' . $arrBiblios[$j]};
473 $template->param('listResults' => \
@listResults) if (@listResults);
474 $template->param('indexPag' => 1 + $offset, 'totalPag' => $arrParamsBusc{'total'}, 'indexPagEnd' => scalar(@arrBiblios) + $offset);
475 $template->param( 'offset' => $offset );
480 OPACShowCheckoutName
=> C4
::Context
->preference("OPACShowCheckoutName"),
483 if ( C4
::Context
->preference('EasyAnalyticalRecords') ) {
484 # adding items linked via host biblios
485 my $analyticfield = '773';
486 if ($marcflavour eq 'MARC21' || $marcflavour eq 'NORMARC'){
487 $analyticfield = '773';
488 } elsif ($marcflavour eq 'UNIMARC') {
489 $analyticfield = '461';
491 foreach my $hostfield ( $record->field($analyticfield)) {
492 my $hostbiblionumber = $hostfield->subfield("0");
493 my $linkeditemnumber = $hostfield->subfield("9");
494 my @hostitemInfos = GetItemsInfo
($hostbiblionumber);
495 foreach my $hostitemInfo (@hostitemInfos){
496 if ($hostitemInfo->{itemnumber
} eq $linkeditemnumber){
497 push(@all_items, $hostitemInfo);
505 # Are there items to hide?
507 $hideitems = 1 if C4
::Context
->preference('hidelostitems') or scalar(@hiddenitems) > 0;
511 for my $itm (@all_items) {
512 if ( C4
::Context
->preference('hidelostitems') ) {
513 push @items, $itm unless $itm->{itemlost
} or any
{ $itm->{'itemnumber'} eq $_ } @hiddenitems;
515 push @items, $itm unless any
{ $itm->{'itemnumber'} eq $_ } @hiddenitems;
524 if (C4
::Context
->userenv){
525 $branch = C4
::Context
->userenv->{branch
};
527 if ( C4
::Context
->preference('HighlightOwnItemsOnOPAC') ) {
529 ( ( C4
::Context
->preference('HighlightOwnItemsOnOPACWhich') eq 'PatronBranch' ) && $branch )
531 C4
::Context
->preference('HighlightOwnItemsOnOPACWhich') eq 'OpacURLBranch'
534 if ( C4
::Context
->preference('HighlightOwnItemsOnOPACWhich') eq 'PatronBranch' ) {
535 $branchcode = $branch;
537 elsif ( C4
::Context
->preference('HighlightOwnItemsOnOPACWhich') eq 'OpacURLBranch' ) {
538 $branchcode = $ENV{'BRANCHCODE'};
544 foreach my $item ( @items ) {
545 if ( $item->{branchcode
} eq $branchcode ) {
546 $item->{'this_branch'} = 1;
547 push( @our_items, $item );
549 push( @other_items, $item );
553 @items = ( @our_items, @other_items );
557 my $dat = &GetBiblioData
($biblionumber);
558 my $HideMARC = $record_processor->filters->[0]->should_hide_marc(
560 frameworkcode
=> $dat->{'frameworkcode'},
564 my $itemtypes = { map { $_->{itemtype
} => $_ } @
{ Koha
::ItemTypes
->search_with_localization->unblessed } };
566 my $itemtype = $dat->{'itemtype'};
568 $dat->{'imageurl'} = getitemtypeimagelocation
( 'opac', $itemtypes->{$itemtype}->{'imageurl'} );
569 $dat->{'description'} = $itemtypes->{$itemtype}->{translated_description
};
573 { map { $_->{authorised_value
} => $_->{opac_description
} } Koha
::AuthorisedValues
->get_descriptions_by_koha_field( { frameworkcode
=> $dat->{frameworkcode
}, kohafield
=> 'items.location' } ) };
575 { map { $_->{authorised_value
} => $_->{opac_description
} } Koha
::AuthorisedValues
->get_descriptions_by_koha_field( { frameworkcode
=> $dat->{frameworkcode
}, kohafield
=> 'items.ccode' } ) };
577 { map { $_->{authorised_value
} => $_->{opac_description
} } Koha
::AuthorisedValues
->get_descriptions_by_koha_field( { frameworkcode
=> $dat->{frameworkcode
}, kohafield
=> 'items.copynumber' } ) };
579 #coping with subscriptions
580 my $subscriptionsnumber = CountSubscriptionFromBiblionumber
($biblionumber);
581 my @subscriptions = SearchSubscriptions
({ biblionumber
=> $biblionumber, orderby
=> 'title' });
584 $dat->{'serial'}=1 if $subscriptionsnumber;
585 foreach my $subscription (@subscriptions) {
586 my $serials_to_display;
588 $cell{subscriptionid
} = $subscription->{subscriptionid
};
589 $cell{subscriptionnotes
} = $subscription->{notes
};
590 $cell{missinglist
} = $subscription->{missinglist
};
591 $cell{opacnote
} = $subscription->{opacnote
};
592 $cell{histstartdate
} = $subscription->{histstartdate
};
593 $cell{histenddate
} = $subscription->{histenddate
};
594 $cell{branchcode
} = $subscription->{branchcode
};
595 $cell{callnumber
} = $subscription->{callnumber
};
596 $cell{location
} = $subscription->{location
};
597 $cell{closed
} = $subscription->{closed
};
598 $cell{letter
} = $subscription->{letter
};
599 $cell{biblionumber
} = $subscription->{biblionumber
};
600 #get the three latest serials.
601 $serials_to_display = $subscription->{opacdisplaycount
};
602 $serials_to_display = C4
::Context
->preference('OPACSerialIssueDisplayCount') unless $serials_to_display;
603 $cell{opacdisplaycount
} = $serials_to_display;
604 $cell{latestserials
} =
605 GetLatestSerials
( $subscription->{subscriptionid
}, $serials_to_display );
606 if ( $borrowernumber ) {
607 my $subscription_object = Koha
::Subscriptions
->find( $subscription->{subscriptionid
} );
608 my $subscriber = $subscription_object->subscribers->find( $borrowernumber );
609 $cell{hasalert
} = 1 if $subscriber;
614 $dat->{'count'} = scalar(@items);
617 my (%item_reserves, %priority);
618 my ($show_holds_count, $show_priority);
619 for ( C4
::Context
->preference("OPACShowHoldQueueDetails") ) {
620 m/holds/o and $show_holds_count = 1;
621 m/priority/ and $show_priority = 1;
624 if ( $show_holds_count || $show_priority) {
625 my $holds = $biblio->holds;
626 $template->param( holds_count
=> $holds->count );
627 while ( my $hold = $holds->next ) {
628 $item_reserves{ $hold->itemnumber }++ if $hold->itemnumber;
629 if ($show_priority && $hold->borrowernumber == $borrowernumber) {
632 ?
($priority{ $hold->itemnumber } = $hold->priority)
633 : ($template->param( priority
=> $hold->priority ));
637 $template->param( show_priority
=> $has_hold ) ;
641 my (@itemloop, @otheritemloop);
642 my $currentbranch = C4
::Context
->userenv ? C4
::Context
->userenv->{branch
} : undef;
643 if ($currentbranch and C4
::Context
->preference('OpacSeparateHoldings')) {
644 $template->param(SeparateHoldings
=> 1);
646 my $separatebranch = C4
::Context
->preference('OpacSeparateHoldingsBranch');
647 my $viewallitems = $query->param('viewallitems');
648 my $max_items_to_display = C4
::Context
->preference('OpacMaxItemsToDisplay') // 50;
651 my ( @itemnumbers_on_order );
652 if ( C4
::Context
->preference('OPACAcquisitionDetails' ) ) {
653 my $orders = C4
::Acquisition
::SearchOrders
({
654 biblionumber
=> $biblionumber,
657 my $total_quantity = 0;
658 for my $order ( @
$orders ) {
659 my $order = Koha
::Acquisition
::Orders
->find( $order->{ordernumber
} );
660 my $basket = $order->basket;
661 if ( $basket->effective_create_items eq 'ordering' ) {
662 @itemnumbers_on_order = $order->items->get_column('itemnumber');
664 $total_quantity += $order->quantity;
666 $template->{VARS
}->{acquisition_details
} = {
667 total_quantity
=> $total_quantity,
671 my $allow_onshelf_holds;
672 if ( not $viewallitems and @items > $max_items_to_display ) {
675 items_count
=> scalar( @items ),
678 for my $itm (@items) {
679 my $item = Koha
::Items
->find( $itm->{itemnumber
} );
680 $itm->{holds_count
} = $item_reserves{ $itm->{itemnumber
} };
681 $itm->{priority
} = $priority{ $itm->{itemnumber
} };
684 && !$itm->{'withdrawn'}
685 && !$itm->{'itemlost'}
686 && ($itm->{'itemnotforloan'}<0 || not $itm->{'itemnotforloan'})
687 && !$itemtypes->{$itm->{'itype'}}->{notforloan
}
688 && $itm->{'itemnumber'};
690 $allow_onshelf_holds = Koha
::IssuingRules
->get_onshelfholds_policy( { item
=> $item, patron
=> $patron } )
691 unless $allow_onshelf_holds;
693 # get collection code description, too
694 my $ccode = $itm->{'ccode'};
695 $itm->{'ccode'} = $collections->{$ccode} if defined($ccode) && $collections && exists( $collections->{$ccode} );
696 my $copynumber = $itm->{'copynumber'};
697 $itm->{'copynumber'} = $copynumbers->{$copynumber} if ( defined($copynumbers) && defined($copynumber) && exists( $copynumbers->{$copynumber} ) );
698 if ( defined $itm->{'location'} ) {
699 $itm->{'location_description'} = $shelflocations->{ $itm->{'location'} };
701 if (exists $itm->{itype
} && defined($itm->{itype
}) && exists $itemtypes->{ $itm->{itype
} }) {
702 $itm->{'imageurl'} = getitemtypeimagelocation
( 'opac', $itemtypes->{ $itm->{itype
} }->{'imageurl'} );
703 $itm->{'description'} = $itemtypes->{ $itm->{itype
} }->{translated_description
};
705 foreach (qw(ccode enumchron copynumber itemnotes location_description uri)) {
706 $itemfields{$_} = 1 if ($itm->{$_});
709 my $reserve_status = C4
::Reserves
::GetReserveStatus
($itm->{itemnumber
});
710 if( $reserve_status eq "Waiting"){ $itm->{'waiting'} = 1; }
711 if( $reserve_status eq "Reserved"){ $itm->{'onhold'} = 1; }
713 my ( $transfertwhen, $transfertfrom, $transfertto ) = GetTransfers
($itm->{itemnumber
});
714 if ( defined( $transfertwhen ) && $transfertwhen ne '' ) {
715 $itm->{transfertwhen
} = $transfertwhen;
716 $itm->{transfertfrom
} = $transfertfrom;
717 $itm->{transfertto
} = $transfertto;
720 if ( C4
::Context
->preference('OPACAcquisitionDetails') ) {
722 if grep /^$itm->{itemnumber}$/, @itemnumbers_on_order;
725 my $itembranch = $itm->{$separatebranch};
726 if ($currentbranch and C4
::Context
->preference('OpacSeparateHoldings')) {
727 if ($itembranch and $itembranch eq $currentbranch) {
728 push @itemloop, $itm;
730 push @otheritemloop, $itm;
733 push @itemloop, $itm;
738 if( $allow_onshelf_holds || CountItemsIssued
($biblionumber) || $biblio->has_items_waiting_or_intransit ) {
739 $template->param( ReservableItems
=> 1 );
742 # Display only one tab if one items list is empty
743 if (scalar(@itemloop) == 0 || scalar(@otheritemloop) == 0) {
744 $template->param(SeparateHoldings
=> 0);
745 if (scalar(@itemloop) == 0) {
746 @itemloop = @otheritemloop;
750 ## get notes and subjects from MARC record
751 if (!C4
::Context
->preference("OPACXSLTDetailsDisplay") ) {
752 my $marcisbnsarray = GetMarcISBN
($record,$marcflavour);
753 my $marcauthorsarray = GetMarcAuthors
($record,$marcflavour);
754 my $marcsubjctsarray = GetMarcSubjects
($record,$marcflavour);
755 my $marcseriesarray = GetMarcSeries
($record,$marcflavour);
756 my $marcurlsarray = GetMarcUrls
($record,$marcflavour);
757 my $marchostsarray = GetMarcHosts
($record,$marcflavour);
760 MARCSUBJCTS
=> $marcsubjctsarray,
761 MARCAUTHORS
=> $marcauthorsarray,
762 MARCSERIES
=> $marcseriesarray,
763 MARCURLS
=> $marcurlsarray,
764 MARCISBNS
=> $marcisbnsarray,
765 MARCHOSTS
=> $marchostsarray,
769 my $marcnotesarray = GetMarcNotes
($record,$marcflavour);
770 my $subtitle = GetRecordValue
('subtitle', $record, GetFrameworkCode
($biblionumber));
772 if( C4
::Context
->preference('ArticleRequests') ) {
773 my $patron = $borrowernumber ? Koha
::Patrons
->find($borrowernumber) : undef;
774 my $itemtype = Koha
::ItemTypes
->find($biblio->itemtype);
775 my $artreqpossible = $patron
776 ?
$biblio->can_article_request( $patron )
778 ?
$itemtype->may_article_request
780 $template->param( artreqpossible
=> $artreqpossible );
784 MARCNOTES
=> $marcnotesarray,
785 norequests
=> $norequests,
786 RequestOnOpac
=> C4
::Context
->preference("RequestOnOpac"),
787 itemdata_ccode
=> $itemfields{ccode
},
788 itemdata_enumchron
=> $itemfields{enumchron
},
789 itemdata_uri
=> $itemfields{uri
},
790 itemdata_copynumber
=> $itemfields{copynumber
},
791 itemdata_itemnotes
=> $itemfields{itemnotes
},
792 itemdata_location
=> $itemfields{location_description
},
793 subtitle
=> $subtitle,
794 OpacStarRatings
=> C4
::Context
->preference("OpacStarRatings"),
797 if (C4
::Context
->preference("AlternateHoldingsField") && scalar @items == 0) {
798 my $fieldspec = C4
::Context
->preference("AlternateHoldingsField");
799 my $subfields = substr $fieldspec, 3;
800 my $holdingsep = C4
::Context
->preference("AlternateHoldingsSeparator") || ' ';
801 my @alternateholdingsinfo = ();
802 my @holdingsfields = $record->field(substr $fieldspec, 0, 3);
804 for my $field (@holdingsfields) {
805 my %holding = ( holding
=> '' );
806 my $havesubfield = 0;
807 for my $subfield ($field->subfields()) {
808 if ((index $subfields, $$subfield[0]) >= 0) {
809 $holding{'holding'} .= $holdingsep if (length $holding{'holding'} > 0);
810 $holding{'holding'} .= $$subfield[1];
815 push(@alternateholdingsinfo, \
%holding);
820 ALTERNATEHOLDINGS
=> \
@alternateholdingsinfo,
824 # FIXME: The template uses this hash directly. Need to filter.
825 foreach ( keys %{$dat} ) {
826 next if ( $HideMARC->{$_} );
827 $template->param( "$_" => defined $dat->{$_} ?
$dat->{$_} : '' );
830 # some useful variables for enhanced content;
831 # in each case, we're grabbing the first value we find in
832 # the record and normalizing it
833 my $upc = GetNormalizedUPC
($record,$marcflavour);
834 my $oclc = GetNormalizedOCLCNumber
($record,$marcflavour);
835 my $isbn = GetNormalizedISBN
(undef,$record,$marcflavour);
836 my $content_identifier_exists;
837 if ( $isbn or $ean or $oclc or $upc ) {
838 $content_identifier_exists = 1;
841 normalized_upc
=> $upc,
842 normalized_ean
=> $ean,
843 normalized_oclc
=> $oclc,
844 normalized_isbn
=> $isbn,
845 content_identifier_exists
=> $content_identifier_exists,
848 # Catch the exception as Koha::Biblio::Metadata->record can explode if the MARCXML is invalid
849 # COinS format FIXME: for books Only
850 my $coins = eval { $biblio->get_coins };
851 $template->param( ocoins
=> $coins );
853 my ( $loggedincommenter, $reviews );
854 if ( C4
::Context
->preference('reviewson') ) {
855 $reviews = Koha
::Reviews
->search(
857 biblionumber
=> $biblionumber,
858 -or => { approved
=> 1, borrowernumber
=> $borrowernumber }
861 order_by
=> { -desc
=> 'datereviewed' }
864 my $libravatar_enabled = 0;
865 if ( C4
::Context
->preference('ShowReviewer') and C4
::Context
->preference('ShowReviewerPhoto') ) {
867 require Libravatar
::URL
;
868 Libravatar
::URL
->import();
871 $libravatar_enabled = 1;
874 for my $review (@
$reviews) {
875 my $review_patron = Koha
::Patrons
->find( $review->{borrowernumber
} ); # FIXME Should be Koha::Review->reviewer or similar
877 # setting some borrower info into this hash
878 if ( $review_patron ) {
879 $review->{patron
} = $review_patron;
880 if ( $libravatar_enabled and $review_patron->email ) {
881 $review->{avatarurl
} = libravatar_url
( email
=> $review_patron->email, https
=> $ENV{HTTPS
} );
884 if ( $review_patron->borrowernumber eq $borrowernumber ) {
885 $loggedincommenter = 1;
891 if ( C4
::Context
->preference("OPACISBD") ) {
892 $template->param( ISBD
=> 1 );
896 itemloop
=> \
@itemloop,
897 otheritemloop
=> \
@otheritemloop,
898 biblionumber
=> $biblionumber,
899 subscriptions
=> \
@subs,
900 subscriptionsnumber
=> $subscriptionsnumber,
902 loggedincommenter
=> $loggedincommenter
906 if (C4
::Context
->preference("virtualshelves") ) {
907 my $shelves = Koha
::Virtualshelves
->search(
909 biblionumber
=> $biblionumber,
913 join => 'virtualshelfcontents',
916 $template->param( shelves
=> $shelves );
920 if (C4
::Context
->preference("OPACFRBRizeEditions")==1) {
923 XISBNS
=> scalar get_xisbns
($isbn, $biblionumber)
926 if ($@
) { warn "XISBN Failed $@"; }
930 my @sc_fields = $record->field(955);
931 my @lc_fields = $marcflavour eq 'UNIMARC'
932 ?
$record->field(930)
933 : $record->field(852);
934 my @serialcollections = ();
936 foreach my $sc_field (@sc_fields) {
939 $row_data{text
} = $sc_field->subfield('r');
940 $row_data{branch
} = $sc_field->subfield('9');
941 foreach my $lc_field (@lc_fields) {
942 $row_data{itemcallnumber
} = $marcflavour eq 'UNIMARC'
943 ?
$lc_field->subfield('a') # 930$a
944 : $lc_field->subfield('h') # 852$h
945 if ($sc_field->subfield('5') eq $lc_field->subfield('5'));
948 if ($row_data{text
} && $row_data{branch
}) {
949 push (@serialcollections, \
%row_data);
953 if (scalar(@serialcollections) > 0) {
955 serialcollection
=> 1,
956 serialcollections
=> \
@serialcollections);
959 # Local cover Images stuff
960 if (C4
::Context
->preference("OPACLocalCoverImages")){
961 $template->param(OPACLocalCoverImages
=> 1);
965 if ( (C4
::Context
->preference("HTML5MediaEnabled") eq 'both') or (C4
::Context
->preference("HTML5MediaEnabled") eq 'opac') ) {
966 $template->param( C4
::HTML5Media
->gethtml5media($record));
969 my $syndetics_elements;
971 if ( C4
::Context
->preference("SyndeticsEnabled") ) {
972 $template->param("SyndeticsEnabled" => 1);
973 $template->param("SyndeticsClientCode" => C4
::Context
->preference("SyndeticsClientCode"));
975 $syndetics_elements = &get_syndetics_index
($isbn,$upc,$oclc);
976 for my $element (values %$syndetics_elements) {
977 $template->param("Syndetics$element"."Exists" => 1 );
978 #warn "Exists: "."Syndetics$element"."Exists";
984 if ( C4
::Context
->preference("SyndeticsEnabled")
985 && C4
::Context
->preference("SyndeticsSummary")
986 && ( exists($syndetics_elements->{'SUMMARY'}) || exists($syndetics_elements->{'AVSUMMARY'}) ) ) {
988 my $syndetics_summary = &get_syndetics_summary
($isbn,$upc,$oclc, $syndetics_elements);
989 $template->param( SYNDETICS_SUMMARY
=> $syndetics_summary );
995 if ( C4
::Context
->preference("SyndeticsEnabled")
996 && C4
::Context
->preference("SyndeticsTOC")
997 && exists($syndetics_elements->{'TOC'}) ) {
999 my $syndetics_toc = &get_syndetics_toc
($isbn,$upc,$oclc);
1000 $template->param( SYNDETICS_TOC
=> $syndetics_toc );
1005 if ( C4
::Context
->preference("SyndeticsEnabled")
1006 && C4
::Context
->preference("SyndeticsExcerpt")
1007 && exists($syndetics_elements->{'DBCHAPTER'}) ) {
1009 my $syndetics_excerpt = &get_syndetics_excerpt
($isbn,$upc,$oclc);
1010 $template->param( SYNDETICS_EXCERPT
=> $syndetics_excerpt );
1015 if ( C4
::Context
->preference("SyndeticsEnabled")
1016 && C4
::Context
->preference("SyndeticsReviews")) {
1018 my $syndetics_reviews = &get_syndetics_reviews
($isbn,$upc,$oclc,$syndetics_elements);
1019 $template->param( SYNDETICS_REVIEWS
=> $syndetics_reviews );
1024 if ( C4
::Context
->preference("SyndeticsEnabled")
1025 && C4
::Context
->preference("SyndeticsAuthorNotes")
1026 && exists($syndetics_elements->{'ANOTES'}) ) {
1028 my $syndetics_anotes = &get_syndetics_anotes
($isbn,$upc,$oclc);
1029 $template->param( SYNDETICS_ANOTES
=> $syndetics_anotes );
1034 # LibraryThingForLibraries ID Code and Tabbed View Option
1035 if( C4
::Context
->preference('LibraryThingForLibrariesEnabled') )
1037 $template->param(LibraryThingForLibrariesID
=>
1038 C4
::Context
->preference('LibraryThingForLibrariesID') );
1039 $template->param(LibraryThingForLibrariesTabbedView
=>
1040 C4
::Context
->preference('LibraryThingForLibrariesTabbedView') );
1044 if( C4
::Context
->preference('NovelistSelectEnabled') )
1046 $template->param(NovelistSelectProfile
=> C4
::Context
->preference('NovelistSelectProfile') );
1047 $template->param(NovelistSelectPassword
=> C4
::Context
->preference('NovelistSelectPassword') );
1048 $template->param(NovelistSelectView
=> C4
::Context
->preference('NovelistSelectView') );
1053 if ( C4
::Context
->preference("Babeltheque") ) {
1056 Babeltheque_url_js
=> C4
::Context
->preference("Babeltheque_url_js"),
1061 if ( C4
::Context
->preference( "SocialNetworks" ) ) {
1062 $template->param( current_url
=> C4
::Context
->preference('OPACBaseURL') . "/cgi-bin/koha/opac-detail.pl?biblionumber=$biblionumber" );
1063 $template->param( SocialNetworks
=> 1 );
1066 # Shelf Browser Stuff
1067 if (C4
::Context
->preference("OPACShelfBrowser")) {
1068 my $starting_itemnumber = $query->param('shelfbrowse_itemnumber');
1069 if (defined($starting_itemnumber)) {
1070 $template->param( OpenOPACShelfBrowser
=> 1) if $starting_itemnumber;
1071 my $nearby = GetNearbyItems
($starting_itemnumber);
1074 starting_itemnumber
=> $starting_itemnumber,
1075 starting_homebranch
=> $nearby->{starting_homebranch
}->{description
},
1076 starting_location
=> $nearby->{starting_location
}->{description
},
1077 starting_ccode
=> $nearby->{starting_ccode
}->{description
},
1078 shelfbrowser_prev_item
=> $nearby->{prev_item
},
1079 shelfbrowser_next_item
=> $nearby->{next_item
},
1080 shelfbrowser_items
=> $nearby->{items
},
1083 # in which tab shelf browser should open ?
1084 if (grep { $starting_itemnumber == $_->{itemnumber
} } @itemloop) {
1085 $template->param(shelfbrowser_tab
=> 'holdings');
1087 $template->param(shelfbrowser_tab
=> 'otherholdings');
1092 $template->param( AmazonTld
=> get_amazon_tld
() ) if ( C4
::Context
->preference("OPACAmazonCoverImages"));
1094 if (C4
::Context
->preference("BakerTaylorEnabled")) {
1096 BakerTaylorEnabled
=> 1,
1097 BakerTaylorImageURL
=> &image_url
(),
1098 BakerTaylorLinkURL
=> &link_url
(),
1099 BakerTaylorBookstoreURL
=> C4
::Context
->preference('BakerTaylorBookstoreURL'),
1101 my ($bt_user, $bt_pass);
1103 $bt_user = C4
::Context
->preference('BakerTaylorUsername') and
1104 $bt_pass = C4
::Context
->preference('BakerTaylorPassword') )
1107 BakerTaylorContentURL
=>
1108 sprintf("https://contentcafe2.btol.com/ContentCafeClient/ContentCafe.aspx?UserID=%s&Password=%s&ItemKey=%s&Options=Y",
1109 $bt_user,$bt_pass,$isbn)
1115 if (C4
::Context
->preference('TagsEnabled') and $tag_quantity = C4
::Context
->preference('TagsShowOnDetail')) {
1118 TagsShowOnDetail
=> $tag_quantity,
1119 TagsInputOnDetail
=> C4
::Context
->preference('TagsInputOnDetail')
1121 $template->param(TagLoop
=> get_tags
({biblionumber
=>$biblionumber, approved
=>1,
1122 'sort'=>'-weight', limit
=>$tag_quantity}));
1125 if (C4
::Context
->preference("OPACURLOpenInNewWindow")) {
1126 # These values are going to be read by Javascript, at least in the case
1127 # of the google covers
1128 $template->param(covernewwindow
=> 'true');
1130 $template->param(covernewwindow
=> 'false');
1133 $template->param(borrowernumber
=> $borrowernumber);
1135 if ( C4
::Context
->preference('OpacStarRatings') !~ /disable/ ) {
1136 my $ratings = Koha
::Ratings
->search({ biblionumber
=> $biblionumber });
1137 my $my_rating = $borrowernumber ?
$ratings->search({ borrowernumber
=> $borrowernumber })->next : undef;
1139 ratings
=> $ratings,
1140 my_rating
=> $my_rating,
1144 #Search for title in links
1145 my $marccontrolnumber = GetMarcControlnumber
($record, $marcflavour);
1146 my $marcissns = GetMarcISSN
( $record, $marcflavour );
1147 my $issn = $marcissns->[0] || '';
1149 if (my $search_for_title = C4
::Context
->preference('OPACSearchForTitleIn')){
1150 $dat->{title
} =~ s/\/+$//; # remove trailing slash
1151 $dat->{title
} =~ s/\s+$//; # remove trailing space
1152 $search_for_title = parametrized_url
(
1155 TITLE
=> $dat->{title
},
1156 AUTHOR
=> $dat->{author
},
1159 CONTROLNUMBER
=> $marccontrolnumber,
1160 BIBLIONUMBER
=> $biblionumber,
1163 $template->param('OPACSearchForTitleIn' => $search_for_title);
1167 if ( C4
::Context
->preference("IDREF") ) {
1168 # If the record comes from the SUDOC
1169 if ( $record->field('009') ) {
1170 my $unimarc3 = $record->field("009")->data;
1171 if ( $unimarc3 =~ /^\d+$/ ) {
1179 # We try to select the best default tab to show, according to what
1180 # the user wants, and what's available for display
1181 my $opac_serial_default = C4
::Context
->preference('opacSerialDefaultTab');
1185 $opac_serial_default eq 'subscriptions' && $subscriptionsnumber
1187 $opac_serial_default eq 'serialcollection' && @serialcollections > 0
1188 ?
'serialcollection' :
1189 $opac_serial_default eq 'holdings' && scalar (@itemloop) > 0
1191 scalar (@itemloop) == 0
1193 $subscriptionsnumber
1195 @serialcollections > 0
1196 ?
'serialcollection' : 'subscriptions';
1197 $template->param('defaulttab' => $defaulttab);
1199 if (C4
::Context
->preference('OPACLocalCoverImages') == 1) {
1200 my @images = ListImagesForBiblio
($biblionumber);
1201 $template->{VARS
}->{localimages
} = \
@images;
1204 $template->{VARS
}->{IDreamBooksReviews
} = C4
::Context
->preference('IDreamBooksReviews');
1205 $template->{VARS
}->{IDreamBooksReadometer
} = C4
::Context
->preference('IDreamBooksReadometer');
1206 $template->{VARS
}->{IDreamBooksResults
} = C4
::Context
->preference('IDreamBooksResults');
1207 $template->{VARS
}->{OPACPopupAuthorsSearch
} = C4
::Context
->preference('OPACPopupAuthorsSearch');
1209 if (C4
::Context
->preference('OpacHighlightedWords')) {
1210 $template->{VARS
}->{query_desc
} = $query->param('query_desc');
1212 $template->{VARS
}->{'trackclicks'} = C4
::Context
->preference('TrackClicks');
1214 if ( C4
::Context
->preference('UseCourseReserves') ) {
1215 foreach my $i ( @items ) {
1216 $i->{'course_reserves'} = GetItemCourseReservesInfo
( itemnumber
=> $i->{'itemnumber'} );
1221 'OpacLocationBranchToDisplay' => C4
::Context
->preference('OpacLocationBranchToDisplay'),
1224 output_html_with_http_headers
$query, $cookie, $template->output;