Next , React , Tailwind , Shadcn , Xata , Stripe , Drizzle

Invoice Generator

Headshot of Jamin Roberts Jamin Roberts
- Feb 27th 2024
Cover image for Invoice Generator
Stack: Next logoReact logoTailwind logoShadcn logoXata logoStripe logoDrizzle logo

Project Description

This project was built as a proof of concept for a client who needed an invoicing solution independent of any specific payment platform. The application allows users to create and manage invoices, which are then tied to a user or organization via an ID. Once the invoice is created, a PDF is generated and sent via email, containing a “pay now” link that redirects the user to a Stripe-hosted payment page. Upon successful payment, the invoice status is updated, and the process is complete.

The primary goal was to demonstrate how flexible and extensible this solution could be using a tech stack consisting of NextJS, React, Stripe, Xata, and a few other tools. The app’s target audience includes any business or individual looking for a custom invoicing platform that doesn’t directly tie into a payment system.

Challenges & Solutions

One of the biggest challenges in this project was generating the PDF as a downloadable file. After researching various options, I chose to implement an API endpoint wrapping the react-pdf/renderer package, which served the generated PDF to the frontend.

Another challenge involved tying invoices to either a user ID or an organization ID, which affected how invoices were fetched and stored. While I used a more barebones approach for now, future improvements will focus on reducing conditional checks and optimizing the save/insert/update logic.

Tech Stack and Rationale

  • NextJS & React: Chosen for their popularity, maintainability, and extensibility. There are abundant resources available for integrating Stripe and NextJS, making them an easy choice for rapid development.
  • Stripe: Stripe’s API documentation made it incredibly easy to tie invoices to a payment flow, allowing for quick integration and a smooth user experience.
  • Xata: Selected for its real-time and full-text search capabilities, making it an ideal choice for the app’s future growth. Xata’s REST API simplifies database management and integrates well with the app.
  • ShadCN & Tailwind: Chosen for accessibility and ease of styling, ensuring a smooth and flexible user interface that can be easily customized.

Lessons Learned

  • PDF generation and email styling: These tasks turned out to be much simpler than expected, thanks to libraries like react-pdf/renderer and react-email.
  • Clerk’s organization capabilities: A great discovery, Clerk provides an easy way to manage both individual and organization-based user roles, adding value to this and other future projects.
  • Xata vs Supabase: While Supabase’s row-level security is something I missed, Xata’s built-in text search and real-time capabilities made it an attractive option for this project. I would likely consider combining these technologies in the future to get the best of both worlds.

Next Steps

  • UI Polish and Design: A larger focus on UI design and user experience improvements is a priority. This will involve refining the interface and adding more visually appealing elements.
  • Template Customization: Adding the ability for users to select from various invoice templates for both the UI and generated PDFs/emails is a key next step.
  • Invoice Filtering: Leveraging Xata’s text search to enable filtering and searching invoices by titles, line items, and other fields is essential for improving usability as the app scales.