splitforms.com
QUOTE REQUEST · NEXT.JS

Quote Request for Next.js

Lead-capture form for agencies, freelancers, and service businesses. Free for 1,000 submissions per month — no backend, no SDK, no plugin.

1,000/mo free·no card·drop-in for Next.js
Form.tsxtsx57 lines
01'use client';
02
03import { useState, type FormEvent } from 'react';
04
05export default function QuoteForm() {
06 const [status, setStatus] = useState<'idle' | 'sending' | 'sent' | 'error'>('idle');
07
08 async function onSubmit(e: FormEvent<HTMLFormElement>) {
09 e.preventDefault();
10 setStatus('sending');
11
12 const data = new FormData(e.currentTarget);
13 data.set('access_key', 'YOUR_ACCESS_KEY');
14 data.set('subject', 'New quote request');
15
16 const res = await fetch('https://splitforms.com/api/submit', {
17 method: 'POST',
18 body: data,
19 headers: { Accept: 'application/json' },
20 });
21
22 const json = await res.json();
23 setStatus(json.success ? 'sent' : 'error');
24 if (json.success) e.currentTarget.reset();
25 }
26
27 if (status === 'sent') return <p>Thanks — we&rsquo;ll be in touch.</p>;
28
29 return (
30 <form onSubmit={onSubmit}>
31 <label htmlFor="name">Name *</label>
32 <input id="name" type="text" name="name" placeholder="Jane Builder" required />
33 <label htmlFor="email">Email *</label>
34 <input id="email" type="email" name="email" placeholder="jane@example.com" required />
35 <label htmlFor="phone">Phone *</label>
36 <input id="phone" type="tel" name="phone" placeholder="+1 415 555 0142" required />
37 <label htmlFor="company">Company</label>
38 <input id="company" type="text" name="company" placeholder="Acme Inc" />
39 <label htmlFor="budget">Budget *</label>
40 <select id="budget" name="budget" required>
41 <option value="">Choose…</option>
42 <option>&lt; $5k</option>
43 <option>$5k–$15k</option>
44 <option>$15k–$50k</option>
45 <option>$50k+</option>
46 </select>
47 <label htmlFor="project">Tell us about the project *</label>
48 <textarea id="project" name="project" placeholder="Goals, timeline, anything we should know." required />
49
50 <button type="submit" disabled={status === 'sending'}>
51 {status === 'sending' ? 'Sending…' : 'Send'}
52 </button>
53
54 {status === 'error' && <p>Something went wrong. Try again.</p>}
55 </form>
56 );
57}
1,000
submissions / mo, free
6
fields, ready to ship
5
code outputs
60s
from copy to inbox
§ 01Quote Request × Next.jswhy this combination, in 80 words

Built for Next.js developers who hate operating a backend.

Splitforms is the form-to-email API for Next.js sites. One POST endpoint, no SDK, no plugin — drop the quote request into a page and ship.

Splitforms is the form-to-email API for Next.js sites. One POST endpoint, spam filtering, signed webhooks, file uploads, a real dashboard — drop-in, no server, no PHP. Free for 1,000 submissions per month, $5/mo Pro for 5,000, $59 for 4 years if you hate recurring SaaS bills.

✦ what you get on the free plan
  • 1,000 form submissions per month
  • Unlimited forms — one key, many pages
  • Spam protection (honeypot + classifier)
  • Webhooks: Slack, Discord, WhatsApp, custom
  • CSV export of all submissions
  • Email notifications (CC and BCC on Pro)
§ 02Copy-paste codeReact / Next.js · 57 lines

Drop into any Next.js project.

Replace YOUR_ACCESS_KEY with your splitforms key, paste into a Next.js page, and ship. No build-time integration required.

Form.tsxtsx57 lines
01'use client';
02
03import { useState, type FormEvent } from 'react';
04
05export default function QuoteForm() {
06 const [status, setStatus] = useState<'idle' | 'sending' | 'sent' | 'error'>('idle');
07
08 async function onSubmit(e: FormEvent<HTMLFormElement>) {
09 e.preventDefault();
10 setStatus('sending');
11
12 const data = new FormData(e.currentTarget);
13 data.set('access_key', 'YOUR_ACCESS_KEY');
14 data.set('subject', 'New quote request');
15
16 const res = await fetch('https://splitforms.com/api/submit', {
17 method: 'POST',
18 body: data,
19 headers: { Accept: 'application/json' },
20 });
21
22 const json = await res.json();
23 setStatus(json.success ? 'sent' : 'error');
24 if (json.success) e.currentTarget.reset();
25 }
26
27 if (status === 'sent') return <p>Thanks — we&rsquo;ll be in touch.</p>;
28
29 return (
30 <form onSubmit={onSubmit}>
31 <label htmlFor="name">Name *</label>
32 <input id="name" type="text" name="name" placeholder="Jane Builder" required />
33 <label htmlFor="email">Email *</label>
34 <input id="email" type="email" name="email" placeholder="jane@example.com" required />
35 <label htmlFor="phone">Phone *</label>
36 <input id="phone" type="tel" name="phone" placeholder="+1 415 555 0142" required />
37 <label htmlFor="company">Company</label>
38 <input id="company" type="text" name="company" placeholder="Acme Inc" />
39 <label htmlFor="budget">Budget *</label>
40 <select id="budget" name="budget" required>
41 <option value="">Choose…</option>
42 <option>&lt; $5k</option>
43 <option>$5k–$15k</option>
44 <option>$15k–$50k</option>
45 <option>$50k+</option>
46 </select>
47 <label htmlFor="project">Tell us about the project *</label>
48 <textarea id="project" name="project" placeholder="Goals, timeline, anything we should know." required />
49
50 <button type="submit" disabled={status === 'sending'}>
51 {status === 'sending' ? 'Sending…' : 'Send'}
52 </button>
53
54 {status === 'error' && <p>Something went wrong. Try again.</p>}
55 </form>
56 );
57}
ALTPrefer plain HTML? View the universal quote request HTML snippet28 lines
form.htmlHTML
<form action="https://splitforms.com/api/submit" method="POST">
  <input type="hidden" name="access_key" value="YOUR_ACCESS_KEY">
  <input type="hidden" name="subject" value="New quote request">

  <label for="name">Name *</label>
  <input id="name" type="text" name="name" placeholder="Jane Builder" required>
  <label for="email">Email *</label>
  <input id="email" type="email" name="email" placeholder="jane@example.com" required>
  <label for="phone">Phone *</label>
  <input id="phone" type="tel" name="phone" placeholder="+1 415 555 0142" required>
  <label for="company">Company</label>
  <input id="company" type="text" name="company" placeholder="Acme Inc">
  <label for="budget">Budget *</label>
  <select id="budget" name="budget" required>
    <option value="">Choose…</option>
    <option>&lt; $5k</option>
    <option>$5k–$15k</option>
    <option>$15k–$50k</option>
    <option>$50k+</option>
  </select>
  <label for="project">Tell us about the project *</label>
  <textarea id="project" name="project" placeholder="Goals, timeline, anything we should know." required></textarea>

  <!-- honeypot — bots fill every field -->
  <input type="checkbox" name="botcheck" style="display:none" tabindex="-1" autocomplete="off">

  <button type="submit">Send</button>
</form>
§ 03Setup3 steps · 60 seconds · zero config

Generate, embed, receive.

Three actions stand between you and your first quote request submission. None of them require a backend, a database, or a CAPTCHA library.

STEP 01GENERATE

Generate a free splitforms key

Sign in at splitforms.com — your access key is created instantly. No credit card, no setup wizard, no SDK to install.

Create your form
key=sk_live_••••••••
STEP 02EMBED

Paste the quote request into your Next.js project

Drop the form snippet into a Next.js page, component, or layout. Replace YOUR_ACCESS_KEY with the key from your dashboard. The form action is a hard-coded URL — no env vars or build-time wiring needed.

snippettsx
'use client';
  …
</form>
STEP 03RECEIVE

Receive submissions

Email arrives within seconds. Webhook fires in parallel. Dashboard updates live. CSV export, Slack/Discord forwarding, BCC to your team — all included free.

inbox · 1 newjust now
FROM contact@yoursite.com
New quote request
Maya Iyer maya@studio71.co
Loved the demo — quick question about pricing on the 4-year plan. Are usage limits per project or account-wide?
§ 04Field-by-field rundown6 fields · names you POST

What every field actually does.

Each field below ships in the quote request template — rename, remove, or add your own. Splitforms accepts any name you POST.

nameREQUIRED
TEXT

Name

Greeting + dashboard label so submissions don't all read 'anonymous'.

placeholder · Jane Builder
emailREQUIRED
EMAIL

Email

Reply-to address — splitforms wires this so hitting reply goes back to the sender.

placeholder · jane@example.com
phoneREQUIRED
PHONE

Phone

Faster qualification — phone leads convert ~3× higher than email-only on B2B forms.

placeholder · +1 415 555 0142
company
TEXT

Company

Lets you sort enterprise vs SMB inquiries before you reply.

placeholder · Acme Inc
budgetREQUIRED
SELECT

Budget

Filters tire-kickers from real buyers without a sales call.

< $5k$5k–$15k$15k–$50k$50k+
projectREQUIRED
TEXTAREA

Tell us about the project

Free-text input — no character limit, expands as the visitor types.

placeholder · Goals, timeline, anything we should know.
§ 06Questions6 answered

Quote Request on Next.jsFAQ.

Direct answers, no marketing fluff. Missing one? Email hello@splitforms.com.

01Does this quote request work on Next.js?
Yes. The form is plain HTML with a single POST endpoint, so it runs on any Next.js site without server-side code, plugins, or SDKs. Drop the snippet into a Next.js page or component and submissions land in your inbox via splitforms.com.
02How much does the quote request cost on Next.js?
Free for 1,000 submissions per month — no credit card, no trial. Pro is $5/mo for 5,000 submissions, and there's a one-time $59 4-year plan (15,000 submissions/mo for 48 months). The same pricing applies regardless of which framework hosts the form.
03Can I customize the fields?
Yes. The template ships with sensible defaults (name, email, phone, company…) — add, remove, or rename any of them. Splitforms accepts whatever fields you POST.
04How does spam protection work on the quote request?
A hidden honeypot field catches dumb bots, and a tuned classifier scores the rest. You only see real submissions in your dashboard. No CAPTCHA, no friction for human users — and it works the same on Next.js as on any other framework.
05Can I send the quote request submissions to Slack or Discord?
Yes. Webhooks are free on every plan, with auto-formatted payloads for Slack, Discord, and WhatsApp (via CallMeBot). Or send raw signed JSON to any URL — Zapier, n8n, your own server. Configure in the splitforms dashboard.
06Will it work on a static Next.js site?
Yes — the form posts directly to splitforms from the browser, so no server is involved. Works on Vercel, Netlify, GitHub Pages, Cloudflare Pages, S3, or any plain Apache host.
§ 07Comparisonvs Web3Forms · vs Formspree

splitforms vs everything else.

Same drop-in API. More free submissions, real webhooks on the free tier, MCP support no other backend has.

FeatureWeb3FormsFormspreesplitforms
Free monthly submissions250501,000
Custom fields beyond contactYesPro tierFree
Webhooks (Slack / Discord)Pro tierPro tierFree, signed
AI / MCP submission inboxNoNoYes
Long-term plan (4-year flat)$59 once
✻ ✻ ✻

Ship a quote request on Next.js in 60 seconds.

1,000 submissions per month, free forever. No credit card. Copy the snippet above and paste it into your Next.js project.

Get free access key →Read the docs
v0.1 · founders pricing locked in · early access open