[WIP] TwingleDonation.Submit API action for SEPA mandates.

This commit is contained in:
Jens Schuppe 2018-10-09 15:05:44 +02:00
parent 3856ef6170
commit 42c40e8c00
5 changed files with 285 additions and 113 deletions

View file

@ -129,6 +129,28 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
TRUE // is required TRUE // is required
); );
$this->add(
'select',
'gender_male',
E::ts('Gender option for submitted value "male"'),
$this->getGenderOptions(),
TRUE
);
$this->add(
'select',
'gender_female',
E::ts('Gender option for submitted value "female"'),
$this->getGenderOptions(),
TRUE
);
$this->add(
'select',
'gender_other',
E::ts('Gender option for submitted value "other"'),
$this->getGenderOptions(),
TRUE
);
$payment_instruments = CRM_Twingle_Profile::paymentInstruments(); $payment_instruments = CRM_Twingle_Profile::paymentInstruments();
$this->assign('payment_instruments', $payment_instruments); $this->assign('payment_instruments', $payment_instruments);
foreach ($payment_instruments as $pi_name => $pi_label) { foreach ($payment_instruments as $pi_name => $pi_label) {
@ -141,6 +163,16 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
); );
} }
if (CRM_Twingle_Submission::civiSepaEnabled()) {
$this->add(
'select',
'sepa_creditor_id',
E::ts('CiviSEPA creditor'),
$this->getSepaCreditors(),
TRUE
);
}
$this->add( $this->add(
'select', // field type 'select', // field type
'newsletter_groups', // field name 'newsletter_groups', // field name
@ -282,17 +314,42 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
* Retrieves campaigns present within the system as options for select form * Retrieves campaigns present within the system as options for select form
* elements. * elements.
*/ */
public function getCampaigns() { public function getGenderOptions() {
$campaigns = array('' => E::ts("no campaign")); $genders = array();
$query = civicrm_api3('Campaign', 'get', array( $query = civicrm_api3('OptionValue', 'get', array(
'option_group_id' => 'gender',
'is_active' => 1, 'is_active' => 1,
'option.limit' => 0, 'option.limit' => 0,
'return' => 'id,title' 'return' => array(
'value',
'label',
),
)); ));
foreach ($query['values'] as $campaign) { foreach ($query['values'] as $gender) {
$campaigns[$campaign['id']] = $campaign['title']; $genders[$gender['value']] = $gender['label'];
} }
return $campaigns; return $genders;
}
/**
* Retrieves CiviSEPA creditors as options for select form elements.
*
* @return array
* @throws \CiviCRM_API3_Exception
*/
public function getSepaCreditors() {
$creditors = array();
if (CRM_Twingle_Submission::civiSepaEnabled()) {
$result = civicrm_api3('SepaCreditor', 'get', array(
'option.limit' => 0,
));
foreach ($result['values'] as $sepa_creditor) {
$creditors[$sepa_creditor['id']] = $sepa_creditor['name'];
}
}
return $creditors;
} }
/** /**
@ -309,7 +366,21 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
'return' => 'value,label' 'return' => 'value,label'
)); ));
foreach ($query['values'] as $payment_instrument) { foreach ($query['values'] as $payment_instrument) {
self::$_paymentInstruments[$payment_instrument['value']] = $payment_instrument['label']; // Do not include CiviSEPA payment instruments, but add a SEPA option if
// enabled.
if (
CRM_Twingle_Submission::civiSepaEnabled()
&& CRM_Sepa_Logic_Settings::isSDD(array(
'payment_instrument_id' => $payment_instrument['value'],
))
) {
if (!isset(self::$_paymentInstruments['sepa'])) {
self::$_paymentInstruments['sepa'] = E::ts('CiviSEPA');
}
}
else {
self::$_paymentInstruments[$payment_instrument['value']] = $payment_instrument['label'];
}
} }
} }
return self::$_paymentInstruments; return self::$_paymentInstruments;
@ -338,7 +409,23 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
} }
} }
else { else {
$groups[''] = E::ts('No newsletter groups available'); $groups[''] = E::ts('No mailing lists available');
}
return $groups;
}
/**
* Retrieves active groups as options for select form elements.
*/
public function getGroups() {
$groups = array();
$query = civicrm_api3('Group', 'get', array(
'is_active' => 1,
'option.limit' => 0,
'return' => 'id,name'
));
foreach ($query['values'] as $group) {
$groups[$group['id']] = $group['name'];
} }
return $groups; return $groups;
} }
@ -348,27 +435,7 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
* options for select form elements. * options for select form elements.
*/ */
public function getPostinfoGroups() { public function getPostinfoGroups() {
$groups = array(); return $this->getGroups();
$group_types = civicrm_api3('OptionValue', 'get', array(
'option_group_id' => 'group_type',
'name' => CRM_Twingle_Submission::GROUP_TYPE_POSTINFO,
));
if ($group_types['count'] > 0) {
$group_type = reset($group_types['values']);
$query = civicrm_api3('Group', 'get', array(
'is_active' => 1,
'group_type' => array('LIKE' => '%' . CRM_Utils_Array::implodePadded($group_type['value']) . '%'),
'option.limit' => 0,
'return' => 'id,name'
));
foreach ($query['values'] as $group) {
$groups[$group['id']] = $group['name'];
}
}
else {
$groups[''] = E::ts('No postal mailing groups available');
}
return $groups;
} }
/** /**
@ -376,27 +443,7 @@ class CRM_Twingle_Form_Profile extends CRM_Core_Form {
* system as options for select form elements. * system as options for select form elements.
*/ */
public function getDonationReceiptGroups() { public function getDonationReceiptGroups() {
$groups = array(); return $this->getGroups();
$group_types = civicrm_api3('OptionValue', 'get', array(
'option_group_id' => 'group_type',
'name' => CRM_Twingle_Submission::GROUP_TYPE_DONATION_RECEIPT,
));
if ($group_types['count'] > 0) {
$group_type = reset($group_types['values']);
$query = civicrm_api3('Group', 'get', array(
'is_active' => 1,
'group_type' => array('LIKE' => '%' . CRM_Utils_Array::implodePadded($group_type['value']) . '%'),
'option.limit' => 0,
'return' => 'id,name'
));
foreach ($query['values'] as $group) {
$groups[$group['id']] = $group['name'];
}
}
else {
$groups[''] = E::ts('No donation receipt groups available');
}
return $groups;
} }
} }

View file

@ -180,6 +180,10 @@ class CRM_Twingle_Profile {
'pi_paydirekt', 'pi_paydirekt',
'pi_applepay', 'pi_applepay',
'pi_googlepay', 'pi_googlepay',
'sepa_creditor_id',
'gender_male',
'gender_female',
'gender_other',
'newsletter_groups', 'newsletter_groups',
'postinfo_groups', 'postinfo_groups',
'donation_receipt_groups' 'donation_receipt_groups'
@ -221,19 +225,23 @@ class CRM_Twingle_Profile {
'location_type_id' => CRM_Twingle_Submission::LOCATION_TYPE_ID_WORK, 'location_type_id' => CRM_Twingle_Submission::LOCATION_TYPE_ID_WORK,
'financial_type_id' => 1, // "Donation" 'financial_type_id' => 1, // "Donation"
'pi_banktransfer' => 5, // "EFT" 'pi_banktransfer' => 5, // "EFT"
'pi_debit_manual' => '', // TODO: SEPA 'pi_debit_manual' => NULL,
'pi_debit_automatic' => 3, // Debit 'pi_debit_automatic' => 3, // Debit
'pi_creditcard' => 1, // "Credit Card" 'pi_creditcard' => 1, // "Credit Card"
'pi_mobilephone_germany' => '', 'pi_mobilephone_germany' => NULL,
'pi_paypal' => '', 'pi_paypal' => NULL,
'pi_sofortueberweisung' => '', 'pi_sofortueberweisung' => NULL,
'pi_amazonpay' => '', 'pi_amazonpay' => NULL,
'pi_paydirekt' => '', 'pi_paydirekt' => NULL,
'pi_applepay' => '', 'pi_applepay' => NULL,
'pi_googlepay' => '', 'pi_googlepay' => NULL,
'newsletter_groups' => '', 'sepa_creditor_id' => NULL,
'postinfo_groups' => '', 'gender_male' => 2,
'donation_receipt_groups' => '', 'gender_female' => 1,
'gender_other' => 3,
'newsletter_groups' => NULL,
'postinfo_groups' => NULL,
'donation_receipt_groups' => NULL,
)); ));
} }

View file

@ -27,16 +27,6 @@ class CRM_Twingle_Submission {
*/ */
const GROUP_TYPE_NEWSLETTER = 'Mailing List'; const GROUP_TYPE_NEWSLETTER = 'Mailing List';
/**
* The option value name of the group type for postal mailing subscribers.
*/
const GROUP_TYPE_POSTINFO = ''; // TODO.
/**
* The option value name of the group type for donation receipt requesters.
*/
const GROUP_TYPE_DONATION_RECEIPT = ''; // TODO.
/** /**
* The default ID of the "Employer of" relationship type. * The default ID of the "Employer of" relationship type.
*/ */
@ -100,7 +90,7 @@ class CRM_Twingle_Submission {
// Get the gender ID defined within the profile, or return an error if none // Get the gender ID defined within the profile, or return an error if none
// matches (i.e. an unknown gender was submitted). // matches (i.e. an unknown gender was submitted).
if (!$gender_id = $profile->getAttribute('gender_' . $params['user_gender'])) { if (!empty($params['user_gender']) && !$gender_id = $profile->getAttribute('gender_' . $params['user_gender'])) {
throw new CiviCRM_API3_Exception( throw new CiviCRM_API3_Exception(
E::ts('Gender could not be matched to existing gender.'), E::ts('Gender could not be matched to existing gender.'),
'invalid_format' 'invalid_format'
@ -249,4 +239,60 @@ class CRM_Twingle_Submission {
} }
} }
/**
* Check whether the CiviSEPA extension is installed and CiviSEPA
* functionality is activated within the Twingle extension settings.
*
* @return bool
* @throws \CiviCRM_API3_Exception
*/
public static function civiSepaEnabled() {
$sepa_extension = civicrm_api3('Extension', 'get', array(
'full_name' => 'org.project60.sepa',
'is_active' => 1,
));
return
CRM_Core_BAO_Setting::getItem(
'de.systopia.twingle',
'twingle_use_sepa'
)
&& $sepa_extension['count'];
}
/**
* Retrieves recurring contribution frequency attributes for a given donation
* rhythm parameter value, according to a static mapping.
*
* @param string $donation_rhythm
* The submitted "donation_rhythm" paramter according to the API action
* specification.
*
* @return array
* An array with "frequency_unit" and "frequency_interval" keys, to be added
* to contribution parameter arrays.
*/
public static function getFrequencyMapping($donation_rhythm) {
$mapping = array(
'halfyearly' => array(
'frequency_unit' => 'month',
'frequency_interval' => 6,
),
'quarterly' => array(
'frequency_unit' => 'month',
'frequency_interval' => 3,
),
'yearly' => array(
'frequency_unit' => 'year',
'frequency_interval' => 1,
),
'monthly' => array(
'frequency_unit' => 'month',
'frequency_interval' => 1,
),
'one_time' => array(),
);
return $mapping[$donation_rhythm];
}
} }

View file

@ -251,7 +251,10 @@ function civicrm_api3_twingle_donation_Submit($params) {
$existing_contribution = civicrm_api3('Contribution', 'get', array( $existing_contribution = civicrm_api3('Contribution', 'get', array(
'trxn_id' => $params['trx_id'] 'trxn_id' => $params['trx_id']
)); ));
if ($existing_contribution['count'] > 0) { $existing_contribution_recur = civicrm_api3('ContributionRecur', 'get', array(
'trxn_id' => $params['trx_id']
));
if ($existing_contribution['count'] > 0 || $existing_contribution_recur['count'] > 0) {
throw new CiviCRM_API3_Exception( throw new CiviCRM_API3_Exception(
E::ts('Contribution with the given transaction ID already exists.'), E::ts('Contribution with the given transaction ID already exists.'),
'api_error' 'api_error'
@ -298,19 +301,17 @@ function civicrm_api3_twingle_donation_Submit($params) {
// Exclude address for now when retrieving/creating the individual contact // Exclude address for now when retrieving/creating the individual contact
// and an organisation is given, as we are checking organisation address // and an organisation is given, as we are checking organisation address
// first and share it with the individual. // first and share it with the individual.
if (!empty($params['organization_name'])) { $submitted_address = array();
$submitted_address = array(); foreach (array(
foreach (array( 'street_address',
'street_address', 'postal_code',
'postal_code', 'city',
'city', 'country',
'country', 'location_type_id',
'location_type_id', ) as $address_component) {
) as $address_component) { if (!empty($params[$address_component])) {
if (!empty($params[$address_component])) { $submitted_address[$address_component] = $params[$address_component];
$submitted_address[$address_component] = $params[$address_component]; unset($params[$address_component]);
unset($params[$address_component]);
}
} }
} }
@ -324,6 +325,7 @@ function civicrm_api3_twingle_donation_Submit($params) {
'user_birthdate' => 'birth_date', 'user_birthdate' => 'birth_date',
'user_email' => 'email', 'user_email' => 'email',
'user_telephone' => 'phone', 'user_telephone' => 'phone',
'user_title' => 'formal_title',
) as $contact_param => $contact_component) { ) as $contact_param => $contact_component) {
if (!empty($params[$contact_param])) { if (!empty($params[$contact_param])) {
$contact_data[$contact_component] = $params[$contact_param]; $contact_data[$contact_component] = $params[$contact_param];
@ -354,7 +356,9 @@ function civicrm_api3_twingle_donation_Submit($params) {
'organization_name' => $params['organization_name'], 'organization_name' => $params['organization_name'],
); );
if (!empty($submitted_address)) { if (!empty($submitted_address)) {
$params += $submitted_address; $organisation_data += $submitted_address;
// Always use WORK address for organisation address.
$organisation_data['location_type_id'] = CRM_Twingle_Submission::LOCATION_TYPE_ID_WORK;
} }
if (!$organisation_id = CRM_Twingle_Submission::getContact( if (!$organisation_id = CRM_Twingle_Submission::getContact(
'Organization', 'Organization',
@ -366,16 +370,18 @@ function civicrm_api3_twingle_donation_Submit($params) {
); );
} }
} }
// Share organisation address as WORK address with individual contact.
$address_shared = ( $address_shared = (
isset($organisation_id) isset($organisation_id)
&& CRM_Twingle_Submission::shareWorkAddress( && CRM_Twingle_Submission::shareWorkAddress(
$contact_id, $contact_id,
$organisation_id, $organisation_id,
$params['location_type_id'] CRM_Twingle_Submission::LOCATION_TYPE_ID_WORK
) )
); );
// Address is not shared, use submitted address. // Address is not shared, use submitted address with configured location
// type.
if (!$address_shared && !empty($submitted_address)) { if (!$address_shared && !empty($submitted_address)) {
$submitted_address['contact_id'] = $contact_id; $submitted_address['contact_id'] = $contact_id;
civicrm_api3('Address', 'create', $submitted_address); civicrm_api3('Address', 'create', $submitted_address);
@ -387,45 +393,88 @@ function civicrm_api3_twingle_donation_Submit($params) {
} }
} }
// TODO: contact into newsletter, postinfo and donation_receipt groups. // If requested, add contact to newsletter groups defined in the profile.
if (!empty($params['newsletter']) && !empty($groups = $profile->getAttribute('newsletter_groups'))) {
foreach ($groups as $group_id) {
civicrm_api3('GroupContact', 'create', array(
'group_id' => $group_id,
'contact_id' => $contact_id,
));
}
}
// Create contribution or SEPA mandate. // If requested, add contact to postinfo groups defined in the profile.
if (!empty($params['postinfo']) && !empty($groups = $profile->getAttribute('postinfo_groups'))) {
foreach ($groups as $group_id) {
civicrm_api3('GroupContact', 'create', array(
'group_id' => $group_id,
'contact_id' => $contact_id,
));
}
}
// If requested, add contact to donation_receipt groups defined in the
// profile.
if (!empty($params['donation_receipt']) && !empty($groups = $profile->getAttribute('donation_receipt_groups'))) {
foreach ($groups as $group_id) {
civicrm_api3('GroupContact', 'create', array(
'group_id' => $group_id,
'contact_id' => $contact_id,
));
}
}
// Create contribution or SEPA mandate. Those attributes are valid for both,
// single and recurring contributions.
$contribution_data = array( $contribution_data = array(
'contact_id' => (isset($organisation_id) ? $organisation_id : $contact_id), 'contact_id' => (isset($organisation_id) ? $organisation_id : $contact_id),
'currency' => $params['currency'], 'currency' => $params['currency'],
'trxn_id' => $params['trx_id'], 'trxn_id' => $params['trx_id'],
'financial_type_id' => $profile->getAttribute('financial_type_id'), 'financial_type_id' => $profile->getAttribute('financial_type_id'),
'payment_instrument_id' => $params['payment_instrument_id'], 'payment_instrument_id' => $params['payment_instrument_id'],
'amount' => $params['amount'], 'amount' => $params['amount'] / 100,
'total_amount' => $params['amount'], 'total_amount' => $params['amount'] / 100,
); );
if (!empty($params['purpose'])) { if (!empty($params['purpose'])) {
$contribution_data['note'] = $params['purpose']; $contribution_data['note'] = $params['purpose'];
} }
$sepa_extension = civicrm_api3('Extension', 'get', array(
'full_name' => 'org.project60.sepa',
'is_active' => 1,
));
if ( if (
CRM_Core_BAO_Setting::getItem( CRM_Twingle_Submission::civiSepaEnabled()
'de.systopia.twingle', && $contribution_data['payment_instrument_id'] == 'sepa'
'twingle_use_sepa'
)
&& $sepa_extension['count']
&& CRM_Sepa_Logic_Settings::isSDD($contribution_data)
) { ) {
// If CiviSEPA is installed and the financial type is a CiviSEPA-one, // If CiviSEPA is installed and the financial type is a CiviSEPA-one,
// create SEPA mandate (and recurring contribution, using "createfull" API // create SEPA mandate (and recurring contribution, using "createfull" API
// action). // action).
$mandate_data = $contribution_data + array( foreach (array(
'debit_iban',
'debit_bic',
) as $sepa_attribute) {
if (empty($params[$sepa_attribute])) {
throw new CiviCRM_API3_Exception(
E::ts('Missing attribute %1 for SEPA mandate', array(
1 => $sepa_attribute,
)),
'invalid_format'
);
}
}
$mandate_data =
$contribution_data
// Add CiviSEPA mandate attributes.
+ array(
'type' => ($params['donation_rhythm'] == 'one_time' ? 'OOFF' : 'RCUR'), 'type' => ($params['donation_rhythm'] == 'one_time' ? 'OOFF' : 'RCUR'),
'iban' => $params['debit_iban'], 'iban' => $params['debit_iban'],
'bic' => $params['debit_bic'], 'bic' => $params['debit_bic'],
'reference' => $params['debit_mandate_reference'], 'reference' => $params['debit_mandate_reference'],
'date' => $params['confirmed_at'], 'date' => $params['confirmed_at'],
'creditor_id' => $profile->getAttribute('sepa_creditor_id'), 'creditor_id' => $profile->getAttribute('sepa_creditor_id'),
); )
// Add frequency unit and interval from static mapping.
+ CRM_Twingle_Submission::getFrequencyMapping($params['donation_rhythm']);
// Let CiviSEPA set the correct payment instrument.
unset($mandate_data['payment_instrument_id']);
$mandate = civicrm_api3('SepaMandate', 'createfull', $mandate_data); $mandate = civicrm_api3('SepaMandate', 'createfull', $mandate_data);
if ($mandate['is_error']) { if ($mandate['is_error']) {
throw new CiviCRM_API3_Exception( throw new CiviCRM_API3_Exception(
@ -434,17 +483,19 @@ function civicrm_api3_twingle_donation_Submit($params) {
); );
} }
$result_values = $mandate; $result_values = $mandate['values'];
} }
else { else {
// Create (recurring) contribution. // Create (recurring) contribution.
if ($params['donation_rhythm'] != 'one_time') { if ($params['donation_rhythm'] != 'one_time') {
// Create recurring contribution first. // Create recurring contribution first.
$contribution_recur_data = $contribution_data + array( $contribution_recur_data =
'frequency_interval' => '', // TODO $contribution_data
+ array(
'contribution_status_id' => 'Pending', // TODO: Or "In Progress"? 'contribution_status_id' => 'Pending', // TODO: Or "In Progress"?
'start_date' => $params['confirmed_at'], 'start_date' => $params['confirmed_at'],
); )
+ CRM_Twingle_Submission::getFrequencyMapping($params['donation_rhythm']);
$contribution_recur = civicrm_api3('contributionRecur', 'create', $contribution_recur_data); $contribution_recur = civicrm_api3('contributionRecur', 'create', $contribution_recur_data);
if ($contribution_recur['is_error']) { if ($contribution_recur['is_error']) {
throw new CiviCRM_API3_Exception( throw new CiviCRM_API3_Exception(
@ -468,7 +519,7 @@ function civicrm_api3_twingle_donation_Submit($params) {
); );
} }
$result_values = $contribution; $result_values = $contribution['values'];
} }
$result = civicrm_api3_create_success($result_values); $result = civicrm_api3_create_success($result_values);

View file

@ -42,6 +42,26 @@
<td class="content">{$form.financial_type_id.html}</td> <td class="content">{$form.financial_type_id.html}</td>
</tr> </tr>
{if isset($form.sepa_creditor_id)}
<tr class="crm-section">
<td class="label">{$form.sepa_creditor_id.label}</td>
<td class="content">{$form.sepa_creditor_id.html}</td>
</tr>
{/if}
<tr class="crm-section">
<td class="label">{$form.gender_male.label}</td>
<td class="content">{$form.gender_male.html}</td>
</tr>
<tr class="crm-section">
<td class="label">{$form.gender_female.label}</td>
<td class="content">{$form.gender_female.html}</td>
</tr>
<tr class="crm-section">
<td class="label">{$form.gender_other.label}</td>
<td class="content">{$form.gender_other.html}</td>
</tr>
</table> </table>
</fieldset> </fieldset>
@ -78,8 +98,8 @@
</tr> </tr>
<tr class="crm-section"> <tr class="crm-section">
<td class="label">{$form.donation_receipts_groups.label}</td> <td class="label">{$form.donation_receipt_groups.label}</td>
<td class="content">{$form.donation_receipts_groups.html}</td> <td class="content">{$form.donation_receipt_groups.html}</td>
</tr> </tr>
</table> </table>