CRM-14538 Partial payments for Memberships

    Details

    • Epic Name:
      Back-office Membership Membership Enhancements
    • Versioning Impact:
      Patch (backwards-compatible bug fixes)
    • Documentation Required?:
      User and Admin Doc
    • Funding Source:
      Contributed Code

      Description

      Description
      Provides back-office staff with the following additional capabilities:

      1. Record partial payment for a Membership Creation. Then record one or more additional payments until the Membership fee is completely paid.

      Part 1:

      Description
      -------------
      Summary
      -------------
      Back-office staff may:

      • Record partial payment for a Membership Creation.
      • Both the contribution and the Membership record will be assigned a status of Partially Paid until the balance owing is zero. This will allow staff to easily locate partially paid registrations.

      ---------------------
      Implementation
      ----------------------
      1. Define new Membership status Rule: 'Partially paid'

      • Status is 'Counted', Class='Positive', Reserved=TRUE, Visibility=Admin, Active=TRUE
      • Add to civicrm_data
      • Insert in upgrades

      2. New Membership Creation (modifications)
      currently (4.4) the Total Amount (Payment Info field set) defaults to total owing AND the user can change that to a smaller amount. However we don't do anything special in that case. We need to make the following changes to postProcessing to handle this condition.

      Client Side form behaviors (jQuery)
      ------------------------------------------------

      • If user enters a Total Amount less than the calculated total fee, set Contribution Status to 'Partially paid' (user MAY override this and change it to something else later)
      • When user clicks "Save", if the Total Amount entered is less than calculated fees AND the contribution status!= 'Partially paid', put up a jScript confirm dialog:
        "Payment amount is less than the amount owed. Expected Contribution status is 'Partially paid'. Are you sure you want to set the contribution status to $statusLabel? Click OK to continue, Cancel to change your entries."

      Post Process changes
      -------------------------------

      • New contribution record is always inserted with the total OWING (calculated fees) as the total_amount. If total_amount = total owing, NO changes in the rest of the postProcess flow.

      If the total_amount on the form is less than the amount owing:

      • Contribution status is set to 'Partially paid'.
      • Financial trxn record is created with total_amount = the amount actually being paid (total_amount from the form). Financial_trxn status is Completed. TO account = the applicable ASSET account (based in payment_instrument)
      • A second financial_trxn record is created for the BALANCE owing. Rules for setting values for this transaction are the same as for a new Pending pay-later transaction:
      • to_financial_account_id = account with "Accounts Receivable Account is" relationship to the financial type of the contribution
      • from_financial_account_id = NULL
      • status = Completed

      Tests
      -------

      • Extend unit tests to cover membership create with partial payment
      • Extend web-tests to cover the above workflow (membership registration with partial payment)

      2 Part:
      Description
      -------------
      Summary
      -------------
      Back-office staff may:

      • Record one or more additional payments against "Partially paid" or "Pending pay later" membership - until the registration fee is completely paid.
      • Members may be sent receipts for each payment
      • Payments will be recorded as financial transactions against the original contribution record.
      • View existing payments (financial transactions)
      • When balance owing is 0, contribution status will be updated to Completed. Membership status can be set in payment form but comply to default rules when balance owing is 0.
      • Record a refund (used when registration selections are changed and fee was overpaid - see CRM-13973).

      ---------------------
      Implementation
      ----------------------
      1. Implement BAO function which calculates 'amount owing' or 'refund due' for a given Membership record.

      • SUM(line_item.line_total) for all line_item rows linked to the membership row MINUS SUM(financial_trxn.total_amount) for all financial_trxn rows linked to the contribution associated with the membership row AND where financial_trxn.status_id = 1 (Completed).
      • Positive number = Amount Owed,
      • Negative number = Refund Due


      2. Using the same form for submitting a Financial Transaction against an existing contribution linked to a membership.

          • IMPORTANT ***: Going forward we are using the same form for applying payments against a Partially Paid membership contribution, so the form class should be structured to support conditional 'integration' with other component transactions.

      3.1 Form elements

      • Form elements are a subset of fields exposed in Contribution form (CRM_Contribute_Form_Contribution), but this is a basic create-only form so we'll make use of new form class which only handles this case. (For Memberships we are using the same form what we have used for Events).
      • For this project, the form will always create a financial_trxn (payment or refund) against a partially paid contribution linked to the passed-in membership record. Contact Name (contact display_name) and Membership title are read-only elements on the top of the form.
      • URL params for contact id and membership id are required / passed to the form.
      • All form fields are columns in financial_trxn table EXCEPT for "Send Receipt" and "Receipt From".
      • add a text area to the form for purpose to provide optional email_text - shown conditionally if [x] Send Receipt checked
      • For 'Refund Due' case:
      • Payment Amount label => Refund Amount
      • Form block title = New Membership Refund
      • Page title = Refund for $displayName
      • Additional client-side form behavior (same as contribution form Additional Details): Net Amount = Payment Amount - Fee Amount

      3.2 Pre process

      • Amount Owed OR Refund Due for this membership should be calculated during preProcess. This value should be the default for the Payment Amount/Refund Due field AND displayed next to the amount input box.
      • If related member (contact) does not have a valid email address, display status re: can not send a receipt and hide the Send Receipt and From fields (same as for contribution form).

      3.3 Form rules

      • If amount owed, payment amount must be < = Amount Owed
      • If refund due, refund amount must = Refund Due (we do not want to support 'partially refunded' membership records at this point).
      • Net Amount = Payment Amount (OR Refund Amount) - Fee Amount

      3.4 Post Process
      NOTE: Use existing Financial Trxn / Financial Item BAO's as much as possible. Do not put transaction process logic in the form class (create new BAO(s) if necessary).

      • Insert new financial_trxn and entity_financial_trxn (keyed to contribution row). Financial Trxn status is completed. Payment instrument, check number, financial type etc are stored per form values.

      ------------------
      For Payments
      ------------------

      • update related contribution status to Completed (from 'Pending refund')
      • update related Membership status to Default Rule (from 'Pending refund')
      • If [x] Send Receipt is checked, we need a new function / message template for sending Payment/Refund Receipts. In the interest of keeping the template simple, The 'Additional Payment Receipt or Refund Notification' message template is ideally intended to be used for both event and membership payments and refunds. Best if you can tweak it / add some variables so that it handles both cases.
      • For Payments – use new activity (type = Payment) for the new payment action. Data pattern similar to Contribution activity:
        Source Contact = logged in user's contact
        Target Contact = payee contact
        Subject = "$paymentAmount - Offline payment for $eventTitle (Title Need to be changed with $membershipType)
      • For Refunds – use new activity (type = Refund) for the new payment action. Data pattern similar to Contribution activity:
        Source Contact = logged in user's contact
        Target Contact = payee contact
        Subject = "$refundAmount - Offline refund for $eventTitle (Title Need to be changed with $membershipType)
      • Cancel on form should return to last form or page that the user was on. Successful submit should return to contact's Memberships tab in most cases. If the form is accessed via Find Memberships, then return to the search form with session key (search results s/b preserved if possible).


      ***Yet to get more information on this Credit Card Block ***
      4. Live Mode (submit credit card payment) - Payments only

      • This mode is available if one or more of the required processor(s) enabled (allowBackofficeCreditCard - same rules as contribution form class). This is NOT available for 'Pending refund' contributions / membership records.
      • Submit button title for this mode is 'Submit Credit Card Payment' (not shown on screenshot)
      • Post process invokes selected processor plugin to submit the payment. Rules for entity updates same as above.
      • Activity subject in LIVE mode = "$paymentAmount - Submit credit card payment for $eventTitle

      5. Modifications to Edit Membership and View Membership forms
      5.1 Selections block: I don't think the intent is to support changing the selected membership type, NOR changing membership-related price set selections. Also even if a membership price set is used we don't need to show the line items in the View or Edit Membership pages, so we may not need to add a Selections section. We do need a new Fees section to show the owed / paid / balance info.

      5.2 If 'Amount Paid' is > 0, add "view payments" link to the Fees table (text link and link on Amount Paid amount value). Clicking link loads a new jQuery dialog pop-up which displays payments already made against the associated contribution (e.g. financial_trxn records). We need to filter the financial_trxn rows to only show actual payments (not transfers to AR account etc.). I think we can do this by only showing transactions where the "TO ACCOUNT" financial_account_type_id is 'Asset' (check with Dave/Pradeep to confirm this).
      **I’m assuming, we can mimic this similar to Events **

      See ViewPaymentsPopup.png screenshot for table elements and layout.


      6. Expose "Record Payment" button / action link for Membership records with "Partially paid" status.

      Users should be able to navigate to 'New Membership Payment' form from the following buttons and action links:

      • Add a 'Record Payment' action link to the Membership selector actions IF Membership status for that row is 'Partially paid' or 'Pending pay later' (this should be available from contact's Membership tab as well as Find Membership / Membership Dashboard). Same behaviour as 'Record Payment' button.

      7. Tests

      Need to check the unit and web tests added for the analogous event/participant issues. Any new or modified BAO's or API's should result in new or extended unit tests. There should be web tests for each of the new UI interactions (i.e. create partially paid membership, make another partial payment, make a completing payment).

      8. Scheduled Reminders

      The use case I would see here is the ability to send reminders to contacts with partially paid memberships (similarly to how we've implemented for event participants). However we don't currently include membership status in the Scheduled Reminders criteria - and we would probably need one or more additional 'events' beyond 'Membership Join Date' and 'Membership End Date' to make this useful (e.g. 'Last Payment Date', 'Membership Start Date'). A simple thing we could do is just add a new token for display of a membership's balance -

      {membership.balance}

      . Then users could create a smart group of partially paid members and thus handle some key use cases.

        Attachments

          Activity

          [CRM-14538] Partial payments for Memberships
          David Greenberg added a comment -

          Pushing this issue to 4.6 because there was no reply to request for status.

          Joe Murray added a comment -

          Dave, could you tell me more about the process and people involved (at least potentially) in this work?

          Jamie Novick and I are thinking of trying to get some client money together to push partial payments forward and would like to coordinate with you. We're identifying scope, which may exclude allocating payment funds to line-items, and include user-facing functionality on top of back-end staff facing functionality.

          What does it mean that this issue is labelled Team_Dave_Yash_Sprint - are you and Yoshoda and the rest of the team thinking of doing a sprint to push this forward?

          I see that this is MIH - what is the amount to be raised and what is the current total raised?

          Thanks for doing the work on https://issues.civicrm.org/jira/browse/CRM-13964. Was that funded by someone who might be interested in funding user-facing functionality for subsequent payments?

          David Greenberg added a comment -

          Joe - This issue was posted by a developer from MillerTech (UK). They had client funding and were supposed to get this done for 4.5 - but the project has been delayed and not sure when / if they are going to continue with it. It is not an MIH. I would suggest contacting Richard Slade or Paul Hunter.

          The event partial payments was funding by the Great Lakes Planetarium Association. I don't think they have funding to extend to front-end additional payments at this time.

          Shai Gluskin added a comment -

          I have a client very interested in partial payments.

          I have some questions about current status.
          1. The title of this issue is "Partial payment for Memberships." Are partial payments for events covered in a different issue? Has any partial-payment functionality been added to 4.5?
          2. Is Alan Shaws and Joe Murray's work as reflected in Allan's blog post (https://civicrm.org/blogs/allenshaw/recording-multiple-payments-one-contribution-and-dividing-one-payment-among-multiple) and Joe's requirement's page (http://wiki.civicrm.org/confluence/display/CRM/CiviAccounts+Specifications-Flexible+User+Payments) guiding this work?

          I see that Joe posted just about a week ago on this issue. Joe, can you report back here on whether you contacted Richard Slade and Paul Hunter and what they said.

          Thanks much.

          David Greenberg added a comment -

          Shai - Backoffice partial payments for events is part of 4.5 thanks to generous sponsorship from Great Lakes Planetarium Association. This has been highlighted in the 4.5 release announcements (e.g. https://civicrm.org/blogs/yashodha/announcing-civicrm-45-beta3-release) - might be good for you to subscribe to the civicrm.org blog rss feed - https://civicrm.org/blog/feed .

          Shai Gluskin added a comment - - edited

          Thanks Dave.

          I do subscribe to the release announcements and I had read that, but maybe a couple of weeks ago. When I went to test it yesterday, I tested it on a contribution, not an event. I guess my mind had mixed the release announcement with the old Allen Shaw blog post. When testing partial payments on a contribution I saw it wasn't there... I started Googling and came to this thread. I somehow missed the work on partial payments for events. Thanks for setting me straight.

          For feedback on the new functionality shall I do that in JIRA at https://issues.civicrm.org/jira/browse/CRM-13964 or in forum.civicrm.org?

          Thanks,

          Shai

          David Greenberg added a comment -

          Feedback to the 4.5 testing forum please. However we are substantially over the sponsored hours already on this project so only actual bugs are likely to be addressed for 4.5 (rather than 'improvements').

          Ramesh added a comment - - edited

          Hi all, Ramesh here from MillerTech

          I've recently been working on the partial payments for membership functionality for version 4.7.x and have assigned this issue to myself.

          This development has been designed based on the similar format as the core event partial payment functionality so this route is very different from what was originally anticipated.

          Description -

          Provides back-office staff with the following functionality on the membership screen.

          Record partial payment on membership creation by overwriting the original amount in the “Amount” field. For example, select “General” membership, the pre-populated figure in the “Amount” field will be 100.00, overwrite this amount to a lower partially paid value, say 50.00, change payment method if required and leave the contribution status as “Completed” and save.

          This action creates a membership with the “New” status and creates a related contribution with the “Partially Paid” status.

          From the “More” option on the related contribution we have the option to “Record Payment”, this screen is similar to when recording additional payments on the event payment careen. It also displays the outstanding balance and does not allow you to enter a greater amount than the outstanding amount.

          The view membership screen displays Total amount paid, balance and an option to record payment.

          When paid in full the contribution status is set to “Completed”.

          Also works with membership price sets.

          To do –

          1. Send receipt upon partial payment.
          2. Code the membership offline receipt to show balance and amount paid.
          3. Reporting aspect of partial membership payment.

           

          Monish Deb added a comment - - edited

          Hi Ramesh

          I have gone through the spec and it meets our requirement for CRM-20569 where we treated this is a bug as there were incorrect financial entries on partial payment for membership.

          I fixed it by handling the submission of partial payment and added few form validations to ensure that the contribution status to be set to 'Partially paid' if amount is less/equal to expected total amount, also added unit tests with few assertions on related contribution record, if it is updated correctly on partial payment and after paying the additional owed amount.

          The only task I didn't manage to fix it yet, is about handling partial payment for multiple membership and raised an issue CRM-20626, as it is not clear/decided how to distribute the partial amount to multiple memberships. This is the PR https://github.com/civicrm/civicrm-core/pull/10352 . I was wondering if I can help you in completing this task.

          Feel free to ping me here or you can reach me at Mattermost ( alias- @monish )

          Ramesh added a comment - - edited

          Hi @monish

          Thank you very much for taking time to look into the code - Appreciate your help

          regarding multiple membership -  I will try to do that too

          and will submit to code

          thanks

          Ramesh added a comment -

          Hi Monish Deb

          Please have a look at my PR https://github.com/civicrm/civicrm-core/pull/10516/commits I have fixed the issue CRM-20626

          so please have a look and let me know the status

          thanks

          Ramesh

          Monish Deb added a comment - - edited

          Hi Ramesh

          Does your patch cover all the points mentioned in the description? If not, please complete your fix and assign me for QA. You can add unit-test of mine - https://github.com/civicrm/civicrm-core/pull/10352/files#diff-3cd3f80dd69269a09e1d88996ce92f6fR533 to assert this use-case of membership partial payment via backoffice form

          Also, for making changes in PHP we follow the Drupal coding standards as mentioned here https://wiki.civicrm.org/confluence/display/CRMDOC/PHP+Code+and+Inline+Documentation
          Can you please correct the indentations on your patch?

          You can change your IDE settings as per Drupal coding standards and here's the link to do that https://wiki.civicrm.org/confluence/display/CRMDOC/IDE+Settings+to+Meet+Coding+Standards

          Feel free to ping me (alias - @monish) in Mattermost dev channel - https://chat.civicrm.org/civicrm/channels/dev

          Thanks!

          Ramesh added a comment - - edited

          Hi Monish Deb

          The Initial spec was written for Version 4.5 has we have changed a lot in later version - but still this patch does the job of creating a partial payment for membership also handles mulitple membership for priceset too.

          I have corrected the code based on Drupal coding standards

          Also I have added your UnitTest in the file and I have pushed it

          This patch is ready for testing - please let me know the status

          thanks for your help

          Ramesh

          Monish Deb added a comment -

          Hi Ramesh

          Thanks for all your work. But I am afraid I won't be able to review your changes in this week as I got some other issues in priority to wrap up. Will try my best to provide feedback by next week itself.

          Thanks!

          Zachary added a comment -

          Is there any progress on this? I'd love to see this go forward or help out. Our org really needs this.

          Ramesh added a comment -

          Hi Zachary as this is my first contribution to core i am stuggling in getting everything right - I am not getting any help from core team - also i am not sure what to do ?

          if you can help me out that would be great - i can send you all the core files which i have changed regarding this release

          have a look PR https://github.com/civicrm/civicrm-core/pull/10516

          thanks

          Joe Murray added a comment -

          If you save a membership purchase as pending pay later, partial payments can be made against it. This works for both simple quick config and complex price set purchases.

          Zachary added a comment -

          @Ramesh, I am not a coder so I can only help by organizing some fund raising.

          @Joe Murray - does this allow an end user to use the Civicontribute page to submit a partial payment, and then set up a payment plan (manual or recurring), and then allow it to hook into scheduled reminders/Civirules? Or, is your method exclusively back end for staff to enter?

          Joe Murray added a comment -

          No, Zachary, this issue, Ramesh's work, and the approach I outline that is already in core are all just for supporting backoffice partial payments by staff. 

          Ramesh, I think there is a bigger concern that needs to be addressed before this should be allowed to be merged: does it change negatively affect existing workflow in ways that users have come to rely on? More specifically, many organizations adjust the price of memberships to provide discounts by entering a lower price. I think a good way to address this would be to add a Membership price field just under the Membership Organization and Type line, above Number of Terms, and for convenience below the Max related field that pops in when memberships are selected that can have related inherited memberships. This field would determine the price of the membership, while the Amount field in the Membership Payment and Receipt section would be used to set the amount of the payment, either partial or full. The new field should be set to the price of the membership type every time that field changes, and be editable by user.

          I would also put this approach to the dev and partner channels for feedback.

          Ramesh added a comment - - edited

          Joe Murray Right now the issue in merging the code to core is my code is failing in unit testing - as this is my first merge to core I am not sure what is going wrong 

          If any core member can guide me want to do I am happy to do the same 

          Right now the current code changes which I have done will satisfy the above issue spec 

          I am happy to develop the additional changes of changing the membership amount as you specified 

          If the code which I have released got merged with core then - first phase of membership partial payment will be available to all - this is just an suggestion 

          correct me if I am wrong - Also right now I am in India any core member who can help please let me know I can meet you personally at your office and ready to sort out the above code merging issue

          thanks

            People

            • Assignee:
              Ramesh
              Reporter:
              sri

              Dates

              • Created:
                Updated: