Merge remote-tracking branch 'upstream/4.5' into 4.5-master-2014-10-21-15-20-57
[civicrm-core.git] / CRM / Contribute / Form / Contribution / Main.php
1 <?php
2 /*
3 +--------------------------------------------------------------------+
4 | CiviCRM version 4.5 |
5 +--------------------------------------------------------------------+
6 | Copyright CiviCRM LLC (c) 2004-2014 |
7 +--------------------------------------------------------------------+
8 | This file is a part of CiviCRM. |
9 | |
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. |
13 | |
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. |
18 | |
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 +--------------------------------------------------------------------+
26 */
27
28 /**
29 *
30 * @package CRM
31 * @copyright CiviCRM LLC (c) 2004-2014
32 * $Id$
33 *
34 */
35
36 /**
37 * This class generates form components for processing a Contribution
38 *
39 */
40 class CRM_Contribute_Form_Contribution_Main extends CRM_Contribute_Form_ContributionBase {
41
42 /**
43 *Define default MembershipType Id
44 *
45 */
46 public $_defaultMemTypeId;
47
48 public $_relatedOrganizationFound;
49
50 public $_onBehalfRequired = FALSE;
51 public $_onbehalf = FALSE;
52 public $_paymentProcessors;
53 protected $_defaults;
54
55 public $_membershipTypeValues;
56
57 public $_useForMember;
58
59 protected $_ppType;
60 protected $_snippet;
61
62 /**
63 * Function to set variables up before form is built
64 *
65 * @return void
66 * @access public
67 */
68 public function preProcess() {
69 parent::preProcess();
70
71 self::preProcessPaymentOptions($this);
72
73 // Make the contributionPageID avilable to the template
74 $this->assign('contributionPageID', $this->_id);
75 $this->assign('isShare', CRM_Utils_Array::value('is_share', $this->_values));
76 $this->assign('isConfirmEnabled', CRM_Utils_Array::value('is_confirm_enabled', $this->_values));
77
78 // make sure we have right permission to edit this user
79 $csContactID = $this->getContactID();
80 $reset = CRM_Utils_Request::retrieve('reset', 'Boolean', CRM_Core_DAO::$_nullObject);
81 $mainDisplay = CRM_Utils_Request::retrieve('_qf_Main_display', 'Boolean', CRM_Core_DAO::$_nullObject);
82
83 if ($reset) {
84 $this->assign('reset', $reset);
85 }
86
87 if ($mainDisplay) {
88 $this->assign('mainDisplay', $mainDisplay);
89 }
90
91 // Possible values for 'is_for_organization':
92 // * 0 - org profile disabled
93 // * 1 - org profile optional
94 // * 2 - org profile required
95 $this->_onbehalf = FALSE;
96 if (!empty($this->_values['is_for_organization'])) {
97 if ($this->_values['is_for_organization'] == 2) {
98 $this->_onBehalfRequired = TRUE;
99 }
100 // Add organization profile if 1 of the following are true:
101 // If the org profile is required
102 if ($this->_onBehalfRequired ||
103 // Or we are building the form for the first time
104 empty($_POST) ||
105 // Or the user has submitted the form and checked the "On Behalf" checkbox
106 !empty($_POST['is_for_organization'])
107 ) {
108 $this->_onbehalf = TRUE;
109 CRM_Contribute_Form_Contribution_OnBehalfOf::preProcess($this);
110 }
111 }
112 $this->assign('onBehalfRequired', $this->_onBehalfRequired);
113
114 if ($this->_honor_block_is_active) {
115 CRM_Contact_Form_ProfileContact::preprocess($this);
116 }
117
118 if ($this->_snippet) {
119 $this->assign('isOnBehalfCallback', CRM_Utils_Array::value('onbehalf', $_GET, FALSE));
120 return;
121 }
122
123 if (!empty($this->_pcpInfo['id']) && !empty($this->_pcpInfo['intro_text'])) {
124 $this->assign('intro_text', $this->_pcpInfo['intro_text']);
125 }
126 elseif (!empty($this->_values['intro_text'])) {
127 $this->assign('intro_text', $this->_values['intro_text']);
128 }
129
130 $qParams = "reset=1&amp;id={$this->_id}";
131 if ($pcpId = CRM_Utils_Array::value('pcp_id', $this->_pcpInfo)) {
132 $qParams .= "&amp;pcpId={$pcpId}";
133 }
134 $this->assign('qParams', $qParams);
135
136 if (!empty($this->_values['footer_text'])) {
137 $this->assign('footer_text', $this->_values['footer_text']);
138 }
139
140 //CRM-5001
141 if (!empty($this->_values['is_for_organization'])) {
142 $msg = ts('Mixed profile not allowed for on behalf of registration/sign up.');
143 $ufJoinParams = array(
144 'module' => 'onBehalf',
145 'entity_table' => 'civicrm_contribution_page',
146 'entity_id' => $this->_id,
147 );
148 $onBehalfProfileIDs = CRM_Core_BAO_UFJoin::getUFGroupIds($ufJoinParams);
149 // getUFGroupIDs returns an array with the first item being the ID we need
150 $onBehalfProfileID = $onBehalfProfileIDs[0];
151 if ($onBehalfProfileID) {
152 $onBehalfProfile = CRM_Core_BAO_UFGroup::profileGroups($onBehalfProfileID);
153 foreach (array(
154 'Individual', 'Organization', 'Household') as $contactType) {
155 if (in_array($contactType, $onBehalfProfile) &&
156 (in_array('Membership', $onBehalfProfile) ||
157 in_array('Contribution', $onBehalfProfile)
158 )
159 ) {
160 CRM_Core_Error::fatal($msg);
161 }
162 }
163 }
164
165 if ($postID = CRM_Utils_Array::value('custom_post_id', $this->_values)) {
166 $postProfile = CRM_Core_BAO_UFGroup::profileGroups($postID);
167 foreach (array(
168 'Individual', 'Organization', 'Household') as $contactType) {
169 if (in_array($contactType, $postProfile) &&
170 (in_array('Membership', $postProfile) ||
171 in_array('Contribution', $postProfile)
172 )
173 ) {
174 CRM_Core_Error::fatal($msg);
175 }
176 }
177 }
178 }
179 }
180
181 /**
182 * set the default values
183 *
184 * @return void
185 * @access public
186 */
187 /**
188 *
189 */
190 function setDefaultValues() {
191 // check if the user is registered and we have a contact ID
192 $contactID = $this->getContactID();
193
194 if (!empty($contactID)) {
195 $fields = array();
196 $removeCustomFieldTypes = array('Contribution', 'Membership');
197 $contribFields = CRM_Contribute_BAO_Contribution::getContributionFields();
198
199 // remove component related fields
200 foreach ($this->_fields as $name => $dontCare) {
201 //don't set custom data Used for Contribution (CRM-1344)
202 if (substr($name, 0, 7) == 'custom_') {
203 $id = substr($name, 7);
204 if (!CRM_Core_BAO_CustomGroup::checkCustomField($id, $removeCustomFieldTypes)) {
205 continue;
206 }
207 // ignore component fields
208 }
209 elseif (array_key_exists($name, $contribFields) || (substr($name, 0, 11) == 'membership_') || (substr($name, 0, 13) == 'contribution_')) {
210 continue;
211 }
212 $fields[$name] = 1;
213 }
214
215 if (!empty($fields)) {
216 CRM_Core_BAO_UFGroup::setProfileDefaults($contactID, $fields, $this->_defaults);
217 }
218
219 $billingDefaults = $this->getProfileDefaults('Billing', $contactID);
220 $this->_defaults = array_merge($this->_defaults, $billingDefaults);
221 }
222
223 //set custom field defaults set by admin if value is not set
224 if (!empty($this->_fields)) {
225 //load default campaign from page.
226 if (array_key_exists('contribution_campaign_id', $this->_fields)) {
227 $this->_defaults['contribution_campaign_id'] = CRM_Utils_Array::value('campaign_id', $this->_values);
228 }
229
230 //set custom field defaults
231 foreach ($this->_fields as $name => $field) {
232 if ($customFieldID = CRM_Core_BAO_CustomField::getKeyID($name)) {
233 if (!isset($this->_defaults[$name])) {
234 CRM_Core_BAO_CustomField::setProfileDefaults($customFieldID, $name, $this->_defaults,
235 NULL, CRM_Profile_Form::MODE_REGISTER
236 );
237 }
238 }
239 }
240 }
241
242 // // hack to simplify credit card entry for testing
243 // $this->_defaults['credit_card_type'] = 'Visa';
244 // $this->_defaults['amount'] = 168;
245 // $this->_defaults['credit_card_number'] = '4111111111111111';
246 // $this->_defaults['cvv2'] = '000';
247 // $this->_defaults['credit_card_exp_date'] = array('Y' => date('Y')+1, 'M' => '05');
248
249 // // hack to simplify direct debit entry for testing
250 // $this->_defaults['account_holder'] = 'Max Müller';
251 // $this->_defaults['bank_account_number'] = '12345678';
252 // $this->_defaults['bank_identification_number'] = '12030000';
253 // $this->_defaults['bank_name'] = 'Bankname';
254
255 //build set default for pledge overdue payment.
256 if (!empty($this->_values['pledge_id'])) {
257 //get all pledge payment records of current pledge id.
258 $pledgePayments = array();
259
260 //used to record completed pledge payment ids used later for honor default
261 $completedContributionIds = array();
262
263 $pledgePayments = CRM_Pledge_BAO_PledgePayment::getPledgePayments($this->_values['pledge_id']);
264
265 $duePayment = FALSE;
266 foreach ($pledgePayments as $payId => $value) {
267 if ($value['status'] == 'Overdue') {
268 $this->_defaults['pledge_amount'][$payId] = 1;
269 }
270 elseif (!$duePayment && $value['status'] == 'Pending') {
271 $this->_defaults['pledge_amount'][$payId] = 1;
272 $duePayment = TRUE;
273 }
274 elseif ($value['status'] == 'Completed' && $value['contribution_id']) {
275 $completedContributionIds[] = $value['contribution_id'];
276 }
277 }
278
279 if ($this->_honor_block_is_active && count($completedContributionIds)) {
280 $softCredit = array();
281 foreach ($completedContributionIds as $id) {
282 $softCredit = CRM_Contribute_BAO_ContributionSoft::getSoftContribution($id);
283 }
284 if (isset($softCredit['soft_credit'])) {
285 $this->_defaults['soft_credit_type_id'] = $softCredit['soft_credit'][1]['soft_credit_type'];
286
287 //since honoree profile fieldname of fields are prefixed with 'honor'
288 //we need to reformat the fieldname to append prefix during setting default values
289 CRM_Core_BAO_UFGroup::setProfileDefaults(
290 $softCredit['soft_credit'][1]['contact_id'],
291 CRM_Core_BAO_UFGroup::getFields($this->_honoreeProfileId),
292 $defaults
293 );
294 foreach ($defaults as $fieldName => $value) {
295 $this->_defaults['honor[' . $fieldName . ']'] = $value;
296 }
297 }
298 }
299 }
300 elseif (!empty($this->_values['pledge_block_id'])) {
301 //set default to one time contribution.
302 $this->_defaults['is_pledge'] = 0;
303 }
304
305 // to process Custom data that are appended to URL
306 $getDefaults = CRM_Core_BAO_CustomGroup::extractGetParams($this, "'Contact', 'Individual', 'Contribution'");
307 if (!empty($getDefaults)) {
308 $this->_defaults = array_merge($this->_defaults, $getDefaults);
309 }
310
311 $config = CRM_Core_Config::singleton();
312 // set default country from config if no country set
313 if (empty($this->_defaults["billing_country_id-{$this->_bltID}"])) {
314 $this->_defaults["billing_country_id-{$this->_bltID}"] = $config->defaultContactCountry;
315 }
316
317 // set default state/province from config if no state/province set
318 if (empty($this->_defaults["billing_state_province_id-{$this->_bltID}"])) {
319 $this->_defaults["billing_state_province_id-{$this->_bltID}"] = $config->defaultContactStateProvince;
320 }
321
322 if ($this->_priceSetId) {
323 if (($this->_useForMember && !empty($this->_currentMemberships)) || $this->_defaultMemTypeId) {
324 $selectedCurrentMemTypes = array();
325 foreach ($this->_priceSet['fields'] as $key => $val) {
326 foreach ($val['options'] as $keys => $values) {
327 $opMemTypeId = CRM_Utils_Array::value('membership_type_id', $values);
328 if ($opMemTypeId &&
329 in_array($opMemTypeId, $this->_currentMemberships) &&
330 !in_array($opMemTypeId, $selectedCurrentMemTypes)
331 ) {
332 if ($val['html_type'] == 'CheckBox') {
333 $this->_defaults["price_{$key}"][$keys] = 1;
334 }
335 else {
336 $this->_defaults["price_{$key}"] = $keys;
337 }
338 $selectedCurrentMemTypes[] = $values['membership_type_id'];
339 }
340 elseif (!empty($values['is_default']) &&
341 !$opMemTypeId &&
342 (!isset($this->_defaults["price_{$key}"]) ||
343 ($val['html_type'] == 'CheckBox' && !isset($this->_defaults["price_{$key}"][$keys]))
344 )
345 ) {
346 if ($val['html_type'] == 'CheckBox') {
347 $this->_defaults["price_{$key}"][$keys] = 1;
348 }
349 else {
350 $this->_defaults["price_{$key}"] = $keys;
351 }
352 }
353 }
354 }
355 }
356 else {
357 CRM_Price_BAO_PriceSet::setDefaultPriceSet($this, $this->_defaults);
358 }
359 }
360
361 if (!empty($this->_paymentProcessors)) {
362 foreach ($this->_paymentProcessors as $pid => $value) {
363 if (!empty($value['is_default'])) {
364 $this->_defaults['payment_processor'] = $pid;
365 }
366 }
367 }
368
369 return $this->_defaults;
370 }
371
372 /**
373 * Function to build the form
374 *
375 * @return void
376 * @access public
377 */
378 public function buildQuickForm() {
379 // build profiles first so that we can determine address fields etc
380 // and then show copy address checkbox
381 $this->buildCustom($this->_values['custom_pre_id'], 'customPre');
382 $this->buildCustom($this->_values['custom_post_id'], 'customPost');
383
384 if (!empty($this->_fields) && !empty($this->_values['custom_pre_id'])) {
385 $profileAddressFields = array();
386 foreach ($this->_fields as $key => $value) {
387 CRM_Core_BAO_UFField::assignAddressField($key, $profileAddressFields, array('uf_group_id' => $this->_values['custom_pre_id']));
388 }
389 $this->set('profileAddressFields', $profileAddressFields);
390 }
391
392 // Build payment processor form
393 if (($this->_ppType || $this->_isBillingAddressRequiredForPayLater) && empty($_GET['onbehalf'])) {
394 CRM_Core_Payment_ProcessorForm::buildQuickForm($this);
395 // Return if we are in an ajax callback
396 if ($this->_snippet) {
397 return;
398 }
399 }
400
401 $config = CRM_Core_Config::singleton();
402
403 if ($this->_onbehalf) {
404 CRM_Contribute_Form_Contribution_OnBehalfOf::buildQuickForm($this);
405 // Return if we are in an ajax callback
406 if ($this->_snippet) {
407 return;
408 }
409 }
410
411 $this->applyFilter('__ALL__', 'trim');
412 $this->add('text', "email-{$this->_bltID}",
413 ts('Email Address'),
414 array('size' => 30, 'maxlength' => 60, 'class' => 'email'),
415 TRUE
416 );
417 $this->addRule("email-{$this->_bltID}", ts('Email is not valid.'), 'email');
418 $pps = array();
419 $onlinePaymentProcessorEnabled = FALSE;
420 if (!empty($this->_paymentProcessors)) {
421 foreach ($this->_paymentProcessors as $key => $name) {
422 if($name['billing_mode'] == 1) {
423 $onlinePaymentProcessorEnabled = TRUE;
424 }
425 $pps[$key] = $name['name'];
426 }
427 }
428 if (!empty($this->_values['is_pay_later'])) {
429 $pps[0] = $this->_values['pay_later_text'];
430 }
431
432 if (count($pps) > 1) {
433 $this->addRadio('payment_processor', ts('Payment Method'), $pps,
434 NULL, "&nbsp;", TRUE
435 );
436 }
437 elseif (!empty($pps)) {
438 $key = array_keys($pps);
439 $key = array_pop($key);
440 $this->addElement('hidden', 'payment_processor', $key);
441 if ($key === 0) {
442 $this->assign('is_pay_later', $this->_values['is_pay_later']);
443 $this->assign('pay_later_text', $this->_values['pay_later_text']);
444 }
445 }
446
447 $contactID = $this->getContactID();
448 if($this->getContactID() === '0') {
449 $this->addCidZeroOptions($onlinePaymentProcessorEnabled);
450 }
451 //build pledge block.
452 $this->_useForMember = 0;
453 //don't build membership block when pledge_id is passed
454 if (empty($this->_values['pledge_id'])) {
455 $this->_separateMembershipPayment = FALSE;
456 if (in_array('CiviMember', $config->enableComponents)) {
457 $isTest = 0;
458 if ($this->_action & CRM_Core_Action::PREVIEW) {
459 $isTest = 1;
460 }
461
462 if ($this->_priceSetId &&
463 (CRM_Core_Component::getComponentID('CiviMember') == CRM_Utils_Array::value('extends', $this->_priceSet))
464 ) {
465 $this->_useForMember = 1;
466 $this->set('useForMember', $this->_useForMember);
467 }
468
469 $this->_separateMembershipPayment = CRM_Member_BAO_Membership::buildMembershipBlock($this,
470 $this->_id,
471 $this->_membershipContactID,
472 TRUE, NULL, FALSE,
473 $isTest
474 );
475 }
476 $this->set('separateMembershipPayment', $this->_separateMembershipPayment);
477 }
478 $this->assign('useForMember', $this->_useForMember);
479 // If we configured price set for contribution page
480 // we are not allow membership signup as well as any
481 // other contribution amount field, CRM-5095
482 if (isset($this->_priceSetId) && $this->_priceSetId) {
483 $this->add('hidden', 'priceSetId', $this->_priceSetId);
484 // build price set form.
485 $this->set('priceSetId', $this->_priceSetId);
486 CRM_Price_BAO_PriceSet::buildPriceSet($this);
487 if ($this->_values['is_monetary'] &&
488 $this->_values['is_recur'] && empty($this->_values['pledge_id'])) {
489 self::buildRecur($this);
490 }
491 }
492
493 if ($this->_priceSetId) {
494 $is_quick_config = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $this->_priceSetId, 'is_quick_config');
495 if ($is_quick_config) {
496 $this->_useForMember = 0;
497 $this->set('useForMember', $this->_useForMember);
498 }
499 }
500
501 if ($this->_values['is_for_organization']) {
502 $this->buildOnBehalfOrganization();
503 }
504
505 //we allow premium for pledge during pledge creation only.
506 if (empty($this->_values['pledge_id'])) {
507 CRM_Contribute_BAO_Premium::buildPremiumBlock($this, $this->_id, TRUE);
508 }
509
510 //add honor block
511 if ($this->_honor_block_is_active) {
512 $this->assign('honor_block_is_active', TRUE);
513
514 //build soft-credit section
515 CRM_Contribute_Form_SoftCredit::buildQuickForm($this);
516 //build honoree profile section
517 CRM_Contact_Form_ProfileContact::buildQuickForm($this);
518 }
519
520
521 //don't build pledge block when mid is passed
522 if (!$this->_mid) {
523 $config = CRM_Core_Config::singleton();
524 if (in_array('CiviPledge', $config->enableComponents) && !empty($this->_values['pledge_block_id'])) {
525 CRM_Pledge_BAO_PledgeBlock::buildPledgeBlock($this);
526 }
527 }
528
529 //to create an cms user
530 if (!$this->_userID) {
531 $createCMSUser = FALSE;
532
533 if ($this->_values['custom_pre_id']) {
534 $profileID = $this->_values['custom_pre_id'];
535 $createCMSUser = CRM_Core_DAO::getFieldValue('CRM_Core_DAO_UFGroup', $profileID, 'is_cms_user');
536 }
537
538 if (!$createCMSUser &&
539 $this->_values['custom_post_id']
540 ) {
541 if (!is_array($this->_values['custom_post_id'])) {
542 $profileIDs = array($this->_values['custom_post_id']);
543 }
544 else {
545 $profileIDs = $this->_values['custom_post_id'];
546 }
547 foreach ($profileIDs as $pid) {
548 if (CRM_Core_DAO::getFieldValue('CRM_Core_DAO_UFGroup', $pid, 'is_cms_user')) {
549 $profileID = $pid;
550 $createCMSUser = TRUE;
551 break;
552 }
553 }
554 }
555
556 if ($createCMSUser) {
557 CRM_Core_BAO_CMSUser::buildForm($this, $profileID, TRUE);
558 }
559 }
560 if ($this->_pcpId) {
561 if ($pcpSupporter = CRM_PCP_BAO_PCP::displayName($this->_pcpId)) {
562 $pcp_supporter_text = ts('This contribution is being made thanks to the effort of <strong>%1</strong>, who supports our campaign.', array(1 => $pcpSupporter));
563 // Only tell people that can also create a PCP if the contribution page has a non-empty value in the "Create Personal Campaign Page link" field.
564 $text = CRM_PCP_BAO_PCP::getPcpBlockStatus($this->_id, 'contribute');
565 if(!empty($text)) {
566 $pcp_supporter_text .= ts("You can support it as well - once you complete the donation, you will be able to create your own Personal Campaign Page!");
567 }
568 $this->assign('pcpSupporterText', $pcp_supporter_text);
569 }
570 $prms = array('id' => $this->_pcpId);
571 CRM_Core_DAO::commonRetrieve('CRM_PCP_DAO_PCP', $prms, $pcpInfo);
572 if ($pcpInfo['is_honor_roll']) {
573 $this->assign('isHonor', TRUE);
574 $this->add('checkbox', 'pcp_display_in_roll', ts('Show my contribution in the public honor roll'), NULL, NULL,
575 array('onclick' => "showHideByValue('pcp_display_in_roll','','nameID|nickID|personalNoteID','block','radio',false); pcpAnonymous( );")
576 );
577 $extraOption = array('onclick' => "return pcpAnonymous( );");
578 $elements = array();
579 $elements[] = &$this->createElement('radio', NULL, '', ts('Include my name and message'), 0, $extraOption);
580 $elements[] = &$this->createElement('radio', NULL, '', ts('List my contribution anonymously'), 1, $extraOption);
581 $this->addGroup($elements, 'pcp_is_anonymous', NULL, '&nbsp;&nbsp;&nbsp;');
582 $this->setDefaults(array('pcp_display_in_roll' => 1));
583 $this->setDefaults(array('pcp_is_anonymous' => 1));
584
585 $this->add('text', 'pcp_roll_nickname', ts('Name'), array('maxlength' => 30));
586 $this->add('textarea', 'pcp_personal_note', ts('Personal Note'), array('style' => 'height: 3em; width: 40em;'));
587 }
588 }
589
590 //we have to load confirm contribution button in template
591 //when multiple payment processor as the user
592 //can toggle with payment processor selection
593 $billingModePaymentProcessors = 0;
594 if ( !empty( $this->_paymentProcessors ) ) {
595 foreach ($this->_paymentProcessors as $key => $values) {
596 if ($values['billing_mode'] == CRM_Core_Payment::BILLING_MODE_BUTTON) {
597 $billingModePaymentProcessors++;
598 }
599 }
600 }
601
602 if ($billingModePaymentProcessors && count($this->_paymentProcessors) == $billingModePaymentProcessors) {
603 $allAreBillingModeProcessors = TRUE;
604 } else {
605 $allAreBillingModeProcessors = FALSE;
606 }
607
608 if (!($allAreBillingModeProcessors && !$this->_values['is_pay_later'])) {
609 $submitButton = array(
610 'type' => 'upload',
611 'name' => CRM_Utils_Array::value('is_confirm_enabled', $this->_values) ? ts('Confirm Contribution') : ts('Contribute'),
612 'spacing' => '&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;',
613 'isDefault' => TRUE,
614 );
615 // Add submit-once behavior when confirm page disabled
616 if (empty($this->_values['is_confirm_enabled'])) {
617 $submitButton['js'] = array('onclick' => "return submitOnce(this,'" . $this->_name . "','" . ts('Processing') . "');");
618 }
619 $this->addButtons(array($submitButton));
620 }
621
622 $this->addFormRule(array('CRM_Contribute_Form_Contribution_Main', 'formRule'), $this);
623 }
624
625 /**
626 * build elements to enable pay on behalf of an organization.
627 *
628 * @access public
629 */
630 function buildOnBehalfOrganization() {
631 if ($this->_membershipContactID) {
632 $entityBlock = array('contact_id' => $this->_membershipContactID);
633 CRM_Core_BAO_Location::getValues($entityBlock, $this->_defaults);
634 }
635
636 if (!$this->_onBehalfRequired) {
637 $this->addElement('checkbox', 'is_for_organization',
638 $this->_values['for_organization'],
639 NULL, array('onclick' => "showOnBehalf( );")
640 );
641 }
642
643 $this->assign('is_for_organization', TRUE);
644 $this->assign('urlPath', 'civicrm/contribute/transact');
645 }
646
647 /**
648 * build elements to collect information for recurring contributions
649 *
650 * @access public
651 */
652 public static function buildRecur(&$form) {
653 $attributes = CRM_Core_DAO::getAttribute('CRM_Contribute_DAO_ContributionRecur');
654 $className = get_class($form);
655
656 $form->assign('is_recur_interval', CRM_Utils_Array::value('is_recur_interval', $form->_values));
657 $form->assign('is_recur_installments', CRM_Utils_Array::value('is_recur_installments', $form->_values));
658
659 $form->add('checkbox', 'is_recur', ts('I want to contribute this amount'), NULL);
660
661 if (!empty($form->_values['is_recur_interval']) || $className == 'CRM_Contribute_Form_Contribution') {
662 $form->add('text', 'frequency_interval', ts('Every'), $attributes['frequency_interval']);
663 $form->addRule('frequency_interval', ts('Frequency must be a whole number (EXAMPLE: Every 3 months).'), 'integer');
664 }
665 else {
666 // make sure frequency_interval is submitted as 1 if given no choice to user.
667 $form->add('hidden', 'frequency_interval', 1);
668 }
669
670 $frUnits = CRM_Utils_Array::value('recur_frequency_unit', $form->_values);
671 if (empty($frUnits) &&
672 $className == 'CRM_Contribute_Form_Contribution'
673 ) {
674 $frUnits = implode(CRM_Core_DAO::VALUE_SEPARATOR,
675 CRM_Core_OptionGroup::values('recur_frequency_units')
676 );
677 }
678
679 $unitVals = explode(CRM_Core_DAO::VALUE_SEPARATOR, $frUnits);
680
681 // CRM 10860, display text instead of a dropdown if there's only 1 frequency unit
682 if(sizeof($unitVals) == 1) {
683 $form->assign('one_frequency_unit', true);
684 $unit = $unitVals[0];
685 $form->add('hidden', 'frequency_unit', $unit);
686 if (!empty($form->_values['is_recur_interval']) || $className == 'CRM_Contribute_Form_Contribution') {
687 $unit .= "(s)";
688 }
689 $form->assign('frequency_unit', $unit);
690 } else {
691 $form->assign('one_frequency_unit', false);
692 $units = array();
693 $frequencyUnits = CRM_Core_OptionGroup::values('recur_frequency_units');
694 foreach ($unitVals as $key => $val) {
695 if (array_key_exists($val, $frequencyUnits)) {
696 $units[$val] = $frequencyUnits[$val];
697 if (!empty($form->_values['is_recur_interval']) || $className == 'CRM_Contribute_Form_Contribution') {
698 $units[$val] = "{$frequencyUnits[$val]}(s)";
699 }
700 }
701 }
702 $frequencyUnit = &$form->add('select', 'frequency_unit', NULL, $units);
703 }
704
705
706 // FIXME: Ideally we should freeze select box if there is only
707 // one option but looks there is some problem /w QF freeze.
708 //if ( count( $units ) == 1 ) {
709 //$frequencyUnit->freeze( );
710 //}
711
712 $form->add('text', 'installments', ts('installments'),
713 $attributes['installments']
714 );
715 $form->addRule('installments', ts('Number of installments must be a whole number.'), 'integer');
716 }
717
718 /**
719 * global form rule
720 *
721 * @param array $fields the input form values
722 * @param array $files the uploaded files if any
723 * @param $self
724 *
725 * @internal param array $options additional user data
726 *
727 * @return true if no errors, else array of errors
728 * @access public
729 * @static
730 */
731 static function formRule($fields, $files, $self) {
732 $errors = array();
733 $amount = self::computeAmount($fields, $self);
734
735 if ((!empty($fields['selectMembership']) &&
736 $fields['selectMembership'] != 'no_thanks'
737 ) ||
738 (!empty($fields['priceSetId']) &&
739 $self->_useForMember
740 )
741 ) {
742 $lifeMember = CRM_Member_BAO_Membership::getAllContactMembership($self->_userID, FALSE, TRUE);
743
744 $membershipOrgDetails = CRM_Member_BAO_MembershipType::getMembershipTypeOrganization();
745
746 $unallowedOrgs = array();
747 foreach (array_keys($lifeMember) as $memTypeId) {
748 $unallowedOrgs[] = $membershipOrgDetails[$memTypeId];
749 }
750 }
751
752 //check for atleast one pricefields should be selected
753 if (!empty($fields['priceSetId'])) {
754 $priceField = new CRM_Price_DAO_PriceField();
755 $priceField->price_set_id = $fields['priceSetId'];
756 $priceField->orderBy('weight');
757 $priceField->find();
758
759 $check = array();
760 $membershipIsActive = TRUE;
761 $previousId = $otherAmount = FALSE;
762 while ($priceField->fetch()) {
763
764 if ($self->_quickConfig && ($priceField->name == 'contribution_amount' || $priceField->name == 'membership_amount')) {
765 $previousId = $priceField->id;
766 if ($priceField->name == 'membership_amount' && !$priceField->is_active ) {
767 $membershipIsActive = FALSE;
768 }
769 }
770 if ($priceField->name == 'other_amount') {
771 if ($self->_quickConfig && empty($fields["price_{$priceField->id}"]) &&
772 array_key_exists("price_{$previousId}", $fields) && isset($fields["price_{$previousId}"]) && $self->_values['fee'][$previousId]['name'] == 'contribution_amount' && empty($fields["price_{$previousId}"])) {
773 $otherAmount = $priceField->id;
774 }
775 elseif (!empty($fields["price_{$priceField->id}"])) {
776 $otherAmountVal = $fields["price_{$priceField->id}"];
777 $min = CRM_Utils_Array::value('min_amount', $self->_values);
778 $max = CRM_Utils_Array::value('max_amount', $self->_values);
779 if ($min && $otherAmountVal < $min) {
780 $errors["price_{$priceField->id}"] = ts('Contribution amount must be at least %1',
781 array(1 => $min)
782 );
783 }
784 if ($max && $otherAmountVal > $max) {
785 $errors["price_{$priceField->id}"] = ts('Contribution amount cannot be more than %1.',
786 array(1 => $max)
787 );
788 }
789 }
790 }
791 if (!empty($fields["price_{$priceField->id}"]) || ($previousId == $priceField->id && isset($fields["price_{$previousId}"])
792 && empty($fields["price_{$previousId}"]))) {
793 $check[] = $priceField->id;
794 }
795 }
796
797 $currentMemberships = NULL;
798 if ($membershipIsActive) {
799 $is_test = $self->_mode != 'live' ? 1 : 0;
800 $memContactID = $self->_membershipContactID;
801
802 // For anonymous user check using dedupe rule
803 // if user has Cancelled Membership
804 if (!$memContactID) {
805 $dedupeParams = CRM_Dedupe_Finder::formatParams($fields, 'Individual');
806 $dedupeParams['check_permission'] = FALSE;
807 $ids = CRM_Dedupe_Finder::dupesByParams($dedupeParams, 'Individual');
808 // if we find more than one contact, use the first one
809 $memContactID = CRM_Utils_Array::value(0, $ids);
810 }
811 $currentMemberships = CRM_Member_BAO_Membership::getContactsCancelledMembership($memContactID,
812 $is_test
813 );
814
815 $errorText = 'Your %1 membership was previously cancelled and can not be renewed online. Please contact the site administrator for assistance.';
816 foreach ($self->_values['fee'] as $fieldKey => $fieldValue) {
817 if ($fieldValue['html_type'] != 'Text' && CRM_Utils_Array::value('price_' . $fieldKey, $fields)) {
818 if (!is_array($fields['price_' . $fieldKey])) {
819 if (array_key_exists('membership_type_id', $fieldValue['options'][$fields['price_' . $fieldKey]])
820 && in_array($fieldValue['options'][$fields['price_' . $fieldKey]]['membership_type_id'], $currentMemberships)) {
821 $errors['price_' . $fieldKey] = ts($errorText, array(1 => CRM_Member_PseudoConstant::membershipType($fieldValue['options'][$fields['price_' . $fieldKey]]['membership_type_id'])));
822 }
823 }
824 else {
825 foreach ($fields['price_' . $fieldKey] as $key => $ignore) {
826 if (array_key_exists('membership_type_id', $fieldValue['options'][$key])
827 && in_array($fieldValue['options'][$key]['membership_type_id'], $currentMemberships)) {
828 $errors['price_' . $fieldKey] = ts($errorText, array(1 => CRM_Member_PseudoConstant::membershipType($fieldValue['options'][$key]['membership_type_id'])));
829 }
830 }
831 }
832 }
833 }
834 }
835
836 // CRM-12233
837 if ($membershipIsActive && !$self->_membershipBlock['is_required']
838 && $self->_values['amount_block_is_active']) {
839 $membershipFieldId = $contributionFieldId = $errorKey = $otherFieldId = NULL;
840 foreach ($self->_values['fee'] as $fieldKey => $fieldValue) {
841 // if 'No thank you' membership is selected then set $membershipFieldId
842 if ($fieldValue['name'] == 'membership_amount' && CRM_Utils_Array::value('price_' . $fieldKey, $fields) == 0) {
843 $membershipFieldId = $fieldKey;
844 }
845 elseif ($membershipFieldId) {
846 if ($fieldValue['name'] == 'other_amount') {
847 $otherFieldId = $fieldKey;
848 }
849 elseif ($fieldValue['name'] == 'contribution_amount') {
850 $contributionFieldId = $fieldKey;
851 }
852
853 if (!$errorKey || CRM_Utils_Array::value('price_' . $contributionFieldId, $fields) == '0') {
854 $errorKey = $fieldKey;
855 }
856 }
857 }
858 // $membershipFieldId is set and additional amount is 'No thank you' or NULL then throw error
859 if ($membershipFieldId && !(CRM_Utils_Array::value('price_' . $contributionFieldId, $fields, -1) > 0) && empty($fields['price_' . $otherFieldId])) {
860 $errors["price_{$errorKey}"] = ts('Additional Contribution is required.');
861 }
862 }
863 if (empty($check)) {
864 if ($self->_useForMember == 1 && $membershipIsActive) {
865 $errors['_qf_default'] = ts('Select at least one option from Membership Type(s).');
866 }
867 else {
868 $errors['_qf_default'] = ts('Select at least one option from Contribution(s).');
869 }
870 }
871 if($otherAmount && !empty($check)) {
872 $errors["price_{$otherAmount}"] = ts('Amount is required field.');
873 }
874
875 if ($self->_useForMember == 1 && !empty($check) && $membershipIsActive) {
876 $priceFieldIDS = array();
877 $priceFieldMemTypes = array();
878
879 foreach ($self->_priceSet['fields'] as $priceId => $value) {
880 if (!empty($fields['price_' . $priceId]) || ($self->_quickConfig && $value['name'] == 'membership_amount' && empty($self->_membershipBlock['is_required']))) {
881 if (!empty($fields['price_' . $priceId]) && is_array($fields['price_' . $priceId])) {
882 foreach ($fields['price_' . $priceId] as $priceFldVal => $isSet) {
883 if ($isSet) {
884 $priceFieldIDS[] = $priceFldVal;
885 }
886 }
887 }
888 elseif (!$value['is_enter_qty'] && !empty($fields['price_' . $priceId])) {
889 // The check for {!$value['is_enter_qty']} is done since, quantity fields allow entering
890 // quantity. And the quantity can't be conisdered as civicrm_price_field_value.id, CRM-9577
891 $priceFieldIDS[] = $fields['price_' . $priceId];
892 }
893
894 if (!empty($value['options'])) {
895 foreach ($value['options'] as $val) {
896 if (!empty($val['membership_type_id']) && (
897 ($fields['price_' . $priceId] == $val['id']) ||
898 (isset($fields['price_' . $priceId]) && !empty($fields['price_' . $priceId][$val['id']]))
899 )
900 ) {
901 $priceFieldMemTypes[] = $val['membership_type_id'];
902 }
903 }
904 }
905 }
906 }
907
908 if (!empty($lifeMember)) {
909 foreach ($priceFieldIDS as $priceFieldId) {
910 if (($id = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceFieldValue', $priceFieldId, 'membership_type_id')) &&
911 in_array($membershipOrgDetails[$id], $unallowedOrgs)
912 ) {
913 $errors['_qf_default'] = ts('You already have a lifetime membership and cannot select a membership with a shorter term.');
914 break;
915 }
916 }
917 }
918
919 if (!empty($priceFieldIDS)) {
920 $ids = implode(',', $priceFieldIDS);
921
922 $priceFieldIDS['id'] = $fields['priceSetId'];
923 $self->set('memberPriceFieldIDS', $priceFieldIDS);
924 $count = CRM_Price_BAO_PriceSet::getMembershipCount($ids);
925 foreach ($count as $id => $occurance) {
926 if ($occurance > 1) {
927 $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. Contact the site administrator if you need assistance.');
928 }
929 }
930 }
931
932 if (empty($priceFieldMemTypes)) {
933 $errors['_qf_default'] = ts('Please select at least one membership option.');
934 }
935 }
936
937 CRM_Price_BAO_PriceSet::processAmount($self->_values['fee'],
938 $fields, $lineItem
939 );
940
941 if ($fields['amount'] < 0) {
942 $errors['_qf_default'] = ts('Contribution can not be less than zero. Please select the options accordingly');
943 }
944 $amount = $fields['amount'];
945 }
946
947 if (isset($fields['selectProduct']) &&
948 $fields['selectProduct'] != 'no_thanks') {
949 $productDAO = new CRM_Contribute_DAO_Product();
950 $productDAO->id = $fields['selectProduct'];
951 $productDAO->find(TRUE);
952 $min_amount = $productDAO->min_contribution;
953
954 if ($amount < $min_amount) {
955 $errors['selectProduct'] = ts('The premium you have selected requires a minimum contribution of %1', array(1 => CRM_Utils_Money::format($min_amount)));
956 CRM_Core_Session::setStatus($errors['selectProduct']);
957 }
958 }
959
960 if (!empty($fields['is_recur'])) {
961 if ($fields['frequency_interval'] <= 0) {
962 $errors['frequency_interval'] = ts('Please enter a number for how often you want to make this recurring contribution (EXAMPLE: Every 3 months).');
963 }
964 if ($fields['frequency_unit'] == '0') {
965 $errors['frequency_unit'] = ts('Please select a period (e.g. months, years ...) for how often you want to make this recurring contribution (EXAMPLE: Every 3 MONTHS).');
966 }
967 }
968
969 if (!empty($fields['is_recur']) &&
970 CRM_Utils_Array::value('payment_processor', $fields) == 0) {
971 $errors['_qf_default'] = ts('You cannot set up a recurring contribution if you are not paying online by credit card.');
972 }
973
974 if (!empty($fields['is_for_organization']) &&
975 !property_exists($self, 'organizationName')
976 ) {
977
978 if (empty($fields['onbehalf']['organization_name'])) {
979 if (!empty($fields['org_option']) && !$fields['onbehalfof_id']) {
980 $errors['organization_id'] = ts('Please select an organization or enter a new one.');
981 }
982 elseif (empty($fields['org_option'])) {
983 $errors['onbehalf']['organization_name'] = ts('Please enter the organization name.');
984 }
985 }
986
987 foreach ($fields['onbehalf'] as $key => $value) {
988 if (strstr($key, 'email')) {
989 $emailLocType = explode('-', $key);
990 }
991 }
992 if (empty($fields['onbehalf']["email-{$emailLocType[1]}"])) {
993 $errors['onbehalf']["email-{$emailLocType[1]}"] = ts('Organization email is required.');
994 }
995 }
996
997 // validate PCP fields - if not anonymous, we need a nick name value
998 if ($self->_pcpId && !empty($fields['pcp_display_in_roll']) &&
999 (CRM_Utils_Array::value('pcp_is_anonymous', $fields) == 0) &&
1000 CRM_Utils_Array::value('pcp_roll_nickname', $fields) == ''
1001 ) {
1002 $errors['pcp_roll_nickname'] = ts('Please enter a name to include in the Honor Roll, or select \'contribute anonymously\'.');
1003 }
1004
1005 // return if this is express mode
1006 $config = CRM_Core_Config::singleton();
1007 if ($self->_paymentProcessor &&
1008 $self->_paymentProcessor['billing_mode'] & CRM_Core_Payment::BILLING_MODE_BUTTON
1009 ) {
1010 if (!empty($fields[$self->_expressButtonName . '_x']) || !empty($fields[$self->_expressButtonName . '_y']) ||
1011 CRM_Utils_Array::value($self->_expressButtonName, $fields)
1012 ) {
1013 return $errors;
1014 }
1015 }
1016
1017 //validate the pledge fields.
1018 if (!empty($self->_values['pledge_block_id'])) {
1019 //validation for pledge payment.
1020 if (!empty($self->_values['pledge_id'])) {
1021 if (empty($fields['pledge_amount'])) {
1022 $errors['pledge_amount'] = ts('At least one payment option needs to be checked.');
1023 }
1024 }
1025 elseif (!empty($fields['is_pledge'])) {
1026 if (CRM_Utils_Rule::positiveInteger(CRM_Utils_Array::value('pledge_installments', $fields)) == FALSE) {
1027 $errors['pledge_installments'] = ts('Please enter a valid number of pledge installments.');
1028 }
1029 else {
1030 if (CRM_Utils_Array::value('pledge_installments', $fields) == NULL) {
1031 $errors['pledge_installments'] = ts('Pledge Installments is required field.');
1032 }
1033 elseif (CRM_Utils_array::value('pledge_installments', $fields) == 1) {
1034 $errors['pledge_installments'] = ts('Pledges consist of multiple scheduled payments. Select one-time contribution if you want to make your gift in a single payment.');
1035 }
1036 elseif (CRM_Utils_array::value('pledge_installments', $fields) == 0) {
1037 $errors['pledge_installments'] = ts('Pledge Installments field must be > 1.');
1038 }
1039 }
1040
1041 //validation for Pledge Frequency Interval.
1042 if (CRM_Utils_Rule::positiveInteger(CRM_Utils_Array::value('pledge_frequency_interval', $fields)) == FALSE) {
1043 $errors['pledge_frequency_interval'] = ts('Please enter a valid Pledge Frequency Interval.');
1044 }
1045 else {
1046 if (CRM_Utils_Array::value('pledge_frequency_interval', $fields) == NULL) {
1047 $errors['pledge_frequency_interval'] = ts('Pledge Frequency Interval. is required field.');
1048 }
1049 elseif (CRM_Utils_array::value('pledge_frequency_interval', $fields) == 0) {
1050 $errors['pledge_frequency_interval'] = ts('Pledge frequency interval field must be > 0');
1051 }
1052 }
1053 }
1054 }
1055
1056 // also return if paylater mode
1057 if (CRM_Utils_Array::value('payment_processor', $fields) == 0 && $self->_isBillingAddressRequiredForPayLater == 0) {
1058 return empty($errors) ? TRUE : $errors;
1059 }
1060
1061 // if the user has chosen a free membership or the amount is less than zero
1062 // i.e. we skip calling the payment processor and hence dont need credit card
1063 // info
1064 if ((float) $amount <= 0.0) {
1065 return $errors;
1066 }
1067
1068 if (!empty($self->_paymentFields)) {
1069 CRM_Core_Form::validateMandatoryFields($self->_paymentFields, $fields, $errors);
1070 }
1071 CRM_Core_Payment_Form::validateCreditCard($fields, $errors);
1072
1073 foreach (CRM_Contact_BAO_Contact::$_greetingTypes as $greeting) {
1074 if ($greetingType = CRM_Utils_Array::value($greeting, $fields)) {
1075 $customizedValue = CRM_Core_OptionGroup::getValue($greeting, 'Customized', 'name');
1076 if ($customizedValue == $greetingType && empty($fielse[$greeting . '_custom'])) {
1077 $errors[$greeting . '_custom'] = ts('Custom %1 is a required field if %1 is of type Customized.',
1078 array(1 => ucwords(str_replace('_', " ", $greeting)))
1079 );
1080 }
1081 }
1082 }
1083
1084 return empty($errors) ? TRUE : $errors;
1085 }
1086
1087 /**
1088 * @param $params
1089 * @param $form
1090 *
1091 * @return int|mixed|null|string
1092 */
1093 public static function computeAmount(&$params, &$form) {
1094 $amount = NULL;
1095
1096 // first clean up the other amount field if present
1097 if (isset($params['amount_other'])) {
1098 $params['amount_other'] = CRM_Utils_Rule::cleanMoney($params['amount_other']);
1099 }
1100
1101 if (CRM_Utils_Array::value('amount', $params) == 'amount_other_radio' || !empty($params['amount_other'])) {
1102 $amount = $params['amount_other'];
1103 }
1104 elseif (!empty($params['pledge_amount'])) {
1105 $amount = 0;
1106 foreach ($params['pledge_amount'] as $paymentId => $dontCare) {
1107 $amount += CRM_Core_DAO::getFieldValue('CRM_Pledge_DAO_PledgePayment', $paymentId, 'scheduled_amount');
1108 }
1109 }
1110 else {
1111 if (!empty($form->_values['amount'])) {
1112 $amountID = CRM_Utils_Array::value('amount', $params);
1113
1114 if ($amountID) {
1115 $params['amount_level'] = CRM_Utils_Array::value('label', $form->_values[$amountID]);
1116 $amount = CRM_Utils_Array::value('value', $form->_values[$amountID]);
1117 }
1118 }
1119 }
1120 return $amount;
1121 }
1122
1123 /**
1124 * Function to process the form
1125 *
1126 * @access public
1127 *
1128 * @return void
1129 */
1130 public function postProcess() {
1131 $config = CRM_Core_Config::singleton();
1132 // we first reset the confirm page so it accepts new values
1133 $this->controller->resetPage('Confirm');
1134
1135 // get the submitted form values.
1136 $params = $this->controller->exportValues($this->_name);
1137
1138 //carry campaign from profile.
1139 if (array_key_exists('contribution_campaign_id', $params)) {
1140 $params['campaign_id'] = $params['contribution_campaign_id'];
1141 }
1142
1143 if (!empty($params['onbehalfof_id'])) {
1144 $params['organization_id'] = $params['onbehalfof_id'];
1145 }
1146
1147 $params['currencyID'] = $config->defaultCurrency;
1148
1149 if (!empty($params['priceSetId'])) {
1150 $is_quick_config = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $this->_priceSetId, 'is_quick_config');
1151 if ($is_quick_config) {
1152 $priceField = new CRM_Price_DAO_PriceField();
1153 $priceField->price_set_id = $params['priceSetId'];
1154 $priceField->orderBy('weight');
1155 $priceField->find();
1156
1157 $priceOptions = array();
1158 while ($priceField->fetch()) {
1159 CRM_Price_BAO_PriceFieldValue::getValues($priceField->id, $priceOptions);
1160 if ($selectedPriceOptionID = CRM_Utils_Array::value("price_{$priceField->id}", $params)) {
1161 switch ($priceField->name) {
1162 case 'membership_amount':
1163 $this->_params['selectMembership'] = $params['selectMembership'] = CRM_Utils_Array::value('membership_type_id', $priceOptions[$selectedPriceOptionID]);
1164 $this->set('selectMembership', $params['selectMembership']);
1165 if (CRM_Utils_Array::value('is_separate_payment', $this->_membershipBlock) == 0) {
1166 $this->_values['amount'] = CRM_Utils_Array::value('amount', $priceOptions[$selectedPriceOptionID]);
1167 }
1168 break;
1169
1170 case 'contribution_amount':
1171 $params['amount'] = $selectedPriceOptionID;
1172 $this->_values['amount'] = CRM_Utils_Array::value('amount', $priceOptions[$selectedPriceOptionID]);
1173 $this->_values[$selectedPriceOptionID]['value'] = CRM_Utils_Array::value('amount', $priceOptions[$selectedPriceOptionID]);
1174 $this->_values[$selectedPriceOptionID]['label'] = CRM_Utils_Array::value('label', $priceOptions[$selectedPriceOptionID]);
1175 $this->_values[$selectedPriceOptionID]['amount_id'] = CRM_Utils_Array::value('id', $priceOptions[$selectedPriceOptionID]);
1176 $this->_values[$selectedPriceOptionID]['weight'] = CRM_Utils_Array::value('weight', $priceOptions[$selectedPriceOptionID]);
1177 break;
1178
1179 case 'other_amount':
1180 $params['amount_other'] = $selectedPriceOptionID;
1181 break;
1182 }
1183 }
1184 }
1185 }
1186 }
1187
1188 if (($this->_values['is_pay_later'] &&
1189 empty($this->_paymentProcessor) &&
1190 !array_key_exists('hidden_processor', $params)) ||
1191 (!empty($params['payment_processor']) && $params['payment_processor'] == 0)) {
1192 $params['is_pay_later'] = 1;
1193 }
1194 else {
1195 $params['is_pay_later'] = 0;
1196 }
1197
1198 $this->set('is_pay_later', $params['is_pay_later']);
1199 // assign pay later stuff
1200 $this->_params['is_pay_later'] = CRM_Utils_Array::value('is_pay_later', $params, FALSE);
1201 $this->assign('is_pay_later', $params['is_pay_later']);
1202 if ($params['is_pay_later']) {
1203 $this->assign('pay_later_text', $this->_values['pay_later_text']);
1204 $this->assign('pay_later_receipt', $this->_values['pay_later_receipt']);
1205 }
1206
1207 // from here on down, $params['amount'] holds a monetary value (or null) rather than an option ID
1208 $params['amount'] = self::computeAmount($params, $this);
1209 $params['separate_amount'] = $params['amount'];
1210 $memFee = NULL;
1211 if (!empty($params['selectMembership'])) {
1212 if (!empty($this->_membershipTypeValues)) {
1213 $membershipTypeValues = $this->_membershipTypeValues[$params['selectMembership']];
1214 }
1215 else {
1216 $membershipTypeValues = CRM_Member_BAO_Membership::buildMembershipTypeValues($this,
1217 $params['selectMembership']
1218 );
1219 }
1220 $memFee = $membershipTypeValues['minimum_fee'];
1221 if (!$params['amount'] && !$this->_separateMembershipPayment) {
1222 $params['amount'] = $memFee ? $memFee : 0;
1223 }
1224 }
1225
1226 //If the membership & contribution is used in contribution page & not separate payment
1227 $fieldId = $memPresent = $membershipLabel = $fieldOption = $is_quick_config = NULL;
1228 $proceFieldAmount = 0;
1229 if (property_exists($this, '_separateMembershipPayment') && $this->_separateMembershipPayment == 0) {
1230 $is_quick_config = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $this->_priceSetId, 'is_quick_config');
1231 if ($is_quick_config) {
1232 foreach ($this->_priceSet['fields'] as $fieldKey => $fieldVal) {
1233 if ($fieldVal['name'] == 'membership_amount' && !empty($params['price_' . $fieldKey ])) {
1234 $fieldId = $fieldVal['id'];
1235 $fieldOption = $params['price_' . $fieldId];
1236 $proceFieldAmount += $fieldVal['options'][$this->_submitValues['price_' . $fieldId]]['amount'];
1237 $memPresent = TRUE;
1238 }
1239 else {
1240 if (!empty($params['price_' . $fieldKey]) && $memPresent && ($fieldVal['name'] == 'other_amount' || $fieldVal['name'] == 'contribution_amount')) {
1241 $fieldId = $fieldVal['id'];
1242 if ($fieldVal['name'] == 'other_amount') {
1243 $proceFieldAmount += $this->_submitValues['price_' . $fieldId];
1244 }
1245 elseif ($fieldVal['name'] == 'contribution_amount' && $this->_submitValues['price_' . $fieldId] > 0) {
1246 $proceFieldAmount += $fieldVal['options'][$this->_submitValues['price_' . $fieldId]]['amount'];
1247 }
1248 unset($params['price_' . $fieldId]);
1249 break;
1250 }
1251 }
1252 }
1253 }
1254 }
1255
1256 if (!isset($params['amount_other'])) {
1257 $this->set('amount_level', CRM_Utils_Array::value('amount_level', $params));
1258 }
1259
1260 if ($priceSetId = CRM_Utils_Array::value('priceSetId', $params)) {
1261 $lineItem = array();
1262 $is_quick_config = CRM_Core_DAO::getFieldValue('CRM_Price_DAO_PriceSet', $priceSetId, 'is_quick_config' );
1263 if ( $is_quick_config ) {
1264 foreach ( $this->_values['fee'] as $key => & $val ) {
1265 if ( $val['name'] == 'other_amount' && $val['html_type'] == 'Text' && array_key_exists( 'price_'.$key, $params ) && $params['price_'.$key] != 0 ) {
1266 foreach ( $val['options'] as $optionKey => & $options ) {
1267 $options['amount'] = CRM_Utils_Array::value( 'price_'.$key, $params );
1268 break;
1269 }
1270 $params['price_'.$key] = 1;
1271 break;
1272 }
1273 }
1274 }
1275
1276 $component = '';
1277 if ($this->_membershipBlock) {
1278 $component = 'membership';
1279 }
1280 CRM_Price_BAO_PriceSet::processAmount($this->_values['fee'], $params, $lineItem[$priceSetId], $component);
1281 if ($params['tax_amount']) {
1282 $this->set('tax_amount', $params['tax_amount']);
1283 }
1284
1285 if ($proceFieldAmount) {
1286 $lineItem[$params['priceSetId']][$fieldOption]['unit_price'] = $proceFieldAmount;
1287 $lineItem[$params['priceSetId']][$fieldOption]['line_total'] = $proceFieldAmount;
1288 if (isset($lineItem[$params['priceSetId']][$fieldOption]['tax_amount'])) {
1289 $proceFieldAmount += $lineItem[$params['priceSetId']][$fieldOption]['tax_amount'];
1290 }
1291 if (!$this->_membershipBlock['is_separate_payment']) {
1292 $params['amount'] = $proceFieldAmount; //require when separate membership not used
1293 }
1294 }
1295 $this->set('lineItem', $lineItem);
1296 }
1297
1298 if ($this->_membershipBlock['is_separate_payment'] && !empty($params['separate_amount'])) {
1299 $this->set('amount', $params['separate_amount']);
1300 } else {
1301 $this->set('amount', $params['amount']);
1302 }
1303
1304 // generate and set an invoiceID for this transaction
1305 $invoiceID = md5(uniqid(rand(), TRUE));
1306 $this->set('invoiceID', $invoiceID);
1307
1308 // required only if is_monetary and valid positive amount
1309 if ($this->_values['is_monetary'] &&
1310 is_array($this->_paymentProcessor) &&
1311 ((float ) $params['amount'] > 0.0 || $memFee > 0.0)
1312 ) {
1313
1314 // default mode is direct
1315 $this->set('contributeMode', 'direct');
1316
1317 if ($this->_paymentProcessor &&
1318 $this->_paymentProcessor['billing_mode'] & CRM_Core_Payment::BILLING_MODE_BUTTON
1319 ) {
1320 //get the button name
1321 $buttonName = $this->controller->getButtonName();
1322 if (in_array($buttonName,
1323 array($this->_expressButtonName, $this->_expressButtonName . '_x', $this->_expressButtonName . '_y')
1324 ) && empty($params['is_pay_later'])) {
1325 $this->set('contributeMode', 'express');
1326
1327 $donateURL = CRM_Utils_System::url('civicrm/contribute', '_qf_Contribute_display=1');
1328 $params['cancelURL'] = CRM_Utils_System::url('civicrm/contribute/transact', "_qf_Main_display=1&qfKey={$params['qfKey']}", TRUE, NULL, FALSE);
1329 $params['returnURL'] = CRM_Utils_System::url('civicrm/contribute/transact', "_qf_Confirm_display=1&rfp=1&qfKey={$params['qfKey']}", TRUE, NULL, FALSE);
1330 $params['invoiceID'] = $invoiceID;
1331
1332 //default action is Sale
1333 $params['payment_action'] = 'Sale';
1334
1335 $payment = CRM_Core_Payment::singleton($this->_mode, $this->_paymentProcessor, $this);
1336 $token = $payment->setExpressCheckout($params);
1337 if (is_a($token, 'CRM_Core_Error')) {
1338 CRM_Core_Error::displaySessionError($token);
1339 CRM_Utils_System::redirect($params['cancelURL']);
1340 }
1341
1342 $this->set('token', $token);
1343
1344 $paymentURL = $this->_paymentProcessor['url_site'] . "/cgi-bin/webscr?cmd=_express-checkout&token=$token";
1345 CRM_Utils_System::redirect($paymentURL);
1346 }
1347 }
1348 elseif ($this->_paymentProcessor &&
1349 $this->_paymentProcessor['billing_mode'] & CRM_Core_Payment::BILLING_MODE_NOTIFY
1350 ) {
1351 $this->set('contributeMode', 'notify');
1352 }
1353 }
1354
1355 // should we skip the confirm page?
1356 if (empty($this->_values['is_confirm_enabled'])) {
1357 // call the post process hook for the main page before we switch to confirm
1358 $this->postProcessHook();
1359
1360 // build the confirm page
1361 $confirmForm = &$this->controller->_pages['Confirm'];
1362 $confirmForm->preProcess();
1363 $confirmForm->buildQuickForm();
1364
1365 // the confirmation page is valid
1366 $data = &$this->controller->container();
1367 $data['valid']['Confirm'] = 1;
1368
1369 // confirm the contribution
1370 // mainProcess calls the hook also
1371 $confirmForm->mainProcess();
1372 $qfKey = $this->controller->_key;
1373
1374 // redirect to thank you page
1375 CRM_Utils_System::redirect(CRM_Utils_System::url('civicrm/contribute/transact', "_qf_ThankYou_display=1&qfKey=$qfKey", TRUE, NULL, FALSE));
1376 }
1377 }
1378
1379 /**
1380 * Handle Payment Processor switching
1381 * For contribution and event registration forms
1382 */
1383 static function preProcessPaymentOptions(&$form, $noFees = FALSE) {
1384 $form->_snippet = CRM_Utils_Array::value('snippet', $_GET);
1385
1386 $form->_paymentProcessors = $noFees ? array() : $form->get('paymentProcessors');
1387 $form->_ppType = NULL;
1388 if ($form->_paymentProcessors) {
1389 // Fetch type during ajax request
1390 if (isset($_GET['type']) && $form->_snippet) {
1391 $form->_ppType = $_GET['type'];
1392 }
1393 // Remember type during form post
1394 elseif (!empty($form->_submitValues)) {
1395 $form->_ppType = CRM_Utils_Array::value('payment_processor', $form->_submitValues);
1396 $form->_paymentProcessor = CRM_Utils_Array::value($form->_ppType, $form->_paymentProcessors);
1397 $form->set('type', $form->_ppType);
1398 $form->set('mode', $form->_mode);
1399 $form->set('paymentProcessor', $form->_paymentProcessor);
1400 }
1401 // Set default payment processor
1402 else {
1403 foreach ($form->_paymentProcessors as $values) {
1404 if (!empty($values['is_default']) || count($form->_paymentProcessors) == 1) {
1405 $form->_ppType = $values['id'];
1406 break;
1407 }
1408 }
1409 }
1410 if ($form->_ppType) {
1411 CRM_Core_Payment_ProcessorForm::preProcess($form);
1412 }
1413
1414 //get payPal express id and make it available to template
1415 foreach ($form->_paymentProcessors as $ppId => $values) {
1416 $payPalExpressId = ($values['payment_processor_type'] == 'PayPal_Express') ? $values['id'] : 0;
1417 $form->assign('payPalExpressId', $payPalExpressId);
1418 if ($payPalExpressId) {
1419 break;
1420 }
1421 }
1422 if (!$form->_snippet) {
1423 // Add JS to show icons for the accepted credit cards
1424 $creditCardTypes = CRM_Core_Payment_Form::getCreditCardCSSNames();
1425 CRM_Core_Resources::singleton()
1426 ->addScriptFile('civicrm', 'templates/CRM/Core/BillingBlock.js', 10)
1427 // workaround for CRM-13634
1428 // ->addSetting(array('config' => array('creditCardTypes' => $creditCardTypes)));
1429 ->addScript('CRM.config.creditCardTypes = ' . json_encode($creditCardTypes) . ';');
1430 }
1431 }
1432 $form->assign('ppType', $form->_ppType);
1433 }
1434 }
1435