3 +--------------------------------------------------------------------+
5 +--------------------------------------------------------------------+
6 | Copyright CiviCRM LLC (c) 2004-2019 |
7 +--------------------------------------------------------------------+
8 | This file is a part of CiviCRM. |
10 | CiviCRM is free software; you can copy, modify, and distribute it |
11 | under the terms of the GNU Affero General Public License |
12 | Version 3, 19 November 2007 and the CiviCRM Licensing Exception. |
14 | CiviCRM 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. |
17 | See the GNU Affero General Public License for more details. |
19 | You should have received a copy of the GNU Affero General Public |
20 | License and the CiviCRM Licensing Exception along |
21 | with this program; if not, contact CiviCRM LLC |
22 | at info[AT]civicrm[DOT]org. If you have questions about the |
23 | GNU Affero General Public License or the licensing of CiviCRM, |
24 | see the CiviCRM license FAQ at http://civicrm.org/licensing |
25 +--------------------------------------------------------------------+
31 * @copyright CiviCRM LLC (c) 2004-2019
35 * This class provides the functionality to delete a group of contacts.
37 * This class provides functionality for the actual deletion.
39 class CRM_Contact_Form_Task_Delete
extends CRM_Contact_Form_Task
{
42 * Are we operating in "single mode", i.e. sending email to one
47 protected $_single = FALSE;
50 * Cache shared address message so we don't query twice
53 protected $_sharedAddressMessage = NULL;
56 * Build all the data structures needed to build the form.
58 public function preProcess() {
60 $cid = CRM_Utils_Request
::retrieve('cid', 'Positive',
64 $this->_searchKey
= CRM_Utils_Request
::retrieve('key', 'String', $this);
66 // sort out whether it’s a delete-to-trash, delete-into-oblivion or restore (and let the template know)
67 $values = $this->controller
->exportValues();
68 $this->_skipUndelete
= (CRM_Core_Permission
::check('access deleted contacts') and (CRM_Utils_Request
::retrieve('skip_undelete', 'Boolean', $this) or CRM_Utils_Array
::value('task', $values) == CRM_Contact_Task
::DELETE_PERMANENTLY
));
69 $this->_restore
= (CRM_Utils_Request
::retrieve('restore', 'Boolean', $this) or CRM_Utils_Array
::value('task', $values) == CRM_Contact_Task
::RESTORE
);
71 if ($this->_restore
&& !CRM_Core_Permission
::check('access deleted contacts')) {
72 CRM_Core_Error
::fatal(ts('You do not have permission to access this contact.'));
74 elseif (!CRM_Core_Permission
::check('delete contacts')) {
75 CRM_Core_Error
::fatal(ts('You do not have permission to delete this contact.'));
78 $this->assign('trash', Civi
::settings()->get('contact_undelete') and !$this->_skipUndelete
);
79 $this->assign('restore', $this->_restore
);
81 if ($this->_restore
) {
82 CRM_Utils_System
::setTitle(ts('Restore Contact'));
86 if (!CRM_Contact_BAO_Contact_Permission
::allow($cid, CRM_Core_Permission
::EDIT
)) {
87 CRM_Core_Error
::fatal(ts('You do not have permission to delete this contact. Note: you can delete contacts if you can edit them.'));
89 elseif (CRM_Contact_BAO_Contact
::checkDomainContact($cid)) {
90 CRM_Core_Error
::fatal(ts('This contact is a special one for the contact information associated with the CiviCRM installation for this domain. No one is allowed to delete it because the information is used for special system purposes.'));
93 $this->_contactIds
= [$cid];
94 $this->_single
= TRUE;
95 $this->assign('totalSelectedContacts', 1);
101 $this->_sharedAddressMessage
= $this->get('sharedAddressMessage');
102 if (!$this->_restore
&& !$this->_sharedAddressMessage
) {
103 // we check for each contact for shared contact address
104 $sharedContactList = [];
105 $sharedAddressCount = 0;
106 foreach ($this->_contactIds
as $contactId) {
107 // check if a contact that is being deleted has any shared addresses
108 $sharedAddressMessage = CRM_Core_BAO_Address
::setSharedAddressDeleteStatus(NULL, $contactId, TRUE);
110 if ($sharedAddressMessage['count'] > 0) {
111 $sharedAddressCount +
= $sharedAddressMessage['count'];
112 $sharedContactList = array_merge($sharedContactList,
113 $sharedAddressMessage['contactList']
118 $this->_sharedAddressMessage
= [
119 'count' => $sharedAddressCount,
120 'contactList' => $sharedContactList,
123 if ($sharedAddressCount > 0) {
124 if (count($this->_contactIds
) > 1) {
125 // more than one contact deleted
126 $message = ts('One of the selected contacts has an address record that is shared with 1 other contact.', [
127 'plural' => 'One or more selected contacts have address records which are shared with %count other contacts.',
128 'count' => $sharedAddressCount,
132 // only one contact deleted
133 $message = ts('This contact has an address record which is shared with 1 other contact.', [
134 'plural' => 'This contact has an address record which is shared with %count other contacts.',
135 'count' => $sharedAddressCount,
138 CRM_Core_Session
::setStatus($message . ' ' . ts('Shared addresses will not be removed or altered but will no longer be shared.'), ts('Shared Addesses Owner'));
141 // set in form controller so that queries are not fired again
142 $this->set('sharedAddressMessage', $this->_sharedAddressMessage
);
147 * Build the form object.
149 public function buildQuickForm() {
150 $label = $this->_restore ?
ts('Restore Contact(s)') : ts('Delete Contact(s)');
152 if ($this->_single
) {
153 // also fix the user context stack in case the user hits cancel
154 $context = CRM_Utils_Request
::retrieve('context', 'Alphanumeric', $this, FALSE, 'basic');
155 if ($context == 'search' && CRM_Utils_Rule
::qfKey($this->_searchKey
)) {
156 $urlParams = "&context=$context&key=$this->_searchKey";
162 $session = CRM_Core_Session
::singleton();
163 $session->replaceUserContext(CRM_Utils_System
::url('civicrm/contact/view',
164 'reset=1&cid=' . $this->_contactIds
[0] . $urlParams
166 $this->addDefaultButtons($label, 'done', 'cancel');
169 $this->addDefaultButtons($label, 'done');
172 $this->addFormRule(['CRM_Contact_Form_Task_Delete', 'formRule'], $this);
178 * @param array $fields
179 * The input form values.
180 * @param array $files
181 * The uploaded files if any.
182 * @param object $self
186 * true if no errors, else array of errors
188 public static function formRule($fields, $files, $self) {
191 if ($self->_skipUndelete
) {
192 CRM_Financial_BAO_FinancialItem
::checkContactPresent($self->_contactIds
, $error);
198 * Process the form after the input has been submitted and validated.
200 public function postProcess() {
201 $session = CRM_Core_Session
::singleton();
202 $currentUserId = $session->get('userID');
204 $context = CRM_Utils_Request
::retrieve('context', 'Alphanumeric', $this, FALSE, 'basic');
205 $urlParams = 'force=1';
206 $urlString = "civicrm/contact/search/$context";
208 if (CRM_Utils_Rule
::qfKey($this->_searchKey
)) {
209 $urlParams .= "&qfKey=$this->_searchKey";
211 elseif ($context == 'search') {
212 $urlParams .= "&qfKey={$this->controller->_key}";
213 $urlString = 'civicrm/contact/search';
215 elseif ($context == 'smog') {
216 $urlParams .= "&qfKey={$this->controller->_key}&context=smog";
217 $urlString = 'civicrm/group/search';
220 $urlParams = "reset=1";
221 $urlString = 'civicrm/dashboard';
224 // Delete/Restore Contacts. Report errors.
227 foreach ($this->_contactIds
as $cid) {
228 $name = CRM_Core_DAO
::getFieldValue('CRM_Contact_DAO_Contact', $cid, 'display_name');
229 if (CRM_Contact_BAO_Contact
::checkDomainContact($cid)) {
230 $session->setStatus(ts("'%1' cannot be deleted because the information is used for special system purposes.", [1 => $name]), 'Cannot Delete Domain Contact', 'error');
233 if ($currentUserId == $cid && !$this->_restore
) {
234 $session->setStatus(ts("You are currently logged in as '%1'. You cannot delete yourself.", [1 => $name]), 'Unable To Delete', 'error');
237 if (CRM_Contact_BAO_Contact
::deleteContact($cid, $this->_restore
, $this->_skipUndelete
)) {
241 $url = CRM_Utils_System
::url('civicrm/contact/view', "reset=1&cid=$cid");
242 $not_deleted[$cid] = "<a href='$url'>$name</a>";
246 $title = ts('Deleted');
247 if ($this->_restore
) {
248 $title = ts('Restored');
249 $status = ts('%1 has been restored from the trash.', [
251 'plural' => '%count contacts restored from trash.',
255 elseif ($this->_skipUndelete
) {
256 $status = ts('%1 has been permanently deleted.', [
258 'plural' => '%count contacts permanently deleted.',
263 $status = ts('%1 has been moved to the trash.', [
265 'plural' => '%count contacts moved to trash.',
269 $session->setStatus($status, $title, 'success');
271 // Alert user of any failures
273 $status = ts('The contact might be the Membership Organization of a Membership Type. You will need to edit the Membership Type and change the Membership Organization before you can delete this contact.');
274 $title = ts('Unable to Delete');
275 $session->setStatus('<ul><li>' . implode('</li><li>', $not_deleted) . '</li></ul>' . $status, $title, 'error');
278 if (isset($this->_sharedAddressMessage
) && $this->_sharedAddressMessage
['count'] > 0 && !$this->_restore
) {
279 if (count($this->_sharedAddressMessage
['contactList']) == 1) {
280 $message = ts('The following contact had been sharing an address with a contact you just deleted. Their address will no longer be shared, but has not been removed or altered.');
283 $message = ts('The following contacts had been sharing addresses with a contact you just deleted. Their addressses will no longer be shared, but have not been removed or altered.');
285 $message .= '<ul><li>' . implode('</li><li>', $this->_sharedAddressMessage
['contactList']) . '</li></ul>';
287 $session->setStatus($message, ts('Shared Addesses Owner Deleted'), 'info', ['expires' => 0]);
289 $this->set('sharedAddressMessage', NULL);
292 if ($this->_single
&& empty($this->_skipUndelete
)) {
293 $session->replaceUserContext(CRM_Utils_System
::url('civicrm/contact/view', "reset=1&cid={$this->_contactIds[0]}"));
296 $session->replaceUserContext(CRM_Utils_System
::url($urlString, $urlParams));