Main Koha release repository https://koha-community.org
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1093 lines
31 KiB

package Koha::Item;
# Copyright ByWater Solutions 2014
#
# This file is part of Koha.
#
# Koha is free software; you can redistribute it and/or modify it
# under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# Koha is distributed in the hope that it will be useful, but
# WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Koha; if not, see <http://www.gnu.org/licenses>.
use Modern::Perl;
use Carp;
use List::MoreUtils qw(any);
use Data::Dumper;
use Try::Tiny;
use Koha::Database;
use Koha::DateUtils qw( dt_from_string );
use C4::Context;
use C4::Circulation;
use C4::Reserves;
use C4::ClassSource; # FIXME We would like to avoid that
use C4::Log qw( logaction );
use Koha::Checkouts;
use Koha::CirculationRules;
use Koha::CoverImages;
use Koha::SearchEngine::Indexer;
use Koha::Item::Transfer::Limits;
use Koha::Item::Transfers;
use Koha::ItemTypes;
use Koha::Patrons;
use Koha::Plugins;
use Koha::Libraries;
use Koha::StockRotationItem;
use Koha::StockRotationRotas;
use base qw(Koha::Object);
=head1 NAME
Koha::Item - Koha Item object class
=head1 API
=head2 Class methods
=cut
=head3 store
$item->store;
$params can take an optional 'skip_record_index' parameter.
If set, the reindexation process will not happen (index_records not called)
NOTE: This is a temporary fix to answer a performance issue when lot of items
are added (or modified) at the same time.
The correct way to fix this is to make the ES reindexation process async.
You should not turn it on if you do not understand what it is doing exactly.
=cut
sub store {
my $self = shift;
my $params = @_ ? shift : {};
my $log_action = $params->{log_action} // 1;
# We do not want to oblige callers to pass this value
# Dev conveniences vs performance?
unless ( $self->biblioitemnumber ) {
$self->biblioitemnumber( $self->biblio->biblioitem->biblioitemnumber );
}
# See related changes from C4::Items::AddItem
unless ( $self->itype ) {
$self->itype($self->biblio->biblioitem->itemtype);
}
my $today = dt_from_string;
my $action = 'create';
unless ( $self->in_storage ) { #AddItem
unless ( $self->permanent_location ) {
$self->permanent_location($self->location);
}
unless ( $self->replacementpricedate ) {
$self->replacementpricedate($today);
}
unless ( $self->datelastseen ) {
$self->datelastseen($today);
}
unless ( $self->dateaccessioned ) {
$self->dateaccessioned($today);
}
if ( $self->itemcallnumber
or $self->cn_source )
{
my $cn_sort = GetClassSort( $self->cn_source, $self->itemcallnumber, "" );
$self->cn_sort($cn_sort);
}
} else { # ModItem
$action = 'modify';
my %updated_columns = $self->_result->get_dirty_columns;
return $self->SUPER::store unless %updated_columns;
# Retrieve the item for comparison if we need to
my $pre_mod_item = (
exists $updated_columns{itemlost}
or exists $updated_columns{withdrawn}
or exists $updated_columns{damaged}
) ? $self->get_from_storage : undef;
# Update *_on fields if needed
# FIXME: Why not for AddItem as well?
my @fields = qw( itemlost withdrawn damaged );
for my $field (@fields) {
# If the field is defined but empty or 0, we are
# removing/unsetting and thus need to clear out
# the 'on' field
if ( exists $updated_columns{$field}
&& defined( $self->$field )
&& !$self->$field )
{
my $field_on = "${field}_on";
$self->$field_on(undef);
}
# If the field has changed otherwise, we much update
# the 'on' field
elsif (exists $updated_columns{$field}
&& $updated_columns{$field}
&& !$pre_mod_item->$field )
{
my $field_on = "${field}_on";
$self->$field_on(
DateTime::Format::MySQL->format_datetime(
dt_from_string()
)
);
}
}
if ( exists $updated_columns{itemcallnumber}
or exists $updated_columns{cn_source} )
{
my $cn_sort = GetClassSort( $self->cn_source, $self->itemcallnumber, "" );
$self->cn_sort($cn_sort);
}
if ( exists $updated_columns{location}
and $self->location ne 'CART'
and $self->location ne 'PROC'
and not exists $updated_columns{permanent_location} )
{
$self->permanent_location( $self->location );
}
# If item was lost and has now been found,
# reverse any list item charges if necessary.
if ( exists $updated_columns{itemlost}
and $updated_columns{itemlost} <= 0
and $pre_mod_item->itemlost > 0 )
{
$self->_set_found_trigger($pre_mod_item);
}
}
unless ( $self->dateaccessioned ) {
$self->dateaccessioned($today);
}
my $result = $self->SUPER::store;
if ( $log_action && C4::Context->preference("CataloguingLog") ) {
$action eq 'create'
? logaction( "CATALOGUING", "ADD", $self->itemnumber, "item" )
: logaction( "CATALOGUING", "MODIFY", $self->itemnumber, "item " . Dumper( $self->unblessed ) );
}
my $indexer = Koha::SearchEngine::Indexer->new({ index => $Koha::SearchEngine::BIBLIOS_INDEX });
$indexer->index_records( $self->biblionumber, "specialUpdate", "biblioserver" )
unless $params->{skip_record_index};
$self->get_from_storage->_after_item_action_hooks({ action => $action });
return $result;
}
=head3 delete
=cut
sub delete {
my $self = shift;
my $params = @_ ? shift : {};
# FIXME check the item has no current issues
# i.e. raise the appropriate exception
my $result = $self->SUPER::delete;
my $indexer = Koha::SearchEngine::Indexer->new({ index => $Koha::SearchEngine::BIBLIOS_INDEX });
$indexer->index_records( $self->biblionumber, "specialUpdate", "biblioserver" )
unless $params->{skip_record_index};
$self->_after_item_action_hooks({ action => 'delete' });
logaction( "CATALOGUING", "DELETE", $self->itemnumber, "item" )
if C4::Context->preference("CataloguingLog");
return $result;
}
=head3 safe_delete
=cut
sub safe_delete {
my $self = shift;
my $params = @_ ? shift : {};
my $safe_to_delete = $self->safe_to_delete;
return $safe_to_delete unless $safe_to_delete eq '1';
$self->move_to_deleted;
return $self->delete($params);
}
=head3 safe_to_delete
returns 1 if the item is safe to delete,
"book_on_loan" if the item is checked out,
"not_same_branch" if the item is blocked by independent branches,
"book_reserved" if the there are holds aganst the item, or
"linked_analytics" if the item has linked analytic records.
"last_item_for_hold" if the item is the last one on a record on which a biblio-level hold is placed
=cut
sub safe_to_delete {
my ($self) = @_;
return "book_on_loan" if $self->checkout;
return "not_same_branch"
if defined C4::Context->userenv
and !C4::Context->IsSuperLibrarian()
and C4::Context->preference("IndependentBranches")
and ( C4::Context->userenv->{branch} ne $self->homebranch );
# check it doesn't have a waiting reserve
return "book_reserved"
if $self->holds->search( { found => [ 'W', 'T' ] } )->count;
return "linked_analytics"
if C4::Items::GetAnalyticsCount( $self->itemnumber ) > 0;
return "last_item_for_hold"
if $self->biblio->items->count == 1
&& $self->biblio->holds->search(
{
itemnumber => undef,
}
)->count;
return 1;
}
=head3 move_to_deleted
my $is_moved = $item->move_to_deleted;
Move an item to the deleteditems table.
This can be done before deleting an item, to make sure the data are not completely deleted.
=cut
sub move_to_deleted {
my ($self) = @_;
my $item_infos = $self->unblessed;
delete $item_infos->{timestamp}; #This ensures the timestamp date in deleteditems will be set to the current timestamp
return Koha::Database->new->schema->resultset('Deleteditem')->create($item_infos);
}
=head3 effective_itemtype
Returns the itemtype for the item based on whether item level itemtypes are set or not.
=cut
sub effective_itemtype {
my ( $self ) = @_;
return $self->_result()->effective_itemtype();
}
=head3 home_branch
=cut
sub home_branch {
my ($self) = @_;
$self->{_home_branch} ||= Koha::Libraries->find( $self->homebranch() );
return $self->{_home_branch};
}
=head3 holding_branch
=cut
sub holding_branch {
my ($self) = @_;
$self->{_holding_branch} ||= Koha::Libraries->find( $self->holdingbranch() );
return $self->{_holding_branch};
}
=head3 biblio
my $biblio = $item->biblio;
Return the bibliographic record of this item
=cut
sub biblio {
my ( $self ) = @_;
my $biblio_rs = $self->_result->biblio;
return Koha::Biblio->_new_from_dbic( $biblio_rs );
}
=head3 biblioitem
my $biblioitem = $item->biblioitem;
Return the biblioitem record of this item
=cut
sub biblioitem {
my ( $self ) = @_;
my $biblioitem_rs = $self->_result->biblioitem;
return Koha::Biblioitem->_new_from_dbic( $biblioitem_rs );
}
=head3 checkout
my $checkout = $item->checkout;
Return the checkout for this item
=cut
sub checkout {
my ( $self ) = @_;
my $checkout_rs = $self->_result->issue;
return unless $checkout_rs;
return Koha::Checkout->_new_from_dbic( $checkout_rs );
}
=head3 holds
my $holds = $item->holds();
my $holds = $item->holds($params);
my $holds = $item->holds({ found => 'W'});
Return holds attached to an item, optionally accept a hashref of params to pass to search
=cut
sub holds {
my ( $self,$params ) = @_;
my $holds_rs = $self->_result->reserves->search($params);
return Koha::Holds->_new_from_dbic( $holds_rs );
}
=head3 get_transfer
my $transfer = $item->get_transfer;
Return the transfer if the item is in transit or undef
=cut
sub get_transfer {
my ( $self ) = @_;
my $transfer_rs = $self->_result->branchtransfers->search({ datearrived => undef })->first;
return unless $transfer_rs;
return Koha::Item::Transfer->_new_from_dbic( $transfer_rs );
}
=head3 last_returned_by
Gets and sets the last borrower to return an item.
Accepts and returns Koha::Patron objects
$item->last_returned_by( $borrowernumber );
$last_returned_by = $item->last_returned_by();
=cut
sub last_returned_by {
my ( $self, $borrower ) = @_;
my $items_last_returned_by_rs = Koha::Database->new()->schema()->resultset('ItemsLastBorrower');
if ($borrower) {
return $items_last_returned_by_rs->update_or_create(
{ borrowernumber => $borrower->borrowernumber, itemnumber => $self->id } );
}
else {
unless ( $self->{_last_returned_by} ) {
my $result = $items_last_returned_by_rs->single( { itemnumber => $self->id } );
if ($result) {
$self->{_last_returned_by} = Koha::Patrons->find( $result->get_column('borrowernumber') );
}
}
return $self->{_last_returned_by};
}
}
=head3 can_article_request
my $bool = $item->can_article_request( $borrower )
Returns true if item can be specifically requested
$borrower must be a Koha::Patron object
=cut
sub can_article_request {
my ( $self, $borrower ) = @_;
my $rule = $self->article_request_type($borrower);
return 1 if $rule && $rule ne 'no' && $rule ne 'bib_only';
return q{};
}
=head3 hidden_in_opac
my $bool = $item->hidden_in_opac({ [ rules => $rules ] })
Returns true if item fields match the hidding criteria defined in $rules.
Returns false otherwise.
Takes HASHref that can have the following parameters:
OPTIONAL PARAMETERS:
$rules : { <field> => [ value_1, ... ], ... }
Note: $rules inherits its structure from the parsed YAML from reading
the I<OpacHiddenItems> system preference.
=cut
sub hidden_in_opac {
my ( $self, $params ) = @_;
my $rules = $params->{rules} // {};
return 1
if C4::Context->preference('hidelostitems') and
$self->itemlost > 0;
my $hidden_in_opac = 0;
foreach my $field ( keys %{$rules} ) {
if ( any { $self->$field eq $_ } @{ $rules->{$field} } ) {
$hidden_in_opac = 1;
last;
}
}
return $hidden_in_opac;
}
=head3 can_be_transferred
$item->can_be_transferred({ to => $to_library, from => $from_library })
Checks if an item can be transferred to given library.
This feature is controlled by two system preferences:
UseBranchTransferLimits to enable / disable the feature
BranchTransferLimitsType to use either an itemnumber or ccode as an identifier
for setting the limitations
Takes HASHref that can have the following parameters:
MANDATORY PARAMETERS:
$to : Koha::Library
OPTIONAL PARAMETERS:
$from : Koha::Library # if not given, item holdingbranch
# will be used instead
Returns 1 if item can be transferred to $to_library, otherwise 0.
To find out whether at least one item of a Koha::Biblio can be transferred, please
see Koha::Biblio->can_be_transferred() instead of using this method for
multiple items of the same biblio.
=cut
sub can_be_transferred {
my ($self, $params) = @_;
my $to = $params->{to};
my $from = $params->{from};
$to = $to->branchcode;
$from = defined $from ? $from->branchcode : $self->holdingbranch;
return 1 if $from eq $to; # Transfer to current branch is allowed
return 1 unless C4::Context->preference('UseBranchTransferLimits');
my $limittype = C4::Context->preference('BranchTransferLimitsType');
return Koha::Item::Transfer::Limits->search({
toBranch => $to,
fromBranch => $from,
$limittype => $limittype eq 'itemtype'
? $self->effective_itemtype : $self->ccode
})->count ? 0 : 1;
}
=head3 pickup_locations
$pickup_locations = $item->pickup_locations( {patron => $patron } )
Returns possible pickup locations for this item, according to patron's home library (if patron is defined and holds are allowed only from hold groups)
and if item can be transferred to each pickup location.
=cut
sub pickup_locations {
my ($self, $params) = @_;
my $patron = $params->{patron};
my $circ_control_branch =
C4::Reserves::GetReservesControlBranch( $self->unblessed(), $patron->unblessed );
my $branchitemrule =
C4::Circulation::GetBranchItemRule( $circ_control_branch, $self->itype );
if(defined $patron) {
return Koha::Libraries->new()->empty if $branchitemrule->{holdallowed} == 3 && !$self->home_branch->validate_hold_sibling( {branchcode => $patron->branchcode} );
return Koha::Libraries->new()->empty if $branchitemrule->{holdallowed} == 1 && $self->home_branch->branchcode ne $patron->branchcode;
}
my $pickup_libraries = Koha::Libraries->search();
if ($branchitemrule->{hold_fulfillment_policy} eq 'holdgroup') {
$pickup_libraries = $self->home_branch->get_hold_libraries;
} elsif ($branchitemrule->{hold_fulfillment_policy} eq 'patrongroup') {
my $plib = Koha::Libraries->find({ branchcode => $patron->branchcode});
$pickup_libraries = $plib->get_hold_libraries;
} elsif ($branchitemrule->{hold_fulfillment_policy} eq 'homebranch') {
$pickup_libraries = Koha::Libraries->search({ branchcode => $self->homebranch });
} elsif ($branchitemrule->{hold_fulfillment_policy} eq 'holdingbranch') {
$pickup_libraries = Koha::Libraries->search({ branchcode => $self->holdingbranch });
};
return $pickup_libraries->search(
{
pickup_location => 1
},
{
order_by => ['branchname']
}
) unless C4::Context->preference('UseBranchTransferLimits');
my $limittype = C4::Context->preference('BranchTransferLimitsType');
my ($ccode, $itype) = (undef, undef);
if( $limittype eq 'ccode' ){
$ccode = $self->ccode;
} else {
$itype = $self->itype;
}
my $limits = Koha::Item::Transfer::Limits->search(
{
fromBranch => $self->holdingbranch,
ccode => $ccode,
itemtype => $itype,
},
{ columns => ['toBranch'] }
);
return $pickup_libraries->search(
{
pickup_location => 1,
branchcode => {
'-not_in' => $limits->_resultset->as_query
}
},
{
order_by => ['branchname']
}
);
}
=head3 article_request_type
my $type = $item->article_request_type( $borrower )
returns 'yes', 'no', 'bib_only', or 'item_only'
$borrower must be a Koha::Patron object
=cut
sub article_request_type {
my ( $self, $borrower ) = @_;
my $branch_control = C4::Context->preference('HomeOrHoldingBranch');
my $branchcode =
$branch_control eq 'homebranch' ? $self->homebranch
: $branch_control eq 'holdingbranch' ? $self->holdingbranch
: undef;
my $borrowertype = $borrower->categorycode;
my $itemtype = $self->effective_itemtype();
my $rule = Koha::CirculationRules->get_effective_rule(
{
rule_name => 'article_requests',
categorycode => $borrowertype,
itemtype => $itemtype,
branchcode => $branchcode
}
);
return q{} unless $rule;
return $rule->rule_value || q{}
}
=head3 current_holds
=cut
sub current_holds {
my ( $self ) = @_;
my $attributes = { order_by => 'priority' };
my $dtf = Koha::Database->new->schema->storage->datetime_parser;
my $params = {
itemnumber => $self->itemnumber,
suspend => 0,
-or => [
reservedate => { '<=' => $dtf->format_date(dt_from_string) },
waitingdate => { '!=' => undef },
],
};
my $hold_rs = $self->_result->reserves->search( $params, $attributes );
return Koha::Holds->_new_from_dbic($hold_rs);
}
=head3 stockrotationitem
my $sritem = Koha::Item->stockrotationitem;
Returns the stock rotation item associated with the current item.
=cut
sub stockrotationitem {
my ( $self ) = @_;
my $rs = $self->_result->stockrotationitem;
return 0 if !$rs;
return Koha::StockRotationItem->_new_from_dbic( $rs );
}
=head3 add_to_rota
my $item = $item->add_to_rota($rota_id);
Add this item to the rota identified by $ROTA_ID, which means associating it
with the first stage of that rota. Should this item already be associated
with a rota, then we will move it to the new rota.
=cut
sub add_to_rota {
my ( $self, $rota_id ) = @_;
Koha::StockRotationRotas->find($rota_id)->add_item($self->itemnumber);
return $self;
}
=head3 has_pending_hold
my $is_pending_hold = $item->has_pending_hold();
This method checks the tmp_holdsqueue to see if this item has been selected for a hold, but not filled yet and returns true or false
=cut
sub has_pending_hold {
my ( $self ) = @_;
my $pending_hold = $self->_result->tmp_holdsqueues;
return $pending_hold->count ? 1: 0;
}
=head3 as_marc_field
my $mss = C4::Biblio::GetMarcSubfieldStructure( '', { unsafe => 1 } );
my $field = $item->as_marc_field({ [ mss => $mss ] });
This method returns a MARC::Field object representing the Koha::Item object
with the current mappings configuration.
=cut
sub as_marc_field {
my ( $self, $params ) = @_;
my $mss = $params->{mss} // C4::Biblio::GetMarcSubfieldStructure( '', { unsafe => 1 } );
my $item_tag = $mss->{'items.itemnumber'}[0]->{tagfield};
my @subfields;
my @columns = $self->_result->result_source->columns;
foreach my $item_field ( @columns ) {
my $mapping = $mss->{ "items.$item_field"}[0];
my $tagfield = $mapping->{tagfield};
my $tagsubfield = $mapping->{tagsubfield};
next if !$tagfield; # TODO: Should we raise an exception instead?
# Feels like safe fallback is better
push @subfields, $tagsubfield => $self->$item_field
if defined $self->$item_field and $item_field ne '';
}
my $unlinked_item_subfields = C4::Items::_parse_unlinked_item_subfields_from_xml($self->more_subfields_xml);
push( @subfields, @{$unlinked_item_subfields} )
if defined $unlinked_item_subfields and $#$unlinked_item_subfields > -1;
my $field;
$field = MARC::Field->new(
"$item_tag", ' ', ' ', @subfields
) if @subfields;
return $field;
}
=head3 renewal_branchcode
Returns the branchcode to be recorded in statistics renewal of the item
=cut
sub renewal_branchcode {
my ($self, $params ) = @_;
my $interface = C4::Context->interface;
my $branchcode;
if ( $interface eq 'opac' ){
my $renewal_branchcode = C4::Context->preference('OpacRenewalBranch');
if( !defined $renewal_branchcode || $renewal_branchcode eq 'opacrenew' ){
$branchcode = 'OPACRenew';
}
elsif ( $renewal_branchcode eq 'itemhomebranch' ) {
$branchcode = $self->homebranch;
}
elsif ( $renewal_branchcode eq 'patronhomebranch' ) {
$branchcode = $self->checkout->patron->branchcode;
}
elsif ( $renewal_branchcode eq 'checkoutbranch' ) {
$branchcode = $self->checkout->branchcode;
}
else {
$branchcode = "";
}
} else {
$branchcode = ( C4::Context->userenv && defined C4::Context->userenv->{branch} )
? C4::Context->userenv->{branch} : $params->{branch};
}
return $branchcode;
}
=head3 cover_images
Return the cover images associated with this item.
=cut
sub cover_images {
my ( $self ) = @_;
my $cover_image_rs = $self->_result->cover_images;
return unless $cover_image_rs;
return Koha::CoverImages->_new_from_dbic($cover_image_rs);
}
=head3 _set_found_trigger
$self->_set_found_trigger
Finds the most recent lost item charge for this item and refunds the patron
appropriately, taking into account any payments or writeoffs already applied
against the charge.
Internal function, not exported, called only by Koha::Item->store.
=cut
sub _set_found_trigger {
my ( $self, $pre_mod_item ) = @_;
## If item was lost, it has now been found, reverse any list item charges if necessary.
my $no_refund_after_days =
C4::Context->preference('NoRefundOnLostReturnedItemsAge');
if ($no_refund_after_days) {
my $today = dt_from_string();
my $lost_age_in_days =
dt_from_string( $pre_mod_item->itemlost_on )->delta_days($today)
->in_units('days');
return $self unless $lost_age_in_days < $no_refund_after_days;
}
my $lostreturn_policy = Koha::CirculationRules->get_lostreturn_policy(
{
item => $self,
return_branch => C4::Context->userenv
? C4::Context->userenv->{'branch'}
: undef,
}
);
if ( $lostreturn_policy ) {
# refund charge made for lost book
my $lost_charge = Koha::Account::Lines->search(
{
itemnumber => $self->itemnumber,
debit_type_code => 'LOST',
status => [ undef, { '<>' => 'FOUND' } ]
},
{
order_by => { -desc => [ 'date', 'accountlines_id' ] },
rows => 1
}
)->single;
if ( $lost_charge ) {
my $patron = $lost_charge->patron;
if ( $patron ) {
my $account = $patron->account;
my $total_to_refund = 0;
# Use cases
if ( $lost_charge->amount > $lost_charge->amountoutstanding ) {
# some amount has been cancelled. collect the offsets that are not writeoffs
# this works because the only way to subtract from this kind of a debt is
# using the UI buttons 'Pay' and 'Write off'
my $credits_offsets = Koha::Account::Offsets->search(
{
debit_id => $lost_charge->id,
credit_id => { '!=' => undef }, # it is not the debit itself
type => { '!=' => 'Writeoff' },
amount => { '<' => 0 } # credits are negative on the DB
}
);
$total_to_refund = ( $credits_offsets->count > 0 )
? $credits_offsets->total * -1 # credits are negative on the DB
: 0;
}
my $credit_total = $lost_charge->amountoutstanding + $total_to_refund;
my $credit;
if ( $credit_total > 0 ) {
my $branchcode =
C4::Context->userenv ? C4::Context->userenv->{'branch'} : undef;
$credit = $account->add_credit(
{
amount => $credit_total,
description => 'Item found ' . $self->itemnumber,
type => 'LOST_FOUND',
interface => C4::Context->interface,
library_id => $branchcode,
item_id => $self->itemnumber,
issue_id => $lost_charge->issue_id
}
);
$credit->apply( { debits => [$lost_charge] } );
$self->{_refunded} = 1;
}
# Update the account status
$lost_charge->status('FOUND');
$lost_charge->store();
# Reconcile balances if required
if ( C4::Context->preference('AccountAutoReconcile') ) {
$account->reconcile_balance;
}
}
}
# restore fine for lost book
if ( $lostreturn_policy eq 'restore' ) {
my $lost_overdue = Koha::Account::Lines->search(
{
itemnumber => $self->itemnumber,
debit_type_code => 'OVERDUE',
status => 'LOST'
},
{
order_by => { '-desc' => 'date' },
rows => 1
}
)->single;
if ( $lost_overdue ) {
my $patron = $lost_overdue->patron;
if ($patron) {
my $account = $patron->account;
# Update status of fine
$lost_overdue->status('FOUND')->store();
# Find related forgive credit
my $refund = $lost_overdue->credits(
{
credit_type_code => 'FORGIVEN',
itemnumber => $self->itemnumber,
status => [ { '!=' => 'VOID' }, undef ]
},
{ order_by => { '-desc' => 'date' }, rows => 1 }
)->single;
if ( $refund ) {
# Revert the forgive credit
$refund->void();
$self->{_restored} = 1;
}
# Reconcile balances if required
if ( C4::Context->preference('AccountAutoReconcile') ) {
$account->reconcile_balance;
}
}
}
} elsif ( $lostreturn_policy eq 'charge' ) {
$self->{_charge} = 1;
}
}
return $self;
}
=head3 to_api_mapping
This method returns the mapping for representing a Koha::Item object
on the API.
=cut
sub to_api_mapping {
return {
itemnumber => 'item_id',
biblionumber => 'biblio_id',
biblioitemnumber => undef,
barcode => 'external_id',
dateaccessioned => 'acquisition_date',
booksellerid => 'acquisition_source',
homebranch => 'home_library_id',
price => 'purchase_price',
replacementprice => 'replacement_price',
replacementpricedate => 'replacement_price_date',
datelastborrowed => 'last_checkout_date',
datelastseen => 'last_seen_date',
stack => undef,
notforloan => 'not_for_loan_status',
damaged => 'damaged_status',
damaged_on => 'damaged_date',
itemlost => 'lost_status',
itemlost_on => 'lost_date',
withdrawn => 'withdrawn',
withdrawn_on => 'withdrawn_date',
itemcallnumber => 'callnumber',
coded_location_qualifier => 'coded_location_qualifier',
issues => 'checkouts_count',
renewals => 'renewals_count',
reserves => 'holds_count',
restricted => 'restricted_status',
itemnotes => 'public_notes',
itemnotes_nonpublic => 'internal_notes',
holdingbranch => 'holding_library_id',
timestamp => 'timestamp',
location => 'location',
permanent_location => 'permanent_location',
onloan => 'checked_out_date',
cn_source => 'call_number_source',
cn_sort => 'call_number_sort',
ccode => 'collection_code',
materials => 'materials_notes',
uri => 'uri',
itype => 'item_type',
more_subfields_xml => 'extended_subfields',
enumchron => 'serial_issue_number',
copynumber => 'copy_number',
stocknumber => 'inventory_number',
new_status => 'new_status'
};
}
=head3 itemtype
my $itemtype = $item->itemtype;
Returns Koha object for effective itemtype
=cut
sub itemtype {
my ( $self ) = @_;
return Koha::ItemTypes->find( $self->effective_itemtype );
}
=head2 Internal methods
=head3 _after_item_action_hooks
Helper method that takes care of calling all plugin hooks
=cut
sub _after_item_action_hooks {
my ( $self, $params ) = @_;
my $action = $params->{action};
Koha::Plugins->call(
'after_item_action',
{
action => $action,
item => $self,
item_id => $self->itemnumber,
}
);
}
=head3 _type
=cut
sub _type {
return 'Item';
}
=head1 AUTHOR
Kyle M Hall <kyle@bywatersolutions.com>
=cut
1;