Merge pull request #18148 from civicrm/5.29
[civicrm-core.git] / CRM / Event / Form / ManageEvent / Fee.php
CommitLineData
6a488035
TO
1<?php
2/*
3 +--------------------------------------------------------------------+
bc77d7c0 4 | Copyright CiviCRM LLC. All rights reserved. |
6a488035 5 | |
bc77d7c0
TO
6 | This work is published under the GNU AGPLv3 license with some |
7 | permitted exceptions and without any warranty. For full license |
8 | and copyright information, see https://civicrm.org/licensing |
6a488035 9 +--------------------------------------------------------------------+
d25dd0ee 10 */
6a488035
TO
11
12/**
6a488035 13 * @package CRM
ca5cec67 14 * @copyright CiviCRM LLC https://civicrm.org/licensing
6a488035
TO
15 */
16
17/**
3bdf1f3a 18 * This class generates form components for Event Fees.
6a488035
TO
19 */
20class CRM_Event_Form_ManageEvent_Fee extends CRM_Event_Form_ManageEvent {
21
22 /**
23 * Constants for number of options for data types of multiple option.
24 */
7da04cde 25 const NUM_OPTION = 11;
6a488035
TO
26
27 /**
28 * Constants for number of discounts for the event.
29 */
7da04cde 30 const NUM_DISCOUNT = 6;
6a488035
TO
31
32 /**
66f9e52b 33 * Page action.
90b461f1 34 * @var int
6a488035
TO
35 */
36 public $_action;
37
38 /**
66f9e52b 39 * In Date.
90b461f1 40 * @var string
6a488035
TO
41 */
42 private $_inDate;
43
44 /**
66f9e52b 45 * Set variables up before form is built.
6a488035 46 */
00be9182 47 public function preProcess() {
6a488035 48 parent::preProcess();
e4b857f8 49 $this->setSelectedChild('fee');
6a488035
TO
50 }
51
52 /**
3bdf1f3a 53 * Set default values for the form.
6a488035 54 *
3bdf1f3a 55 * For edit/view mode the default values are retrieved from the database.
6a488035 56 */
00be9182 57 public function setDefaultValues() {
1d05b167 58 parent::setDefaultValues();
6a488035 59
353ffa53 60 $eventId = $this->_id;
be2fb01f
CW
61 $params = [];
62 $defaults = [];
6a488035 63 if (isset($eventId)) {
be2fb01f 64 $params = ['id' => $eventId];
6a488035
TO
65 }
66
67 CRM_Event_BAO_Event::retrieve($params, $defaults);
68
69 if (isset($eventId)) {
9da8dc8c 70 $price_set_id = CRM_Price_BAO_PriceSet::getFor('civicrm_event', $eventId, NULL, 1);
6a488035
TO
71
72 if ($price_set_id) {
73 $defaults['price_set_id'] = $price_set_id;
74 }
75 else {
9da8dc8c 76 $priceSetId = CRM_Price_BAO_PriceSet::getFor('civicrm_event', $eventId, NULL);
6a488035 77 if ($priceSetId) {
9da8dc8c 78 if ($isQuick = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $priceSetId, 'is_quick_config')) {
6a488035 79 $this->assign('isQuick', $isQuick);
9da8dc8c 80 $priceField = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceField', $priceSetId, 'id', 'price_set_id');
be2fb01f 81 $options = [];
0479b4c8 82 $priceFieldOptions = CRM_Price_BAO_PriceFieldValue::getValues($priceField, $options, 'weight', TRUE);
6a488035
TO
83 $defaults['price_field_id'] = $priceField;
84 $countRow = 0;
85 foreach ($options as $optionId => $optionValue) {
86 $countRow++;
6fe4c6f8 87 $defaults['value'][$countRow] = CRM_Utils_Money::format($optionValue['amount'], NULL, '%a');
6a488035
TO
88 $defaults['label'][$countRow] = $optionValue['label'];
89 $defaults['name'][$countRow] = $optionValue['name'];
90 $defaults['weight'][$countRow] = $optionValue['weight'];
91 $defaults['price_field_value'][$countRow] = $optionValue['id'];
92 if ($optionValue['is_default']) {
93 $defaults['default'] = $countRow;
94 }
95 }
96 }
97 }
98 }
99 }
100
101 //check if discounted
102 $discountedEvent = CRM_Core_BAO_Discount::getOptionGroup($this->_id, 'civicrm_event');
103 if (!empty($discountedEvent)) {
104 $defaults['is_discount'] = $i = 1;
be2fb01f 105 $totalLables = $maxSize = $defaultDiscounts = [];
6a488035
TO
106 foreach ($discountedEvent as $optionGroupId) {
107 $defaults['discount_price_set'][] = $optionGroupId;
1d05b167
MWMC
108 $defaults["discount_name[$i]"] = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $optionGroupId, 'title');
109
110 $defaults["discount_start_date[$i]"] = CRM_Core_DAO::getFieldValue('CRM_Core_DAO_Discount', $optionGroupId,
111 'start_date', 'price_set_id');
112 $defaults["discount_end_date[$i]"] = CRM_Core_DAO::getFieldValue('CRM_Core_DAO_Discount', $optionGroupId,
113 'end_date', 'price_set_id');
9da8dc8c 114 $defaultDiscounts[] = CRM_Price_BAO_PriceSet::getSetDetail($optionGroupId);
6a488035
TO
115 $i++;
116 }
117
118 //avoid moving up value of lable when some labels don't
119 //have a value ,fixed for CRM-3088
120 $rowCount = 1;
121 foreach ($defaultDiscounts as $val) {
122 $discountFields = current($val);
123 $discountFields = current($discountFields['fields']);
124
125 foreach ($discountFields['options'] as $discountFieldsval) {
96fb2c1c 126 $defaults['discounted_label'][$discountFieldsval['weight']] = $discountFieldsval['label'];
389bcebf 127 $defaults['discounted_value'][$discountFieldsval['weight']][$rowCount] = CRM_Utils_Money::format($discountFieldsval['amount'], NULL, '%a');
0479b4c8 128 $defaults['discount_option_id'][$rowCount][$discountFieldsval['weight']] = $discountFieldsval['id'];
a7488080 129 if (!empty($discountFieldsval['is_default'])) {
6a488035
TO
130 $defaults['discounted_default'] = $discountFieldsval['weight'];
131 }
132 }
133 $rowCount++;
134 }
5e4b4477
PN
135 //CRM-12970
136 ksort($defaults['discounted_value']);
93c3579d 137 ksort($defaults['discounted_label']);
5e4b4477
PN
138 $rowCount = 1;
139 foreach ($defaults['discounted_label'] as $key => $value) {
140 if ($key != $rowCount) {
141 $defaults['discounted_label'][$rowCount] = $value;
142 $defaults['discounted_value'][$rowCount] = $defaults['discounted_value'][$key];
143 unset($defaults['discounted_value'][$key]);
144 unset($defaults['discounted_label'][$key]);
145 foreach ($defaults['discount_option_id'] as &$optionIds) {
146 if (array_key_exists($key, $optionIds)) {
147 $optionIds[$rowCount] = $optionIds[$key];
148 unset($optionIds[$key]);
149 }
b2cdd843 150 }
5e4b4477
PN
151 }
152 $rowCount++;
153 }
6a488035
TO
154
155 $this->set('discountSection', 1);
156 $this->buildQuickForm();
157 }
158 elseif (!empty($defaults['label'])) {
159 //if Regular Fees are present in DB and event fee page is in update mode
160 $defaults['discounted_label'] = $defaults['label'];
161 }
a7488080 162 elseif (!empty($this->_submitValues['label'])) {
6a488035
TO
163 //if event is newly created, use submitted values for
164 //discount labels
165 if (is_array($this->_submitValues['label'])) {
166 $k = 1;
167 foreach ($this->_submitValues['label'] as $value) {
168 if ($value) {
169 $defaults['discounted_label'][$k] = $value;
170 $k++;
171 }
172 }
173 }
174 }
175 $defaults['id'] = $eventId;
176 if (!empty($totalLables)) {
177 $maxKey = count($totalLables) - 1;
8cc574cf 178 if (isset($maxKey) && !empty($totalLables[$maxKey]['value'])) {
6a488035
TO
179 foreach ($totalLables[$maxKey]['value'] as $i => $v) {
180 if ($totalLables[$maxKey]['amount_id'][$i] == CRM_Utils_Array::value('default_discount_fee_id', $defaults)) {
181 $defaults['discounted_default'] = $i;
182 break;
183 }
184 }
185 }
186 }
187
188 if (!isset($defaults['discounted_default'])) {
189 $defaults['discounted_default'] = 1;
190 }
191
192 if (!isset($defaults['is_monetary'])) {
193 $defaults['is_monetary'] = 1;
194 }
195
196 if (!isset($defaults['fee_label'])) {
197 $defaults['fee_label'] = ts('Event Fee(s)');
198 }
199
200 if (!isset($defaults['pay_later_text']) ||
201 empty($defaults['pay_later_text'])
202 ) {
203 $defaults['pay_later_text'] = ts('I will send payment by check');
204 }
205
206 $this->_showHide = new CRM_Core_ShowHideBlocks();
207 if (!$defaults['is_monetary']) {
208 $this->_showHide->addHide('event-fees');
209 }
210
211 if (isset($defaults['price_set_id'])) {
212 $this->_showHide->addHide('map-field');
213 }
214 $this->_showHide->addToTemplate();
215 $this->assign('inDate', $this->_inDate);
a7488080 216 if (!empty($defaults['payment_processor'])) {
63e8ece7 217 $defaults['payment_processor'] = array_fill_keys(explode(CRM_Core_DAO::VALUE_SEPARATOR,
6a488035 218 $defaults['payment_processor']
63e8ece7 219 ), '1');
6a488035 220 }
63e8ece7 221
6a488035
TO
222 return $defaults;
223 }
224
225 /**
66f9e52b 226 * Build the form object.
6a488035
TO
227 */
228 public function buildQuickForm() {
6a488035
TO
229 $this->addYesNo('is_monetary',
230 ts('Paid Event'),
231 NULL,
232 NULL,
be2fb01f 233 ['onclick' => "return showHideByValue('is_monetary','0','event-fees','block','radio',false);"]
6a488035
TO
234 );
235
236 //add currency element.
237 $this->addCurrency('currency', ts('Currency'), FALSE);
238
239 $paymentProcessor = CRM_Core_PseudoConstant::paymentProcessor();
240
241 $this->assign('paymentProcessor', $paymentProcessor);
63e8ece7 242 $this->addCheckBox('payment_processor', ts('Payment Processor'),
243 array_flip($paymentProcessor),
244 NULL, NULL, NULL, NULL,
be2fb01f 245 ['&nbsp;&nbsp;', '&nbsp;&nbsp;', '&nbsp;&nbsp;', '<br/>']
63e8ece7 246 );
03e04002 247
248 // financial type
40c655aa
E
249 if (!CRM_Financial_BAO_FinancialType::isACLFinancialTypeStatus() ||
250 (CRM_Financial_BAO_FinancialType::isACLFinancialTypeStatus() && CRM_Core_Permission::check('administer CiviCRM Financial Types'))) {
a4ef4eac
E
251 $this->addSelect('financial_type_id');
252 }
40c655aa 253 else {
573fd305 254 CRM_Financial_BAO_FinancialType::getAvailableFinancialTypes($financialTypes, CRM_Core_Action::ADD);
be2fb01f 255 $this->addSelect('financial_type_id', ['context' => 'search', 'options' => $financialTypes]);
a4ef4eac 256 }
6a488035
TO
257 // add pay later options
258 $this->addElement('checkbox', 'is_pay_later', ts('Enable Pay Later option?'), NULL,
be2fb01f 259 ['onclick' => "return showHideByValue('is_pay_later','','payLaterOptions','block','radio',false);"]
6a488035
TO
260 );
261 $this->addElement('textarea', 'pay_later_text', ts('Pay Later Label'),
262 CRM_Core_DAO::getAttribute('CRM_Event_DAO_Event', 'pay_later_text'),
263 FALSE
264 );
5d51a2f9 265 $this->add('wysiwyg', 'pay_later_receipt', ts('Pay Later Instructions'), CRM_Core_DAO::getAttribute('CRM_Event_DAO_Event', 'pay_later_receipt'));
6a488035 266
2b3fa30d 267 $this->addElement('checkbox', 'is_billing_required', ts('Billing address required'));
6a488035
TO
268 $this->add('text', 'fee_label', ts('Fee Label'));
269
9da8dc8c 270 $price = CRM_Price_BAO_PriceSet::getAssoc(FALSE, 'CiviEvent');
6a488035
TO
271 if (CRM_Utils_System::isNull($price)) {
272 $this->assign('price', FALSE);
273 }
274 else {
275 $this->assign('price', TRUE);
276 }
a0a2c91d
MWMC
277 $this->addField('price_set_id', [
278 'entity' => 'PriceSet',
279 'options' => $price,
280 'onchange' => "return showHideByValue('price_set_id', '', 'map-field', 'block', 'select', false);",
281 ]);
282
be2fb01f
CW
283 $default = [$this->createElement('radio', NULL, NULL, NULL, 0)];
284 $this->add('hidden', 'price_field_id', '', ['id' => 'price_field_id']);
6a488035
TO
285 for ($i = 1; $i <= self::NUM_OPTION; $i++) {
286 // label
287 $this->add('text', "label[$i]", ts('Label'), CRM_Core_DAO::getAttribute('CRM_Core_DAO_OptionValue', 'label'));
be2fb01f 288 $this->add('hidden', "price_field_value[$i]", '', ['id' => "price_field_value[$i]"]);
6a488035
TO
289
290 // value
291 $this->add('text', "value[$i]", ts('Value'), CRM_Core_DAO::getAttribute('CRM_Core_DAO_OptionValue', 'value'));
be2fb01f 292 $this->addRule("value[$i]", ts('Please enter a valid money value for this field (e.g. %1).', [1 => CRM_Utils_Money::format('99.99', ' ')]), 'money');
6a488035
TO
293
294 // default
295 $default[] = $this->createElement('radio', NULL, NULL, NULL, $i);
296 }
297
298 $this->addGroup($default, 'default');
299
300 $this->addElement('checkbox', 'is_discount', ts('Discounts by Signup Date?'), NULL,
be2fb01f 301 ['onclick' => "warnDiscountDel(); return showHideByValue('is_discount','','discount','block','radio',false);"]
6a488035
TO
302 );
303 $discountSection = $this->get('discountSection');
304
305 $this->assign('discountSection', $discountSection);
306
307 // form fields of Discount sets
be2fb01f 308 $defaultOption = [];
6a488035
TO
309 $_showHide = new CRM_Core_ShowHideBlocks('', '');
310
311 for ($i = 1; $i <= self::NUM_DISCOUNT; $i++) {
312 //the show hide blocks
313 $showBlocks = 'discount_' . $i;
314 if ($i > 2) {
315 $_showHide->addHide($showBlocks);
316 }
317 else {
318 $_showHide->addShow($showBlocks);
319 }
320
321 //Increment by 1 of start date of previous end date.
322 if (is_array($this->_submitValues) &&
323 !empty($this->_submitValues['discount_name'][$i]) &&
324 !empty($this->_submitValues['discount_name'][$i + 1]) &&
325 isset($this->_submitValues['discount_end_date']) &&
326 isset($this->_submitValues['discount_end_date'][$i]) &&
327 $i < self::NUM_DISCOUNT - 1
328 ) {
6a488035
TO
329 if (!empty($this->_submitValues['discount_end_date'][$i + 1])
330 && empty($this->_submitValues['discount_start_date'][$i + 1])
331 ) {
1d05b167 332 $this->_submitValues['discount_start_date'][$i + 1] = date('Y-m-d', strtotime("+1 days " . $this->_submitValues['discount_end_date'][$i]));
6a488035
TO
333 }
334 }
335 //Decrement by 1 of end date from next start date.
336 if ($i > 1 &&
337 is_array($this->_submitValues) &&
338 !empty($this->_submitValues['discount_name'][$i]) &&
339 !empty($this->_submitValues['discount_name'][$i - 1]) &&
340 isset($this->_submitValues['discount_start_date']) &&
341 isset($this->_submitValues['discount_start_date'][$i])
342 ) {
6a488035
TO
343 if (!empty($this->_submitValues['discount_start_date'][$i])
344 && empty($this->_submitValues['discount_end_date'][$i - 1])
345 ) {
1d05b167 346 list($this->_submitValues['discount_end_date'][$i - 1]) = date('Y-m-d', strtotime("-1 days " . $this->_submitValues['discount_start_date'][$i]));
6a488035
TO
347 }
348 }
349
6a488035 350 $this->add('text', 'discount_name[' . $i . ']', ts('Discount Name'),
9da8dc8c 351 CRM_Core_DAO::getAttribute('CRM_Price_DAO_PriceSet', 'title')
6a488035 352 );
be2fb01f
CW
353 $this->add('hidden', "discount_price_set[$i]", '', ['id' => "discount_price_set[$i]"]);
354 $this->add('datepicker', 'discount_start_date[' . $i . ']', ts('Discount Start Date'), [], FALSE, ['time' => FALSE]);
355 $this->add('datepicker', 'discount_end_date[' . $i . ']', ts('Discount End Date'), [], FALSE, ['time' => FALSE]);
6a488035
TO
356 }
357 $_showHide->addToTemplate();
9d0008ba
AH
358 $this->addElement('xbutton', $this->getButtonName('submit'), ts('Add Discount Set to Fee Table'),
359 [
360 'type' => 'submit',
361 'class' => 'crm-form-submit cancel',
88e661f5 362 'value' => 1,
9d0008ba 363 ]
6a488035 364 );
a17bec97 365 if (Civi::settings()->get('deferred_revenue_enabled')) {
cebff547
PN
366 $deferredFinancialType = CRM_Financial_BAO_FinancialAccount::getDeferredFinancialType();
367 $this->assign('deferredFinancialType', array_keys($deferredFinancialType));
368 }
6a488035
TO
369 $this->buildAmountLabel();
370 parent::buildQuickForm();
371 }
372
373 /**
66f9e52b 374 * Add local and global form rules.
6a488035 375 */
00be9182 376 public function addRules() {
be2fb01f 377 $this->addFormRule(['CRM_Event_Form_ManageEvent_Fee', 'formRule']);
6a488035
TO
378 }
379
380 /**
66f9e52b 381 * Global validation rules for the form.
6a488035 382 *
d4dd1e85
TO
383 * @param array $values
384 * Posted values of the form.
6a488035 385 *
a6c01b45
CW
386 * @return array
387 * list of errors to be posted back to the form
6a488035 388 */
00be9182 389 public static function formRule($values) {
be2fb01f 390 $errors = [];
a7488080 391 if (!empty($values['is_discount'])) {
353ffa53
TO
392 $occurDiscount = array_count_values($values['discount_name']);
393 $countemptyrows = 0;
6a488035
TO
394 $countemptyvalue = 0;
395 for ($i = 1; $i <= self::NUM_DISCOUNT; $i++) {
396 $start_date = $end_date = NULL;
a7488080
CW
397 if (!empty($values['discount_name'][$i])) {
398 if (!empty($values['discount_start_date'][$i])) {
6a488035
TO
399 $start_date = ($values['discount_start_date'][$i]) ? CRM_Utils_Date::processDate($values['discount_start_date'][$i]) : 0;
400 }
a7488080 401 if (!empty($values['discount_end_date'][$i])) {
6a488035
TO
402 $end_date = ($values['discount_end_date'][$i]) ? CRM_Utils_Date::processDate($values['discount_end_date'][$i]) : 0;
403 }
404
405 if ($start_date && $end_date && strcmp($end_date, $start_date) < 0) {
406 $errors["discount_end_date[$i]"] = ts('The discount end date cannot be prior to the start date.');
407 }
408
409 if (!$start_date && !$end_date) {
410 $errors["discount_start_date[$i]"] = $errors["discount_end_date[$i]"] = ts('Please specify either start date or end date.');
411 }
412
413 if ($i > 1) {
414 $end_date_1 = ($values['discount_end_date'][$i - 1]) ? CRM_Utils_Date::processDate($values['discount_end_date'][$i - 1]) : 0;
415 if ($start_date && $end_date_1 && strcmp($end_date_1, $start_date) >= 0) {
416 $errors["discount_start_date[$i]"] = ts('Select non-overlapping discount start date.');
417 }
418 elseif (!$start_date && !$end_date_1) {
419 $j = $i - 1;
420 $errors["discount_start_date[$i]"] = $errors["discount_end_date[$j]"] = ts('Select either of the dates.');
421 }
422 }
423
4f99ca55
TO
424 foreach ($occurDiscount as $key => $value) {
425 if ($value > 1 && $key <> '') {
0479b4c8 426 if ($key == $values['discount_name'][$i]) {
be2fb01f 427 $errors['discount_name[' . $i . ']'] = ts('%1 is already used for Discount Name.', [1 => $key]);
0479b4c8 428 }
4f99ca55 429 }
6a488035
TO
430 }
431
432 //validation for discount labels and values
433 for ($index = (self::NUM_OPTION); $index > 0; $index--) {
434 $label = TRUE;
435 if (empty($values['discounted_label'][$index]) && !empty($values['discounted_value'][$index][$i])) {
436 $label = FALSE;
437 if (!$label) {
438 $errors["discounted_label[{$index}]"] = ts('Label cannot be empty.');
439 }
440 }
441 if (!empty($values['discounted_label'][$index])) {
442 $duplicateIndex = CRM_Utils_Array::key($values['discounted_label'][$index], $values['discounted_label']);
443
444 if ((!($duplicateIndex === FALSE)) && (!($duplicateIndex == $index))) {
445 $errors["discounted_label[{$index}]"] = ts('Duplicate label value');
446 }
447 }
448 if (empty($values['discounted_label'][$index]) && empty($values['discounted_value'][$index][$i])) {
449 $countemptyrows++;
450 }
451 if (empty($values['discounted_value'][$index][$i])) {
452 $countemptyvalue++;
453 }
454 }
a7488080 455 if (!empty($values['_qf_Fee_next']) && ($countemptyrows == 11 || $countemptyvalue == 11)) {
6a488035
TO
456 $errors["discounted_label[1]"] = $errors["discounted_value[1][$i]"] = ts('At least one fee should be entered for your Discount Set. If you do not see the table to enter discount fees, click the "Add Discount Set to Fee Table" button.');
457 }
458 }
459 }
460 }
03e04002 461 if ($values['is_monetary']) {
6a488035
TO
462 //check if financial type is selected
463 if (!$values['financial_type_id']) {
464 $errors['financial_type_id'] = ts("Please select financial type.");
465 }
466
467 //check for the event fee label (mandatory)
468 if (!$values['fee_label']) {
469 $errors['fee_label'] = ts('Please enter the fee label for the paid event.');
470 }
471
a7488080 472 if (empty($values['price_set_id'])) {
6a488035
TO
473 //check fee label and amount
474 $check = 0;
be2fb01f 475 $optionKeys = [];
6a488035
TO
476 foreach ($values['label'] as $key => $val) {
477 if (trim($val) && trim($values['value'][$key])) {
478 $optionKeys[$key] = $key;
479 $check++;
480 }
481 }
482
9c1bc317 483 $default = $values['default'] ?? NULL;
6a488035
TO
484 if ($default && !in_array($default, $optionKeys)) {
485 $errors['default'] = ts('Please select an appropriate option as default.');
486 }
487
488 if (!$check) {
489 if (!$values['label'][1]) {
490 $errors['label[1]'] = ts('Please enter a label for at least one fee level.');
491 }
492 if (!$values['value'][1]) {
493 $errors['value[1]'] = ts('Please enter an amount for at least one fee level.');
494 }
495 }
496 }
497 if (isset($values['is_pay_later'])) {
498 if (empty($values['pay_later_text'])) {
499 $errors['pay_later_text'] = ts('Please enter the Pay Later prompt to be displayed on the Registration form.');
500 }
501 if (empty($values['pay_later_receipt'])) {
502 $errors['pay_later_receipt'] = ts('Please enter the Pay Later instructions to be displayed to your users.');
503 }
504 }
fd5bb3c3
YC
505 else {
506 if (empty($values['payment_processor'])) {
507 $errors['payment_processor'] = ts('You have indicated that this is a paid event, but no payment option has been selected. If this is not a paid event, please select the "No" option at the top of the page. If this is a paid event, please select at least one payment processor and/or enable the pay later option.');
508 }
509 }
6a488035
TO
510 }
511 return empty($errors) ? TRUE : $errors;
512 }
513
514 public function buildAmountLabel() {
be2fb01f 515 $default = [];
6a488035
TO
516 for ($i = 1; $i <= self::NUM_OPTION; $i++) {
517 // label
518 $this->add('text', "discounted_label[$i]", ts('Label'), CRM_Core_DAO::getAttribute('CRM_Core_DAO_OptionValue', 'label'));
519 // value
520 for ($j = 1; $j <= self::NUM_DISCOUNT; $j++) {
be2fb01f
CW
521 $this->add('text', "discounted_value[$i][$j]", ts('Value'), ['size' => 10]);
522 $this->addRule("discounted_value[$i][$j]", ts('Please enter a valid money value for this field (e.g. %1).', [1 => CRM_Utils_Money::format('99.99', ' ')]), 'money');
6a488035
TO
523 }
524
525 // default
526 $default[] = $this->createElement('radio', NULL, NULL, NULL, $i);
527 }
528
529 $this->addGroup($default, 'discounted_default');
530 }
531
532 /**
66f9e52b 533 * Process the form.
6a488035
TO
534 */
535 public function postProcess() {
6a488035 536 $eventTitle = '';
57463d81 537 $params = $this->cleanMoneyFields($this->exportValues());
6a488035
TO
538
539 $this->set('discountSection', 0);
540
a7488080 541 if (!empty($_POST['_qf_Fee_submit'])) {
6a488035
TO
542 $this->buildAmountLabel();
543 $this->set('discountSection', 2);
544 return;
545 }
546
9f573c51 547 if (!empty($params['payment_processor'])) {
476627e4 548 $params['payment_processor'] = array_keys($params['payment_processor']);
6a488035
TO
549 }
550 else {
551 $params['payment_processor'] = 'null';
552 }
553
554 $params['is_pay_later'] = CRM_Utils_Array::value('is_pay_later', $params, 0);
e45f6ab0 555 $params['is_billing_required'] = CRM_Utils_Array::value('is_billing_required', $params, 0);
6a488035
TO
556
557 if ($this->_id) {
558
559 // delete all the prior label values or discounts in the custom options table
560 // and delete a price set if one exists
b2cdd843
EM
561 //@todo note that this removes the reference from existing participants -
562 // even where there is not change - redress?
563 // note that a more tentative form of this is invoked by passing price_set_id as an array
564 // to event.create see CRM-14069
565 // @todo get all of this logic out of form layer (currently partially in BAO/api layer)
9da8dc8c 566 if (CRM_Price_BAO_PriceSet::removeFrom('civicrm_event', $this->_id)) {
0479b4c8 567 CRM_Core_BAO_Discount::del($this->_id, 'civicrm_event');
6a488035
TO
568 }
569 }
570
571 if ($params['is_monetary']) {
a7488080 572 if (!empty($params['price_set_id'])) {
b2cdd843
EM
573 //@todo this is now being done in the event BAO if passed price_set_id as an array
574 // per notes on that fn - looking at the api converting to an array
575 // so calling via the api may cause this to be done in the api
9da8dc8c 576 CRM_Price_BAO_PriceSet::addTo('civicrm_event', $this->_id, $params['price_set_id']);
a7488080 577 if (!empty($params['price_field_id'])) {
9da8dc8c 578 $priceSetID = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceField', $params['price_field_id'], 'price_set_id');
0479b4c8 579 CRM_Price_BAO_PriceSet::setIsQuickConfig($priceSetID, 0);
6a488035
TO
580 }
581 }
582 else {
583 // if there are label / values, create custom options for them
9c1bc317
CW
584 $labels = $params['label'] ?? NULL;
585 $values = $params['value'] ?? NULL;
586 $default = $params['default'] ?? NULL;
be2fb01f 587 $options = [];
6a488035
TO
588 if (!CRM_Utils_System::isNull($labels) && !CRM_Utils_System::isNull($values)) {
589 for ($i = 1; $i < self::NUM_OPTION; $i++) {
590 if (!empty($labels[$i]) && !CRM_Utils_System::isNull($values[$i])) {
be2fb01f 591 $options[] = [
353ffa53 592 'label' => trim($labels[$i]),
57463d81 593 'value' => $values[$i],
6a488035
TO
594 'weight' => $i,
595 'is_active' => 1,
21dfd5f5 596 'is_default' => $default == $i,
be2fb01f 597 ];
6a488035
TO
598 }
599 }
600 if (!empty($options)) {
601 $params['default_fee_id'] = NULL;
a7488080
CW
602 if (empty($params['price_set_id'])) {
603 if (empty($params['price_field_id'])) {
6a488035
TO
604 $setParams['title'] = $eventTitle = ($this->_isTemplate) ? $this->_defaultValues['template_title'] : $this->_defaultValues['title'];
605 $eventTitle = strtolower(CRM_Utils_String::munge($eventTitle, '_', 245));
9da8dc8c 606 if (!CRM_Core_DAO::getFieldValue('CRM_Price_BAO_PriceSet', $eventTitle, 'id', 'name')) {
6a488035
TO
607 $setParams['name'] = $eventTitle;
608 }
9da8dc8c 609 elseif (!CRM_Core_DAO::getFieldValue('CRM_Price_BAO_PriceSet', $eventTitle . '_' . $this->_id, 'id', 'name')) {
6a488035
TO
610 $setParams['name'] = $eventTitle . '_' . $this->_id;
611 }
612 else {
0479b4c8 613 $timeSec = explode('.', microtime(TRUE));
6a488035
TO
614 $setParams['name'] = $eventTitle . '_' . date('is', $timeSec[0]) . $timeSec[1];
615 }
616 $setParams['is_quick_config'] = 1;
cc9b58f3 617 $setParams['financial_type_id'] = $params['financial_type_id'];
6a488035 618 $setParams['extends'] = CRM_Core_Component::getComponentID('CiviEvent');
9da8dc8c 619 $priceSet = CRM_Price_BAO_PriceSet::create($setParams);
6a488035 620
5778c444 621 $fieldParams['name'] = strtolower(CRM_Utils_String::munge($params['fee_label'], '_', 245));
6a488035
TO
622 $fieldParams['price_set_id'] = $priceSet->id;
623 }
624 else {
625 foreach ($params['price_field_value'] as $arrayID => $fieldValueID) {
626 if (empty($params['label'][$arrayID]) && empty($params['value'][$arrayID]) && !empty($fieldValueID)) {
9da8dc8c 627 CRM_Price_BAO_PriceFieldValue::setIsActive($fieldValueID, '0');
6a488035
TO
628 unset($params['price_field_value'][$arrayID]);
629 }
630 }
9c1bc317 631 $fieldParams['id'] = $params['price_field_id'] ?? NULL;
6a488035 632 $fieldParams['option_id'] = $params['price_field_value'];
cdeb4bdf 633
9da8dc8c 634 $priceSet = new CRM_Price_BAO_PriceSet();
635 $priceSet->id = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceField', CRM_Utils_Array::value('price_field_id', $params), 'price_set_id');
cdeb4bdf 636
cc9b58f3 637 if ($this->_defaultValues['financial_type_id'] != $params['financial_type_id']) {
cdeb4bdf 638 CRM_Core_DAO::setFieldValue('CRM_Price_DAO_PriceSet', $priceSet->id, 'financial_type_id', $params['financial_type_id']);
cc9b58f3 639 }
6a488035 640 }
5778c444 641 $fieldParams['label'] = $params['fee_label'];
6a488035 642 $fieldParams['html_type'] = 'Radio';
9da8dc8c 643 CRM_Price_BAO_PriceSet::addTo('civicrm_event', $this->_id, $priceSet->id);
6a488035
TO
644 $fieldParams['option_label'] = $params['label'];
645 $fieldParams['option_amount'] = $params['value'];
646 $fieldParams['financial_type_id'] = $params['financial_type_id'];
ddca8f33
TO
647 foreach ($options as $value) {
648 $fieldParams['option_weight'][$value['weight']] = $value['weight'];
0479b4c8 649 }
6a488035 650 $fieldParams['default_option'] = $params['default'];
9da8dc8c 651 $priceField = CRM_Price_BAO_PriceField::create($fieldParams);
6a488035
TO
652 }
653 }
654 }
655
be2fb01f
CW
656 $discountPriceSets = !empty($this->_defaultValues['discount_price_set']) ? $this->_defaultValues['discount_price_set'] : [];
657 $discountFieldIDs = !empty($this->_defaultValues['discount_option_id']) ? $this->_defaultValues['discount_option_id'] : [];
6a488035
TO
658 if (CRM_Utils_Array::value('is_discount', $params) == 1) {
659 // if there are discounted set of label / values,
660 // create custom options for them
9c1bc317
CW
661 $labels = $params['discounted_label'] ?? NULL;
662 $values = $params['discounted_value'] ?? NULL;
663 $default = $params['discounted_default'] ?? NULL;
6a488035
TO
664
665 if (!CRM_Utils_System::isNull($labels) && !CRM_Utils_System::isNull($values)) {
666 for ($j = 1; $j <= self::NUM_DISCOUNT; $j++) {
be2fb01f 667 $discountOptions = [];
6a488035
TO
668 for ($i = 1; $i < self::NUM_OPTION; $i++) {
669 if (!empty($labels[$i]) &&
d6399f06 670 !CRM_Utils_System::isNull(CRM_Utils_Array::value($j, $values[$i]))
671 ) {
be2fb01f 672 $discountOptions[] = [
d6399f06 673 'label' => trim($labels[$i]),
57463d81 674 'value' => $values[$i][$j],
6a488035
TO
675 'weight' => $i,
676 'is_active' => 1,
21dfd5f5 677 'is_default' => $default == $i,
be2fb01f 678 ];
6a488035
TO
679 }
680 }
681
682 if (!empty($discountOptions)) {
be2fb01f 683 $fieldParams = [];
6a488035 684 $params['default_discount_fee_id'] = NULL;
0479b4c8 685 $keyCheck = $j - 1;
be2fb01f 686 $setParams = [];
a7488080 687 if (empty($discountPriceSets[$keyCheck])) {
6a488035
TO
688 if (!$eventTitle) {
689 $eventTitle = strtolower(CRM_Utils_String::munge($this->_defaultValues['title'], '_', 200));
690 }
691 $setParams['title'] = $params['discount_name'][$j];
9da8dc8c 692 if (!CRM_Core_DAO::getFieldValue('CRM_Price_BAO_PriceSet', $eventTitle . '_' . $params['discount_name'][$j], 'id', 'name')) {
6a488035
TO
693 $setParams['name'] = $eventTitle . '_' . $params['discount_name'][$j];
694 }
9da8dc8c 695 elseif (!CRM_Core_DAO::getFieldValue('CRM_Price_BAO_PriceSet', $eventTitle . '_' . $params['discount_name'][$j] . '_' . $this->_id, 'id', 'name')) {
6a488035
TO
696 $setParams['name'] = $eventTitle . '_' . $params['discount_name'][$j] . '_' . $this->_id;
697 }
698 else {
0479b4c8 699 $timeSec = explode('.', microtime(TRUE));
6a488035
TO
700 $setParams['name'] = $eventTitle . '_' . $params['discount_name'][$j] . '_' . date('is', $timeSec[0]) . $timeSec[1];
701 }
702 $setParams['is_quick_config'] = 1;
cc9b58f3 703 $setParams['financial_type_id'] = $params['financial_type_id'];
6a488035 704 $setParams['extends'] = CRM_Core_Component::getComponentID('CiviEvent');
9da8dc8c 705 $priceSet = CRM_Price_BAO_PriceSet::create($setParams);
6a488035 706 $priceSetID = $priceSet->id;
b2cdd843 707 }
d8fe9408 708 else {
0479b4c8 709 $priceSetID = $discountPriceSets[$j - 1];
be2fb01f 710 $setParams = [
d8fe9408
PN
711 'title' => $params['discount_name'][$j],
712 'id' => $priceSetID,
be2fb01f 713 ];
cc9b58f3 714 if ($this->_defaultValues['financial_type_id'] != $params['financial_type_id']) {
d8fe9408 715 $setParams['financial_type_id'] = $params['financial_type_id'];
cc9b58f3 716 }
d8fe9408 717 CRM_Price_BAO_PriceSet::create($setParams);
0479b4c8 718 unset($discountPriceSets[$j - 1]);
9da8dc8c 719 $fieldParams['id'] = CRM_Core_DAO::getFieldValue('CRM_Price_BAO_PriceField', $priceSetID, 'id', 'price_set_id');
6a488035
TO
720 }
721
722 $fieldParams['name'] = $fieldParams['label'] = $params['fee_label'];
723 $fieldParams['is_required'] = 1;
724 $fieldParams['price_set_id'] = $priceSetID;
725 $fieldParams['html_type'] = 'Radio';
726 $fieldParams['financial_type_id'] = $params['financial_type_id'];
727 foreach ($discountOptions as $value) {
728 $fieldParams['option_label'][$value['weight']] = $value['label'];
729 $fieldParams['option_amount'][$value['weight']] = $value['value'];
730 $fieldParams['option_weight'][$value['weight']] = $value['weight'];
a7488080 731 if (!empty($value['is_default'])) {
6a488035
TO
732 $fieldParams['default_option'] = $value['weight'];
733 }
8cc574cf 734 if (!empty($discountFieldIDs[$j]) && !empty($discountFieldIDs[$j][$value['weight']])) {
d8fe9408
PN
735 $fieldParams['option_id'][$value['weight']] = $discountFieldIDs[$j][$value['weight']];
736 unset($discountFieldIDs[$j][$value['weight']]);
6a488035
TO
737 }
738 }
739 //create discount priceset
9da8dc8c 740 $priceField = CRM_Price_BAO_PriceField::create($fieldParams);
d8fe9408 741 if (!empty($discountFieldIDs[$j])) {
9b873358 742 foreach ($discountFieldIDs[$j] as $fID) {
9da8dc8c 743 CRM_Price_BAO_PriceFieldValue::setIsActive($fID, '0');
6a488035
TO
744 }
745 }
746
be2fb01f 747 $discountParams = [
6a488035
TO
748 'entity_table' => 'civicrm_event',
749 'entity_id' => $this->_id,
750 'price_set_id' => $priceSetID,
1d05b167
MWMC
751 'start_date' => $params['discount_start_date'][$j],
752 'end_date' => $params['discount_end_date'][$j],
be2fb01f 753 ];
6a488035
TO
754 CRM_Core_BAO_Discount::add($discountParams);
755 }
756 }
757 }
758 }
759 if (!empty($discountPriceSets)) {
760 foreach ($discountPriceSets as $setId) {
9da8dc8c 761 CRM_Price_BAO_PriceSet::setIsQuickConfig($setId, 0);
6a488035
TO
762 }
763 }
764 }
a06f8589
PJ
765 }
766 else {
a7488080 767 if (!empty($params['price_field_id'])) {
9da8dc8c 768 $priceSetID = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceField', $params['price_field_id'], 'price_set_id');
0479b4c8 769 CRM_Price_BAO_PriceSet::setIsQuickConfig($priceSetID, 0);
6a488035
TO
770 }
771 $params['financial_type_id'] = '';
a06f8589 772 $params['is_pay_later'] = 0;
e45f6ab0
PB
773 $params['is_billing_required'] = 0;
774 }
775
d75f2f47 776 //update 'is_billing_required'
e45f6ab0 777 if (empty($params['is_pay_later'])) {
0479b4c8 778 $params['is_billing_required'] = FALSE;
6a488035
TO
779 }
780
781 //update events table
782 $params['id'] = $this->_id;
cc9b58f3
PN
783 // skip update of financial type in price set
784 $params['skipFinancialType'] = TRUE;
6a488035
TO
785 CRM_Event_BAO_Event::add($params);
786
5d92a7e7
CW
787 // Update tab "disabled" css class
788 $this->ajaxResponse['tabValid'] = !empty($params['is_monetary']);
6a488035
TO
789 parent::endPostProcess();
790 }
791
792 /**
793 * Return a descriptive name for the page, used in wizard header
794 *
795 * @return string
6a488035
TO
796 */
797 public function getTitle() {
798 return ts('Event Fees');
799 }
96025800 800
57463d81 801 /**
802 * Clean money fields in submitted params to remove formatting.
803 *
804 * @param array $params
805 *
806 * @return array
807 */
808 protected function cleanMoneyFields($params) {
809 foreach ($params['value'] as $index => $value) {
810 if (CRM_Utils_System::isNull($value)) {
811 unset($params['value'][$index]);
812 }
813 else {
814 $params['value'][$index] = CRM_Utils_Rule::cleanMoney(trim($value));
815 }
816 }
817 foreach ($params['discounted_value'] as $index => $discountedValueSet) {
818 foreach ($discountedValueSet as $innerIndex => $value) {
819 if (CRM_Utils_System::isNull($value)) {
820 unset($params['discounted_value'][$index][$innerIndex]);
821 }
822 else {
823 $params['discounted_value'][$index][$innerIndex] = CRM_Utils_Rule::cleanMoney(trim($value));
824 }
825 }
826 if (empty($params['discounted_value'][$index])) {
827 unset($params['discounted_value'][$index]);
828 }
829 }
830 return $params;
831 }
832
6a488035 833}