[REF] Refactor price field form to allow for unit testing of the form
[civicrm-core.git] / CRM / Price / Form / Field.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/**
13 *
14 * @package CRM
ca5cec67 15 * @copyright CiviCRM LLC https://civicrm.org/licensing
6a488035
TO
16 */
17
18/**
19 * form to process actions on the field aspect of Price
20 */
21class CRM_Price_Form_Field extends CRM_Core_Form {
22
44374cbe 23 use CRM_Core_Form_EntityFormTrait;
24
25 /**
26 * Explicitly declare the entity api name.
27 */
28 public function getDefaultEntity() {
29 return 'PriceField';
30 }
31
32 /**
33 * Explicitly declare the form context.
34 */
35 public function getDefaultContext() {
36 return 'create';
37 }
38
39 /**
40 * Get the entity id being edited.
41 *
42 * @return int|null
43 */
44 public function getEntityId() {
45 return $this->_fid;
46 }
47
6a488035
TO
48 /**
49 * Constants for number of options for data types of multiple option.
50 */
7da04cde 51 const NUM_OPTION = 15;
6a488035
TO
52
53 /**
fe482240 54 * The custom set id saved to the session for an update.
6a488035
TO
55 *
56 * @var int
6a488035
TO
57 */
58 protected $_sid;
59
60 /**
61 * The field id, used when editing the field
62 *
63 * @var int
6a488035
TO
64 */
65 protected $_fid;
66
67 /**
fe482240 68 * The extended component Id.
6a488035
TO
69 *
70 * @var array
6a488035
TO
71 */
72 protected $_extendComponentId;
73
74 /**
fe482240 75 * Variable is set if price set is used for membership.
971e129b 76 * @var bool
6a488035
TO
77 */
78 protected $_useForMember;
79
f961a8cd
SL
80 /**
81 * Set the price Set Id (only used in tests)
82 */
83 public function setPriceSetId($priceSetId) {
84 $this->_sid = $priceSetId;
85 }
86
6a488035 87 /**
fe482240 88 * Set variables up before form is built.
6a488035
TO
89 */
90 public function preProcess() {
91
44374cbe 92 $this->_sid = CRM_Utils_Request::retrieve('sid', 'Positive', $this);
93 $this->_fid = CRM_Utils_Request::retrieve('fid', 'Positive', $this);
353ffa53 94 $url = CRM_Utils_System::url('civicrm/admin/price/field', "reset=1&action=browse&sid={$this->_sid}");
be2fb01f 95 $breadCrumb = [['title' => ts('Price Set Fields'), 'url' => $url]];
6a488035 96
be2fb01f 97 $this->_extendComponentId = [];
9da8dc8c 98 $extendComponentId = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $this->_sid, 'extends', 'id');
6a488035
TO
99 if ($extendComponentId) {
100 $this->_extendComponentId = explode(CRM_Core_DAO::VALUE_SEPARATOR, $extendComponentId);
101 }
102
103 CRM_Utils_System::appendBreadCrumb($breadCrumb);
704f21c0 104
e2046b33 105 $this->setPageTitle(ts('Price Field'));
6a488035
TO
106 }
107
108 /**
bf15c191 109 * Set default values for the form.
6a488035 110 *
a6c01b45
CW
111 * @return array
112 * array of default values
44374cbe 113 * @throws \CRM_Core_Exception
6a488035 114 */
00be9182 115 public function setDefaultValues() {
be2fb01f 116 $defaults = [];
6a488035 117 // is it an edit operation ?
44374cbe 118 if ($this->getEntityId()) {
119 $params = ['id' => $this->getEntityId()];
120 $this->assign('fid', $this->getEntityId());
9da8dc8c 121 CRM_Price_BAO_PriceField::retrieve($params, $defaults);
6a488035
TO
122 $this->_sid = $defaults['price_set_id'];
123
124 // if text, retrieve price
125 if ($defaults['html_type'] == 'Text') {
fc8edfbf 126 $isActive = $defaults['is_active'];
44374cbe 127 $valueParams = ['price_field_id' => $this->getEntityId()];
6a488035 128
9da8dc8c 129 CRM_Price_BAO_PriceFieldValue::retrieve($valueParams, $defaults);
6a488035
TO
130
131 // fix the display of the monetary value, CRM-4038
132 $defaults['price'] = CRM_Utils_Money::format($defaults['amount'], NULL, '%a');
ff33ddc8 133 $defaults['is_active'] = $isActive;
6a488035
TO
134 }
135
6a488035
TO
136 }
137 else {
138 $defaults['is_active'] = 1;
139 for ($i = 1; $i <= self::NUM_OPTION; $i++) {
140 $defaults['option_status[' . $i . ']'] = 1;
141 $defaults['option_weight[' . $i . ']'] = $i;
2db35bf6 142 $defaults['option_visibility_id[' . $i . ']'] = 1;
6a488035
TO
143 }
144 }
145
146 if ($this->_action & CRM_Core_Action::ADD) {
be2fb01f 147 $fieldValues = ['price_set_id' => $this->_sid];
9da8dc8c 148 $defaults['weight'] = CRM_Utils_Weight::getDefaultWeight('CRM_Price_DAO_PriceField', $fieldValues);
6a488035
TO
149 $defaults['options_per_line'] = 1;
150 $defaults['is_display_amounts'] = 1;
151 }
0495a6a4 152 $enabledComponents = CRM_Core_Component::getEnabledComponents();
153 $eventComponentId = NULL;
ba1dcfda 154 if (array_key_exists('CiviEvent', $enabledComponents)) {
353ffa53 155 $eventComponentId = CRM_Core_Component::getComponentID('CiviEvent');
0495a6a4 156 }
6a488035
TO
157
158 if (isset($this->_sid) && $this->_action == CRM_Core_Action::ADD) {
9da8dc8c 159 $financialTypeId = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $this->_sid, 'financial_type_id');
6a488035
TO
160 $defaults['financial_type_id'] = $financialTypeId;
161 for ($i = 1; $i <= self::NUM_OPTION; $i++) {
162 $defaults['option_financial_type_id[' . $i . ']'] = $financialTypeId;
163 }
164 }
165 return $defaults;
166 }
167
168 /**
fe482240 169 * Build the form object.
6a488035
TO
170 */
171 public function buildQuickForm() {
172 // lets trim all the whitespace
173 $this->applyFilter('__ALL__', 'trim');
174
175 // add a hidden field to remember the price set id
176 // this get around the browser tab issue
177 $this->add('hidden', 'sid', $this->_sid);
44374cbe 178 $this->add('hidden', 'fid', $this->getEntityId());
6a488035
TO
179
180 // label
9da8dc8c 181 $this->add('text', 'label', ts('Field Label'), CRM_Core_DAO::getAttribute('CRM_Price_DAO_PriceField', 'label'), TRUE);
6a488035
TO
182
183 // html_type
184 $javascript = 'onchange="option_html_type(this.form)";';
185
9da8dc8c 186 $htmlTypes = CRM_Price_BAO_PriceField::htmlTypes();
6a488035
TO
187
188 // Text box for Participant Count for a field
189
190 // Financial Type
191 $financialType = CRM_Financial_BAO_FinancialType::getIncomeFinancialType();
53df1142 192 foreach ($financialType as $finTypeId => $type) {
40c655aa 193 if (CRM_Financial_BAO_FinancialType::isACLFinancialTypeStatus()
66af7c48
PN
194 && !CRM_Core_Permission::check('add contributions of type ' . $type)
195 ) {
53df1142
E
196 unset($financialType[$finTypeId]);
197 }
198 }
6a488035
TO
199 if (count($financialType)) {
200 $this->assign('financialType', $financialType);
201 }
2db35bf6
AF
202
203 //Visibility Type Options
204 $visibilityType = CRM_Core_PseudoConstant::visibility();
205 $this->assign('visibilityType', $visibilityType);
206
0495a6a4 207 $enabledComponents = CRM_Core_Component::getEnabledComponents();
208 $eventComponentId = $memberComponentId = NULL;
ba1dcfda 209 if (array_key_exists('CiviEvent', $enabledComponents)) {
353ffa53 210 $eventComponentId = CRM_Core_Component::getComponentID('CiviEvent');
0495a6a4 211 }
ba1dcfda 212 if (array_key_exists('CiviMember', $enabledComponents)) {
0495a6a4 213 $memberComponentId = CRM_Core_Component::getComponentID('CiviMember');
214 }
366fe2a3 215
353ffa53 216 $attributes = CRM_Core_DAO::getAttribute('CRM_Price_DAO_PriceFieldValue');
366fe2a3 217
218 $this->add('select', 'financial_type_id',
219 ts('Financial Type'),
be2fb01f 220 [' ' => ts('- select -')] + $financialType
6a488035
TO
221 );
222
223 $this->assign('useForMember', FALSE);
224 if (in_array($eventComponentId, $this->_extendComponentId)) {
225 $this->add('text', 'count', ts('Participant Count'), $attributes['count']);
226
227 $this->addRule('count', ts('Participant Count should be a positive number'), 'positiveInteger');
228
229 $this->add('text', 'max_value', ts('Max Participants'), $attributes['max_value']);
230 $this->addRule('max_value', ts('Please enter a valid Max Participants.'), 'positiveInteger');
231
232 $this->assign('useForEvent', TRUE);
233 }
234 else {
235 if (in_array($memberComponentId, $this->_extendComponentId)) {
236 $this->_useForMember = 1;
237 $this->assign('useForMember', $this->_useForMember);
238 }
239 $this->assign('useForEvent', FALSE);
240 }
241
242 $sel = $this->add('select', 'html_type', ts('Input Field Type'),
243 $htmlTypes, TRUE, $javascript
244 );
245
246 // price (for text inputs)
247 $this->add('text', 'price', ts('Price'));
248 $this->registerRule('price', 'callback', 'money', 'CRM_Utils_Rule');
249 $this->addRule('price', ts('must be a monetary value'), 'money');
250
5afce5ad 251 $this->add('text', 'non_deductible_amount', ts('Non-deductible Amount'), NULL);
252 $this->registerRule('non_deductible_amount', 'callback', 'money', 'CRM_Utils_Rule');
253 $this->addRule('non_deductible_amount', ts('Please enter a monetary value for this field.'), 'money');
254
6a488035
TO
255 if ($this->_action == CRM_Core_Action::UPDATE) {
256 $this->freeze('html_type');
257 }
258
259 // form fields of Custom Option rows
260 $_showHide = new CRM_Core_ShowHideBlocks('', '');
261
262 for ($i = 1; $i <= self::NUM_OPTION; $i++) {
263
264 //the show hide blocks
265 $showBlocks = 'optionField_' . $i;
266 if ($i > 2) {
267 $_showHide->addHide($showBlocks);
268 if ($i == self::NUM_OPTION) {
269 $_showHide->addHide('additionalOption');
270 }
271 }
272 else {
273 $_showHide->addShow($showBlocks);
274 }
275 // label
276 $attributes['label']['size'] = 25;
277 $this->add('text', 'option_label[' . $i . ']', ts('Label'), $attributes['label']);
278
279 // amount
280 $this->add('text', 'option_amount[' . $i . ']', ts('Amount'), $attributes['amount']);
281 $this->addRule('option_amount[' . $i . ']', ts('Please enter a valid amount for this field.'), 'money');
282
283 //Financial Type
284 $this->add(
366fe2a3 285 'select',
92fcb95f 286 'option_financial_type_id[' . $i . ']',
366fe2a3 287 ts('Financial Type'),
be2fb01f 288 ['' => ts('- select -')] + $financialType
6a488035
TO
289 );
290 if (in_array($eventComponentId, $this->_extendComponentId)) {
291 // count
292 $this->add('text', 'option_count[' . $i . ']', ts('Participant Count'), $attributes['count']);
293 $this->addRule('option_count[' . $i . ']', ts('Please enter a valid Participants Count.'), 'positiveInteger');
294
295 // max_value
296 $this->add('text', 'option_max_value[' . $i . ']', ts('Max Participants'), $attributes['max_value']);
297 $this->addRule('option_max_value[' . $i . ']', ts('Please enter a valid Max Participants.'), 'positiveInteger');
298
299 // description
300 //$this->add('textArea', 'option_description['.$i.']', ts('Description'), array('rows' => 1, 'cols' => 40 ));
301 }
302 elseif (in_array($memberComponentId, $this->_extendComponentId)) {
303 $membershipTypes = CRM_Member_PseudoConstant::membershipType();
be2fb01f 304 $js = ['onchange' => "calculateRowValues( $i );"];
6a488035
TO
305
306 $this->add('select', 'membership_type_id[' . $i . ']', ts('Membership Type'),
be2fb01f 307 ['' => ' '] + $membershipTypes, FALSE, $js
353ffa53 308 );
896bd22c 309 $this->add('text', 'membership_num_terms[' . $i . ']', ts('Number of Terms'), CRM_Utils_Array::value('membership_num_terms', $attributes));
6a488035
TO
310 }
311
312 // weight
f20d2b0d 313 $this->add('number', 'option_weight[' . $i . ']', ts('Order'), $attributes['weight']);
6a488035
TO
314
315 // is active ?
316 $this->add('checkbox', 'option_status[' . $i . ']', ts('Active?'));
317
2bc7601f 318 $this->add('select', 'option_visibility_id[' . $i . ']', ts('Visibility'), $visibilityType);
6a488035
TO
319 $defaultOption[$i] = $this->createElement('radio', NULL, NULL, NULL, $i);
320
321 //for checkbox handling of default option
322 $this->add('checkbox', "default_checkbox_option[$i]", NULL);
323 }
324 //default option selection
325 $this->addGroup($defaultOption, 'default_option');
326 $_showHide->addToTemplate();
327
328 // is_display_amounts
329 $this->add('checkbox', 'is_display_amounts', ts('Display Amount?'));
330
331 // weight
f20d2b0d 332 $this->add('number', 'weight', ts('Order'), CRM_Core_DAO::getAttribute('CRM_Price_DAO_PriceField', 'weight'), TRUE);
6a488035
TO
333 $this->addRule('weight', ts('is a numeric field'), 'numeric');
334
335 // checkbox / radio options per line
336 $this->add('text', 'options_per_line', ts('Options Per Line'));
337 $this->addRule('options_per_line', ts('must be a numeric value'), 'numeric');
338
139f5f76
JP
339 $this->add('textarea', 'help_pre', ts('Pre Field Help'),
340 CRM_Core_DAO::getAttribute('CRM_Price_DAO_PriceField', 'help_post')
341 );
342
6a488035 343 // help post, mask, attributes, javascript ?
139f5f76 344 $this->add('textarea', 'help_post', ts('Post Field Help'),
9da8dc8c 345 CRM_Core_DAO::getAttribute('CRM_Price_DAO_PriceField', 'help_post')
6a488035
TO
346 );
347
be2fb01f 348 $this->add('datepicker', 'active_on', ts('Active On'), [], FALSE, ['time' => TRUE]);
6a488035
TO
349
350 // expire_on
be2fb01f 351 $this->add('datepicker', 'expire_on', ts('Expire On'), [], FALSE, ['time' => TRUE]);
6a488035
TO
352
353 // is required ?
354 $this->add('checkbox', 'is_required', ts('Required?'));
355
356 // is active ?
357 $this->add('checkbox', 'is_active', ts('Active?'));
358
359 // add buttons
be2fb01f
CW
360 $this->addButtons([
361 [
c5c263ca
AH
362 'type' => 'next',
363 'name' => ts('Save'),
364 'isDefault' => TRUE,
be2fb01f
CW
365 ],
366 [
c5c263ca
AH
367 'type' => 'next',
368 'name' => ts('Save and New'),
369 'subName' => 'new',
be2fb01f
CW
370 ],
371 [
c5c263ca
AH
372 'type' => 'cancel',
373 'name' => ts('Cancel'),
be2fb01f
CW
374 ],
375 ]);
6a488035
TO
376 // is public?
377 $this->add('select', 'visibility_id', ts('Visibility'), CRM_Core_PseudoConstant::visibility());
378
379 // add a form rule to check default value
be2fb01f 380 $this->addFormRule(['CRM_Price_Form_Field', 'formRule'], $this);
6a488035
TO
381
382 // if view mode pls freeze it with the done button.
383 if ($this->_action & CRM_Core_Action::VIEW) {
384 $this->freeze();
385 $url = CRM_Utils_System::url('civicrm/admin/price/field', 'reset=1&action=browse&sid=' . $this->_sid);
9d30c8d1 386 $this->addElement('button',
6a488035
TO
387 'done',
388 ts('Done'),
9d30c8d1 389 ['onclick' => "location.href='$url'"]
6a488035
TO
390 );
391 }
392 }
393
394 /**
fe482240 395 * Global validation rules for the form.
6a488035 396 *
414c1420
TO
397 * @param array $fields
398 * Posted values of the form.
77b97be7 399 *
44374cbe 400 * @param array $files
401 * @param self $form
6a488035 402 *
a6c01b45
CW
403 * @return array
404 * if errors then list of errors to be posted back to the form,
6a488035 405 * true otherwise
6a488035 406 */
00be9182 407 public static function formRule($fields, $files, $form) {
6a488035
TO
408
409 // all option fields are of type "money"
be2fb01f 410 $errors = [];
6a488035
TO
411
412 /** Check the option values entered
413 * Appropriate values are required for the selected datatype
414 * Incomplete row checking is also required.
415 */
416 if (($form->_action & CRM_Core_Action::ADD || $form->_action & CRM_Core_Action::UPDATE) &&
44374cbe 417 $fields['html_type'] === 'Text'
6a488035 418 ) {
e68e1c9a
PN
419 if ($fields['price'] == NULL) {
420 $errors['price'] = ts('Price is a required field');
421 }
422 if ($fields['financial_type_id'] == '') {
423 $errors['financial_type_id'] = ts('Financial Type is a required field');
424 }
6a488035 425 }
366fe2a3 426
6a488035 427 //avoid the same price field label in Within PriceSet
9da8dc8c 428 $priceFieldLabel = new CRM_Price_DAO_PriceField();
6a488035
TO
429 $priceFieldLabel->label = $fields['label'];
430 $priceFieldLabel->price_set_id = $form->_sid;
431
432 $dupeLabel = FALSE;
44374cbe 433 if ($priceFieldLabel->find(TRUE) && $form->getEntityId() != $priceFieldLabel->id) {
6a488035
TO
434 $dupeLabel = TRUE;
435 }
436
437 if ($dupeLabel) {
438 $errors['label'] = ts('Name already exists in Database.');
439 }
440
441 if ((is_numeric(CRM_Utils_Array::value('count', $fields)) &&
de6c59ca 442 empty($fields['count'])
6a488035
TO
443 ) &&
444 (CRM_Utils_Array::value('html_type', $fields) == 'Text')
445 ) {
446 $errors['count'] = ts('Participant Count must be greater than zero.');
447 }
448
449 if ($form->_action & CRM_Core_Action::ADD) {
450 if ($fields['html_type'] != 'Text') {
451 $countemptyrows = 0;
a13c171d 452 $publicOptionCount = $_flagOption = $_rowError = 0;
6a488035
TO
453
454 $_showHide = new CRM_Core_ShowHideBlocks('', '');
39868387 455 $visibilityOptions = CRM_Price_BAO_PriceFieldValue::buildOptions('visibility_id', 'validate');
6a488035
TO
456
457 for ($index = 1; $index <= self::NUM_OPTION; $index++) {
458
459 $noLabel = $noAmount = $noWeight = 1;
460 if (!empty($fields['option_label'][$index])) {
461 $noLabel = 0;
462 $duplicateIndex = CRM_Utils_Array::key($fields['option_label'][$index],
463 $fields['option_label']
464 );
465
466 if ((!($duplicateIndex === FALSE)) &&
467 (!($duplicateIndex == $index))
468 ) {
469 $errors["option_label[{$index}]"] = ts('Duplicate label value');
470 $_flagOption = 1;
471 }
472 }
473 if ($form->_useForMember) {
474 if (!empty($fields['membership_type_id'][$index])) {
475 $memTypesIDS[] = $fields['membership_type_id'][$index];
476 }
477 }
478
479 // allow for 0 value.
480 if (!empty($fields['option_amount'][$index]) ||
481 strlen($fields['option_amount'][$index]) > 0
482 ) {
483 $noAmount = 0;
484 }
485
486 if (!empty($fields['option_weight'][$index])) {
487 $noWeight = 0;
488 $duplicateIndex = CRM_Utils_Array::key($fields['option_weight'][$index],
489 $fields['option_weight']
490 );
491
492 if ((!($duplicateIndex === FALSE)) &&
493 (!($duplicateIndex == $index))
494 ) {
495 $errors["option_weight[{$index}]"] = ts('Duplicate weight value');
496 $_flagOption = 1;
497 }
498 }
8cc574cf 499 if (!$noLabel && !$noAmount && !empty($fields['option_financial_type_id']) && $fields['option_financial_type_id'][$index] == '' && $fields['html_type'] != 'Text') {
6a488035 500 $errors["option_financial_type_id[{$index}]"] = ts('Financial Type is a Required field.');
366fe2a3 501 }
6a488035
TO
502 if ($noLabel && !$noAmount) {
503 $errors["option_label[{$index}]"] = ts('Label cannot be empty.');
504 $_flagOption = 1;
505 }
506
507 if (!$noLabel && $noAmount) {
508 $errors["option_amount[{$index}]"] = ts('Amount cannot be empty.');
509 $_flagOption = 1;
510 }
511
512 if ($noLabel && $noAmount) {
513 $countemptyrows++;
514 $_emptyRow = 1;
515 }
516 elseif (!empty($fields['option_max_value'][$index]) &&
517 !empty($fields['option_count'][$index]) &&
518 ($fields['option_count'][$index] > $fields['option_max_value'][$index])
519 ) {
520 $errors["option_max_value[{$index}]"] = ts('Participant count can not be greater than max participants.');
521 $_flagOption = 1;
522 }
523
524 $showBlocks = 'optionField_' . $index;
525 if ($_flagOption) {
526 $_showHide->addShow($showBlocks);
527 $_rowError = 1;
528 }
529
530 if (!empty($_emptyRow)) {
531 $_showHide->addHide($showBlocks);
532 }
533 else {
534 $_showHide->addShow($showBlocks);
535 }
536 if ($index == self::NUM_OPTION) {
537 $hideBlock = 'additionalOption';
538 $_showHide->addHide($hideBlock);
539 }
540
a13c171d
CR
541 if (!empty($fields['option_visibility_id'][$index]) && (!$noLabel || !$noAmount)) {
542 if ($visibilityOptions[$fields['option_visibility_id'][$index]] == 'public') {
543 $publicOptionCount++;
544 }
545 }
546
6a488035
TO
547 $_flagOption = $_emptyRow = 0;
548 }
549
550 if (!empty($memTypesIDS)) {
551 // check for checkboxes allowing user to select multiple memberships from same membership organization
552 if ($fields['html_type'] == 'CheckBox') {
ba1dcfda 553 $foundDuplicate = FALSE;
be2fb01f 554 $orgIds = [];
ba1dcfda
TO
555 foreach ($memTypesIDS as $key => $val) {
556 $org = CRM_Member_BAO_MembershipType::getMembershipTypeOrganization($val);
557 if (in_array($org[$val], $orgIds)) {
558 $foundDuplicate = TRUE;
559 break;
560 }
561 $orgIds[$val] = $org[$val];
6a488035 562
ba1dcfda
TO
563 }
564 if ($foundDuplicate) {
565 $errors['_qf_default'] = ts('You have selected multiple memberships for the same organization or entity. Please review your selections and choose only one membership per entity.');
566 }
6a488035 567 }
366fe2a3 568
6a488035
TO
569 // CRM-10390 - Only one price field in a set can include auto-renew membership options
570 $foundAutorenew = FALSE;
571 foreach ($memTypesIDS as $key => $val) {
572 // see if any price field option values in this price field are for memberships with autorenew
573 $memTypeDetails = CRM_Member_BAO_MembershipType::getMembershipTypeDetails($val);
a7488080 574 if (!empty($memTypeDetails['auto_renew'])) {
6a488035
TO
575 $foundAutorenew = TRUE;
576 break;
577 }
578 }
579
580 if ($foundAutorenew) {
581 // if so, check for other fields in this price set which also have auto-renew membership options
9da8dc8c 582 $otherFieldAutorenew = CRM_Price_BAO_PriceSet::checkAutoRenewForPriceSet($form->_sid);
6a488035
TO
583 if ($otherFieldAutorenew) {
584 $errors['_qf_default'] = ts('You can include auto-renew membership choices for only one price field in a price set. Another field in this set already contains one or more auto-renew membership options.');
585 }
586 }
587 }
588 $_showHide->addToTemplate();
589
831e0fa5 590 if ($countemptyrows == 15) {
6a488035
TO
591 $errors['option_label[1]'] = $errors['option_amount[1]'] = ts('Label and value cannot be empty.');
592 $_flagOption = 1;
593 }
a13c171d
CR
594
595 if ($visibilityOptions[$fields['visibility_id']] == 'public' && $publicOptionCount == 0) {
596 $errors['visibility_id'] = ts('You have selected to make this field public but have not enabled any public price options. Please update your selections to include a public price option, or make this field admin visibility only.');
597 for ($index = 1; $index <= self::NUM_OPTION; $index++) {
598 if (!empty($fields['option_label'][$index]) || !empty($fields['option_amount'][$index])) {
599 $errors["option_visibility_id[{$index}]"] = ts('Public field should at least have one public option.');
600 }
601 }
602 }
603
604 if ($visibilityOptions[$fields['visibility_id']] == 'admin' && $publicOptionCount > 0) {
605 $errors['visibility_id'] = ts('Field with \'Admin\' visibility should only contain \'Admin\' options.');
606
607 for ($index = 1; $index <= self::NUM_OPTION; $index++) {
608
609 $isOptionSet = !empty($fields['option_label'][$index]) || !empty($fields['option_amount'][$index]);
9c1bc317 610 $currentOptionVisibility = $visibilityOptions[$fields['option_visibility_id'][$index]] ?? NULL;
a13c171d
CR
611
612 if ($isOptionSet && $currentOptionVisibility == 'public') {
613 $errors["option_visibility_id[{$index}]"] = ts('\'Admin\' field should only have \'Admin\' visibility options.');
614 }
615 }
616 }
6a488035
TO
617 }
618 elseif (!empty($fields['max_value']) &&
619 !empty($fields['count']) &&
620 ($fields['count'] > $fields['max_value'])
621 ) {
622 $errors['max_value'] = ts('Participant count can not be greater than max participants.');
623 }
624
625 // do not process if no option rows were submitted
626 if (empty($fields['option_amount']) && empty($fields['option_label'])) {
627 return TRUE;
628 }
629
630 if (empty($fields['option_name'])) {
be2fb01f 631 $fields['option_amount'] = [];
6a488035
TO
632 }
633
634 if (empty($fields['option_label'])) {
be2fb01f 635 $fields['option_label'] = [];
6a488035
TO
636 }
637 }
638
639 return empty($errors) ? TRUE : $errors;
640 }
641
642 /**
fe482240 643 * Process the form.
a72a9544 644 *
645 * @throws \CRM_Core_Exception
431c56b2 646 * @throws \CiviCRM_API3_Exception
6a488035
TO
647 */
648 public function postProcess() {
649 // store the submitted values in an array
650 $params = $this->controller->exportValues('Field');
f961a8cd
SL
651 $params['id'] = $this->getEntityId();
652 $priceField = $this->submit($params);
653 if (!is_a($priceField, 'CRM_Core_Error')) {
654 // Required by extensions implementing the postProcess hook (to get the ID of new entities)
655 $this->setEntityId($priceField->id);
656 CRM_Core_Session::setStatus(ts('Price Field \'%1\' has been saved.', [1 => $priceField->label]), ts('Saved'), 'success');
657 }
658 $buttonName = $this->controller->getButtonName();
659 $session = CRM_Core_Session::singleton();
660 if ($buttonName == $this->getButtonName('next', 'new')) {
661 CRM_Core_Session::setStatus(ts(' You can add another price set field.'), '', 'info');
662 $session->replaceUserContext(CRM_Utils_System::url('civicrm/admin/price/field', 'reset=1&action=add&sid=' . $this->_sid));
663 }
664 else {
665 $session->replaceUserContext(CRM_Utils_System::url('civicrm/admin/price/field', 'reset=1&action=browse&sid=' . $this->_sid));
666 }
667 }
668
669 public function submit($params) {
431c56b2 670 $params['price'] = CRM_Utils_Rule::cleanMoney($params['price']);
09e6f541 671 foreach ($params['option_amount'] as $key => $amount) {
672 $params['option_amount'][$key] = CRM_Utils_Rule::cleanMoney($amount);
673 }
6a488035 674
6a488035
TO
675 $params['is_display_amounts'] = CRM_Utils_Array::value('is_display_amounts', $params, FALSE);
676 $params['is_required'] = CRM_Utils_Array::value('is_required', $params, FALSE);
677 $params['is_active'] = CRM_Utils_Array::value('is_active', $params, FALSE);
353ffa53 678 $params['financial_type_id'] = CRM_Utils_Array::value('financial_type_id', $params, FALSE);
6a488035
TO
679 $params['visibility_id'] = CRM_Utils_Array::value('visibility_id', $params, FALSE);
680 $params['count'] = CRM_Utils_Array::value('count', $params, FALSE);
681
682 // need the FKEY - price set id
683 $params['price_set_id'] = $this->_sid;
684
685 if ($this->_action & (CRM_Core_Action::UPDATE | CRM_Core_Action::ADD)) {
be2fb01f 686 $fieldValues = ['price_set_id' => $this->_sid];
6a488035 687 $oldWeight = NULL;
44374cbe 688 if ($this->getEntityId()) {
689 $oldWeight = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceField', $this->getEntityId(), 'weight', 'id');
6a488035 690 }
9da8dc8c 691 $params['weight'] = CRM_Utils_Weight::updateOtherWeights('CRM_Price_DAO_PriceField', $oldWeight, $params['weight'], $fieldValues);
6a488035
TO
692 }
693
694 // make value <=> name consistency.
695 if (isset($params['option_name'])) {
696 $params['option_value'] = $params['option_name'];
697 }
698 $params['is_enter_qty'] = CRM_Utils_Array::value('is_enter_qty', $params, FALSE);
699
a72a9544 700 if ($params['html_type'] === 'Text') {
6a488035
TO
701 // if html type is Text, force is_enter_qty on
702 $params['is_enter_qty'] = 1;
703 // modify params values as per the option group and option
704 // value
be2fb01f
CW
705 $params['option_amount'] = [1 => $params['price']];
706 $params['option_label'] = [1 => $params['label']];
707 $params['option_count'] = [1 => $params['count']];
708 $params['option_max_value'] = [1 => CRM_Utils_Array::value('max_value', $params)];
6a488035 709 //$params['option_description'] = array( 1 => $params['description'] );
be2fb01f
CW
710 $params['option_weight'] = [1 => $params['weight']];
711 $params['option_financial_type_id'] = [1 => $params['financial_type_id']];
712 $params['option_visibility_id'] = [1 => CRM_Utils_Array::value('visibility_id', $params)];
6a488035
TO
713 }
714
6a488035
TO
715 $params['membership_num_terms'] = (!empty($params['membership_type_id'])) ? CRM_Utils_Array::value('membership_num_terms', $params, 1) : NULL;
716
9da8dc8c 717 $priceField = CRM_Price_BAO_PriceField::create($params);
f961a8cd 718 return $priceField;
6a488035 719 }
96025800 720
6a488035 721}