Saving $50,000/month through payment transparency · Project

At Springboard, our payment system is built using AngularJS with a GraphQL BFF (Backend-for-Frontend) layer, and it makes extensive use of RxJS and observables to track the state of different components. One day, we started noticing a serious issue – though rare, it had a big impact. Some new customers reported that after making a payment, they weren’t sure if their purchase had gone through. Their money was deducted, but they weren’t navigated to the next step.


This was a huge problem because Springboard courses sometimes cost over US$10,000. Any friction in the checkout flow, especially one that caused doubt about a successful purchase, could immediately break trust. Some users even requested refunds because they weren’t sure whether their payment had been processed.


Investigating the Issue


We did a top-down analysis of the entire user flow during checkout. While we found areas that could be improved, the core problem was a navigation failure. Here’s what was happening:

  1. The user enters their card details and clicks the payment submit button.

  2. The button is then disabled, and a GraphQL charge API call is triggered in the background.

  3. If successful, the API sends a response back to the front end, and the user is supposed to be redirected to their profile page to complete their setup.

  4. If unsuccessful, the error is displayed on the payment page.

For a small number of users, even when the API returned a successful response, they were not navigated to the profile page. Worse, there was no indication that their payment had gone through, leaving them in an uncertain state.


Finding the Root Cause


This was tricky because we couldn’t replicate the issue in our development environment. Without being able to reproduce it, we had no way of directly fixing it. After discussing it with other developers and testing multiple scenarios, we still couldn’t figure out why navigation was failing for these users.

That’s when I decided to shift the focus from fixing the failure itself (which we couldn’t reproduce) to improving transparency in the user experience so that even if the navigation failed, users wouldn’t feel lost.


The Solution: Improving UI Feedback


To make the payment process more transparent and reduce user anxiety, I focused on state management and feedback visibility. Here’s how I improved it:

  1. Better Button States

    • When a user clicks submit, instead of just disabling the button, a spinner loader appears inside it. This visually indicates that the payment is being processed.

    • If the GraphQL API call succeeds, the spinner is replaced with a checkmark.

  2. Clear Success Messaging & Backup Navigation

    • Below the button, a message appears:

      "Payment successful. You will be redirected to your profile page in 3…2…1…"

    • If automatic navigation doesn’t happen, a backup link appears:

      "If you are not redirected, click here to continue."

This ensures that even if navigation fails, the user knows their payment was successful and has a way to proceed.


Button statesButton states

Tracking the Impact with GA4 Analytics


Since we still didn’t know how many users were affected, I set up tracking using GA4 (Google Analytics 4). I measured:

After three months, analytics showed that our fix had saved Springboard around $50,000/mo by preventing refunds due to payment uncertainty. The number of users who had to click the backup navigation link was small, but their concerns were serious enough that addressing them significantly improved user trust.


Ensuring Long-Term Stability


To prevent future issues, I also:


Conclusion


This experience taught me that even if a bug is rare, its impact matters. If users feel lost, especially during critical moments like payments, trust is broken instantly. By improving feedback and giving users a clear way forward, we turned a hidden issue into a transparent, reliable experience.