Note
This is my old workflow. These days I use Cakedesk, an invoicing app for creative freelancers.
I always enjoy behind-the-scenes looks into other peoples' businesses, so I thought I'd provide one of my own.
This is the workflow I've created to generate invoices for my clients:
1. Create Client and Project in Airtable
I keep two Airtables: "Clients" and "Projects".
My Clients Airtable contains important data for the invoice such as the client's address. My Projects Airtable contains a reference to the client and the project's price.
I input this data by hand when I start working with a client.
data:image/s3,"s3://crabby-images/d5b02/d5b02be8d702f6f7de20c868e37f8e08f8dc0a17" alt="Screenshot of two Windows showing Clients and Projects in Airtable"
2. Generate the Invoice via Node.js CLI
When it's time to create the invoice for the project, I use a little Node.js script to fetch the client + project data from Airtable and render the invoice from it it as HTML.
This gives me the freedom to take complete control over the content and layout of the invoice.
data:image/s3,"s3://crabby-images/5f0c5/5f0c516d60a6efe35c60c868c2f2882b6f995f2a" alt="An animated GIF showing CLI prompts to gather invoice data"
The script also uses Puppeteer to generate a PDF-version of the invoice that I can send out to the client.
data:image/s3,"s3://crabby-images/59e67/59e67cf5a6b3158374acb2f03a1491fa1e520f92" alt="A PDF invoice"
I'm really happy that I invested the time to build this little system when I started freelancing. I can customize it completely to my needs and I don't need to fiddle around with spreadsheets or word processors.
Some More Details
- Depending on the client's language (which I also keep in Airtable), the invoice can be generated in English or German
- To generate the prompts during the invoice generation, I use Inquirer.js
- For the invoice template, I use Pug (formerly Jade) and plain CSS
- My Airtable setup isn't optimal because each project can have multiple invoices associated with it – I'll improve this in the future :)