3 # Copyright ByWater Solutions 2014
4 # Copyright PTFS Europe 2016
6 # This file is part of Koha.
8 # Koha is free software; you can redistribute it and/or modify it under the
9 # terms of the GNU General Public License as published by the Free Software
10 # Foundation; either version 3 of the License, or (at your option) any later
13 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
14 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
15 # A PARTICULAR PURPOSE. See the GNU General Public License for more details.
17 # You should have received a copy of the GNU General Public License along
18 # with Koha; if not, write to the Free Software Foundation, Inc.,
19 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
32 use Koha::Patron::Categories;
33 use Koha::Patron::HouseboundProfiles;
34 use Koha::Patron::Images;
36 use Koha::Virtualshelves;
38 use base qw(Koha::Object);
42 Koha::Patron - Koha Patron Object class
54 Delete patron's holds, lists and finally the patron.
56 Lists owned by the borrower are deleted, but entries from the borrower to
65 $self->_result->result_source->schema->txn_do(
67 # Delete Patron's holds
68 # FIXME Should be $patron->get_holds
69 $_->delete for Koha::Holds->search( { borrowernumber => $self->borrowernumber } );
71 # Delete all lists and all shares of this borrower
72 # Consistent with the approach Koha uses on deleting individual lists
73 # Note that entries in virtualshelfcontents added by this borrower to
74 # lists of others will be handled by a table constraint: the borrower
75 # is set to NULL in those entries.
77 # We could handle the above deletes via a constraint too.
78 # But a new BZ report 11889 has been opened to discuss another approach.
79 # Instead of deleting we could also disown lists (based on a pref).
80 # In that way we could save shared and public lists.
81 # The current table constraints support that idea now.
82 # This pref should then govern the results of other routines/methods such as
83 # Koha::Virtualshelf->new->delete too.
84 # FIXME Could be $patron->get_lists
85 $_->delete for Koha::Virtualshelves->search( { owner => $self->borrowernumber } );
87 $deleted = $self->SUPER::delete;
89 logaction( "MEMBERS", "DELETE", $self->borrowernumber, "" ) if C4::Context->preference("BorrowersLog");
97 Returns a Koha::Patron object for this patron's guarantor
104 return unless $self->guarantorid();
106 return Koha::Patrons->find( $self->guarantorid() );
112 return Koha::Patron::Images->find( $self->borrowernumber )
117 Returns the guarantees (list of Koha::Patron) of this patron
124 return Koha::Patrons->search( { guarantorid => $self->borrowernumber } );
127 =head3 housebound_profile
129 Returns the HouseboundProfile associated with this patron.
133 sub housebound_profile {
136 return Koha::Patron::HouseboundProfiles->find($self->borrowernumber);
141 Returns the siblings of this patron.
148 my $guarantor = $self->guarantor;
150 return unless $guarantor;
152 return Koha::Patrons->search(
156 '=' => $guarantor->id,
159 '!=' => $self->borrowernumber,
165 =head3 wants_check_for_previous_checkout
167 $wants_check = $patron->wants_check_for_previous_checkout;
169 Return 1 if Koha needs to perform PrevIssue checking, else 0.
173 sub wants_check_for_previous_checkout {
175 my $syspref = C4::Context->preference("checkPrevCheckout");
178 ## Hard syspref trumps all
179 return 1 if ($syspref eq 'hardyes');
180 return 0 if ($syspref eq 'hardno');
181 ## Now, patron pref trumps all
182 return 1 if ($self->checkprevcheckout eq 'yes');
183 return 0 if ($self->checkprevcheckout eq 'no');
185 # More complex: patron inherits -> determine category preference
186 my $checkPrevCheckoutByCat = Koha::Patron::Categories
187 ->find($self->categorycode)->checkprevcheckout;
188 return 1 if ($checkPrevCheckoutByCat eq 'yes');
189 return 0 if ($checkPrevCheckoutByCat eq 'no');
191 # Finally: category preference is inherit, default to 0
192 if ($syspref eq 'softyes') {
199 =head3 do_check_for_previous_checkout
201 $do_check = $patron->do_check_for_previous_checkout($item);
203 Return 1 if the bib associated with $ITEM has previously been checked out to
204 $PATRON, 0 otherwise.
208 sub do_check_for_previous_checkout {
209 my ( $self, $item ) = @_;
211 # Find all items for bib and extract item numbers.
212 my @items = Koha::Items->search({biblionumber => $item->{biblionumber}});
214 foreach my $item (@items) {
215 push @item_nos, $item->itemnumber;
218 # Create (old)issues search criteria
220 borrowernumber => $self->borrowernumber,
221 itemnumber => \@item_nos,
224 # Check current issues table
225 my $issues = Koha::Issues->search($criteria);
226 return 1 if $issues->count; # 0 || N
228 # Check old issues table
229 my $old_issues = Koha::OldIssues->search($criteria);
230 return $old_issues->count; # 0 || N
235 my $debarment_expiration = $patron->is_debarred;
237 Returns the date a patron debarment will expire, or undef if the patron is not
245 return unless $self->debarred;
246 return $self->debarred
247 if $self->debarred =~ '^9999'
248 or dt_from_string( $self->debarred ) > dt_from_string;
252 =head2 update_password
254 my $updated = $patron->update_password( $userid, $password );
256 Update the userid and the password of a patron.
257 If the userid already exists, returns and let DBIx::Class warns
258 This will add an entry to action_logs if BorrowersLog is set.
262 sub update_password {
263 my ( $self, $userid, $password ) = @_;
264 eval { $self->userid($userid)->store; };
265 return if $@; # Make sure the userid is not already in used by another patron
266 $self->password($password)->store;
267 logaction( "MEMBERS", "CHANGE PASS", $self->borrowernumber, "" ) if C4::Context->preference("BorrowersLog");
273 my $new_expiry_date = $patron->renew_account
275 Extending the subscription to the expiry date.
283 C4::Context->preference('BorrowerRenewalPeriodBase') eq 'dateexpiry'
284 ? dt_from_string( $self->dateexpiry )
286 my $patron_category = Koha::Patron::Categories->find( $self->categorycode ); # FIXME Should be $self->category
287 my $expiry_date = $patron_category->get_expiry_date($date);
289 $self->dateexpiry($expiry_date)->store;
291 C4::Members::AddEnrolmentFeeIfNeeded( $self->categorycode, $self->borrowernumber );
293 logaction( "MEMBERS", "RENEW", $self->borrowernumber, "Membership renewed" ) if C4::Context->preference("BorrowersLog");
294 return dt_from_string( $expiry_date )->truncate( to => 'day' );
299 my $has_overdues = $patron->has_overdues;
301 Returns the number of patron's overdues
307 my $dtf = Koha::Database->new->schema->storage->datetime_parser;
308 return $self->_result->issues->search({ date_due => { '<' => $dtf->format_datetime( dt_from_string() ) } })->count;
313 $patron->track_login;
314 $patron->track_login({ force => 1 });
316 Tracks a (successful) login attempt.
317 The preference TrackLastPatronActivity must be enabled. Or you
318 should pass the force parameter.
323 my ( $self, $params ) = @_;
326 !C4::Context->preference('TrackLastPatronActivity');
327 $self->lastseen( dt_from_string() )->store;
330 =head2 move_to_deleted
332 my $is_moved = $patron->move_to_deleted;
334 Move a patron to the deletedborrowers table.
335 This can be done before deleting a patron, to make sure the data are not completely deleted.
339 sub move_to_deleted {
341 my $patron_infos = $self->unblessed;
342 return Koha::Database->new->schema->resultset('Deletedborrower')->create($patron_infos);
355 Kyle M Hall <kyle@bywatersolutions.com>
356 Alex Sassmannshausen <alex.sassmannshausen@ptfs-europe.com>