For Cleaning Businesses

Stop Answering
"How Much for Cleaning?"
Let a Calculator Do It

Free quote calculator for your cleaning business website. Customers get instant prices. You get qualified leads. Setup in 5 minutes.

No credit card required
Free forever for calculators
Works on Wix, WordPress, any site
๐Ÿงน Get Your Cleaning Quote โœจ
๐Ÿ  Property Details
 
 
 
๐Ÿงผ Cleaning Options
 
 
โž• Add-On Services
 
 
๐Ÿ’ฐ Your Quote
$ 250.00
๐Ÿ’ก Choose recurring service to save up to 20%
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
export function cleaningQuoteForm(form: FormTs) {
form.setTitle(() => '๐Ÿงน Get Your Cleaning Quote โœจ');
 
form.configureCompletionScreen({
type: 'text',
title: () => '๐ŸŽ‰ Quote Ready!',
message: () => 'Your cleaning quote has been generated. Download your PDF to save the details.'
});
 
// Pricing data
const baseRates: Record<string, number> = {
'apartment': 80,
'house': 120,
'condo': 95,
'townhouse': 110
};
 
const cleaningMultipliers: Record<string, number> = {
'standard': 1.0,
'deep': 1.5,
'move-out': 1.75
};
 
const frequencyDiscounts: Record<string, number> = {
'one-time': 0,
'weekly': 20,
'bi-weekly': 15,
'monthly': 10
};
 
// Add-on prices
const addonPrices: Record<string, number> = {
'insideOven': 35,
'insideFridge': 35,
'insideCabinets': 50,
'windowsInterior': 45,
'laundry': 30,
'dishes': 25,
'organizing': 60,
'petHairRemoval': 40
};
 
// Property Details
const property = form.addSubform('property', {
title: () => '๐Ÿ  Property Details',
mobileBreakpoint: 0
});
 
property.addRow(row => {
row.addDropdown('type', {
label: 'Property Type',
defaultValue: 'house',
isRequired: true,
options: [
{ id: 'apartment', name: '๐Ÿข Apartment' },
{ id: 'house', name: '๐Ÿก House' },
{ id: 'condo', name: '๐Ÿฌ Condo' },
{ id: 'townhouse', name: '๐Ÿ˜๏ธ Townhouse' }
]
}, '1fr');
 
row.addInteger('sqft', {
label: 'Square Feet',
min: 500,
max: 5000,
defaultValue: 1500,
isRequired: true,
placeholder: 'e.g. 1500'
}, '1fr');
});
 
property.addRow(row => {
row.addInteger('bedrooms', {
label: 'Bedrooms',
min: 1,
max: 6,
defaultValue: 3,
isRequired: true
}, '1fr');
 
row.addInteger('bathrooms', {
label: 'Bathrooms',
min: 1,
max: 5,
defaultValue: 2,
isRequired: true
}, '1fr');
});
 
// Cleaning Options
const options = form.addSubform('options', {
title: () => '๐Ÿงผ Cleaning Options',
mobileBreakpoint: 0
});
 
options.addRow(row => {
row.addRadioButton('cleaningType', {
label: 'Cleaning Type',
defaultValue: 'standard',
isRequired: true,
orientation: 'vertical',
options: [
{ id: 'standard', name: '๐Ÿงน Standard' },
{ id: 'deep', name: '๐Ÿ”ฌ Deep Clean' },
{ id: 'move-out', name: '๐Ÿ“ฆ Move-Out' }
]
});
 
row.addRadioButton('frequency', {
label: 'Frequency',
defaultValue: 'one-time',
isRequired: true,
orientation: 'vertical',
options: [
{ id: 'one-time', name: '1๏ธโƒฃ One-Time' },
{ id: 'weekly', name: '๐Ÿ“† Weekly' },
{ id: 'bi-weekly', name: '๐Ÿ—“๏ธ Bi-Weekly' },
{ id: 'monthly', name: '๐Ÿ“… Monthly' }
]
});
});
 
// Add-ons as CheckboxList
const addons = form.addSubform('addons', {
title: () => 'โž• Add-On Services',
mobileBreakpoint: 500
});
 
addons.addRow(row => {
row.addCheckboxList('kitchenExtras', {
label: 'Kitchen Extras',
orientation: 'vertical',
options: [
{ id: 'insideOven', name: '๐Ÿ”ฅ Inside Oven' },
{ id: 'insideFridge', name: 'โ„๏ธ Inside Fridge' },
{ id: 'insideCabinets', name: '๐Ÿ—„๏ธ Inside Cabinets' },
{ id: 'dishes', name: '๐Ÿฝ๏ธ Wash Dishes' }
]
}, '1fr');
 
row.addCheckboxList('homeExtras', {
label: 'Home Extras',
orientation: 'vertical',
options: [
{ id: 'windowsInterior', name: '๐ŸชŸ Interior Windows' },
{ id: 'laundry', name: '๐Ÿงบ Laundry' },
{ id: 'organizing', name: '๐Ÿ“š Organizing' },
{ id: 'petHairRemoval', name: '๐Ÿพ Pet Hair Removal' }
]
}, '1fr');
});
 
// Quote Summary
const summary = form.addSubform('summary', {
title: () => '๐Ÿ’ฐ Your Quote',
isCollapsible: false,
});
 
// Calculate base price
const getBasePrice = () => {
const type = property.dropdown('type')?.value() || 'house';
const sqft = property.integer('sqft')?.value() || 1500;
const bedrooms = property.integer('bedrooms')?.value() || 3;
const bathrooms = property.integer('bathrooms')?.value() || 2;
 
const baseRate = baseRates[type] || 120;
const sqftCost = Math.floor(sqft / 500) * 15;
const roomCost = (bedrooms * 15) + (bathrooms * 20);
 
return baseRate + sqftCost + roomCost;
};
 
// Calculate add-ons total
const getAddonsTotal = () => {
let total = 0;
const kitchenExtras = addons.checkboxList('kitchenExtras')?.value() || [];
const homeExtras = addons.checkboxList('homeExtras')?.value() || [];
const allSelected = [...kitchenExtras, ...homeExtras];
for (const addonId of allSelected) {
total += addonPrices[addonId] || 0;
}
return total;
};
 
// Get selected add-ons for display
const getSelectedAddons = () => {
const kitchenExtras = addons.checkboxList('kitchenExtras')?.value() || [];
const homeExtras = addons.checkboxList('homeExtras')?.value() || [];
return [...kitchenExtras, ...homeExtras];
};
 
// Calculate final price
const getFinalPrice = () => {
const cleaningType = options.radioButton('cleaningType')?.value() || 'standard';
const frequency = options.radioButton('frequency')?.value() || 'one-time';
 
const basePrice = getBasePrice();
const multiplier = cleaningMultipliers[cleaningType] || 1.0;
const adjustedBase = basePrice * multiplier;
 
const addonsTotal = getAddonsTotal();
const subtotal = adjustedBase + addonsTotal;
 
const discountPercent = frequencyDiscounts[frequency] || 0;
const discount = subtotal * (discountPercent / 100);
 
return Math.round(subtotal - discount);
};
 
// Get discount amount
const getDiscountAmount = () => {
const cleaningType = options.radioButton('cleaningType')?.value() || 'standard';
const frequency = options.radioButton('frequency')?.value() || 'one-time';
 
const basePrice = getBasePrice();
const multiplier = cleaningMultipliers[cleaningType] || 1.0;
const adjustedBase = basePrice * multiplier;
 
const addonsTotal = getAddonsTotal();
const subtotal = adjustedBase + addonsTotal;
 
const discountPercent = frequencyDiscounts[frequency] || 0;
return Math.round(subtotal * (discountPercent / 100));
};
 
summary.addRow(row => {
row.addPriceDisplay('totalPrice', {
label: 'Estimated Price',
computedValue: () => getFinalPrice(),
alignment: 'center',
variant: 'large',
suffix: () => {
const frequency = options.radioButton('frequency')?.value();
if (frequency === 'weekly') return '/week';
if (frequency === 'bi-weekly') return '/visit';
if (frequency === 'monthly') return '/month';
return '';
}
});
});
 
summary.addRow(row => {
row.addTextPanel('savings', {
computedValue: () => {
const frequency = options.radioButton('frequency')?.value() || 'one-time';
const discountPercent = frequencyDiscounts[frequency] || 0;
 
if (discountPercent > 0) {
return `๐ŸŽ‰ You save ${discountPercent}% with ${frequency} service!`;
}
return '๐Ÿ’ก Choose recurring service to save up to 20%';
},
customStyles: {
fontSize: '0.9rem',
color: '#059669',
fontWeight: '500',
textAlign: 'center'
}
});
});
 
form.configureSubmitButton({
label: () => 'Get My Quote'
});
 
// Configure PDF generation
form.configurePdf('quoteDocument', (pdf) => {
pdf.configure({
filename: 'cleaning-quote.pdf',
pageSize: 'A4',
allowUserDownload: true,
downloadButtonLabel: 'Download Your Quote',
header: {
title: 'Cleaning Service Quote',
subtitle: 'Professional Home Cleaning Services'
},
footer: {
text: 'Thank you for choosing our cleaning services!',
showPageNumbers: true
}
});
 
// Property Summary Section
pdf.addSection('Property Details', (section) => {
section.addRow((row) => {
const propertyTypeMap: Record<string, string> = {
'apartment': 'Apartment',
'house': 'House',
'condo': 'Condo',
'townhouse': 'Townhouse'
};
const type = property.dropdown('type')?.value() || 'house';
row.addField('Property Type', propertyTypeMap[type] || type);
row.addField('Square Feet', `${property.integer('sqft')?.value() || 1500} sq ft`);
});
section.addRow((row) => {
row.addField('Bedrooms', `${property.integer('bedrooms')?.value() || 3}`);
row.addField('Bathrooms', `${property.integer('bathrooms')?.value() || 2}`);
});
});
 
// Cleaning Options Section
pdf.addSection('Service Details', (section) => {
section.addRow((row) => {
const cleaningTypeMap: Record<string, string> = {
'standard': 'Standard Cleaning',
'deep': 'Deep Clean',
'move-out': 'Move-Out Cleaning'
};
const type = options.radioButton('cleaningType')?.value() || 'standard';
row.addField('Cleaning Type', cleaningTypeMap[type] || type);
});
section.addRow((row) => {
const frequencyMap: Record<string, string> = {
'one-time': 'One-Time Service',
'weekly': 'Weekly',
'bi-weekly': 'Bi-Weekly',
'monthly': 'Monthly'
};
const freq = options.radioButton('frequency')?.value() || 'one-time';
row.addField('Frequency', frequencyMap[freq] || freq);
});
});
 
// Add-ons Section (only if any selected)
const selectedAddons = getSelectedAddons();
if (selectedAddons.length > 0) {
pdf.addSection('Add-On Services', (section) => {
const addonLabels: Record<string, string> = {
'insideOven': 'Inside Oven',
'insideFridge': 'nside Fridge',
'insideCabinets': 'Inside Cabinets',
'dishes': 'Wash Dishes',
'windowsInterior': 'Interior Windows',
'laundry': 'Laundry',
'organizing': 'Organizing',
'petHairRemoval': 'Pet Hair Removal'
};
 
const tableRows = selectedAddons.map(addonId => [
addonLabels[addonId] || addonId,
`$${addonPrices[addonId] || 0}`
]);
 
section.addTable(['Service', 'Price'], tableRows);
section.addSpacer(10);
section.addRow((row) => {
row.addField('Add-ons Subtotal', `$${getAddonsTotal()}`);
});
});
}
 
// Pricing Summary Section
pdf.addSection('Quote Summary', (section) => {
const cleaningType = options.radioButton('cleaningType')?.value() || 'standard';
const frequency = options.radioButton('frequency')?.value() || 'one-time';
const multiplier = cleaningMultipliers[cleaningType] || 1.0;
const basePrice = getBasePrice();
const adjustedBase = Math.round(basePrice * multiplier);
 
section.addRow((row) => {
row.addField('Base Service Price', `$${adjustedBase}`);
});
 
if (selectedAddons.length > 0) {
section.addRow((row) => {
row.addField('Add-ons Total', `$${getAddonsTotal()}`);
});
}
 
const discountAmount = getDiscountAmount();
if (discountAmount > 0) {
const discountPercent = frequencyDiscounts[frequency] || 0;
section.addRow((row) => {
row.addField(`Discount (${discountPercent}% ${frequency})`, `-$${discountAmount}`);
});
}
 
section.addDivider();
section.addSpacer(10);
 
const suffix = frequency === 'weekly' ? '/week'
: frequency === 'bi-weekly' ? '/visit'
: frequency === 'monthly' ? '/month'
: '';
 
section.addRow((row) => {
row.addField('TOTAL', `$${getFinalPrice()}${suffix}`);
});
 
if (discountAmount > 0) {
section.addSpacer(10);
section.addText(`You're saving $${discountAmount} with your ${frequency} service!`);
}
});
 
// Terms Section
pdf.addSection('Terms & Conditions', (section) => {
section.addText('โ€ข This quote is valid for 30 days from the date of generation.');
section.addText('โ€ข Prices may vary based on actual property condition upon inspection.');
section.addText('โ€ข Cancellations require 24-hour advance notice.');
section.addText('โ€ข Recurring service discounts apply to the total including add-ons.');
section.addSpacer(10);
section.addText('Questions? Contact us at support@cleaningservice.com');
});
});
}

Why Cleaning Businesses Love It

Stop playing phone tag. Let customers quote themselves while you focus on what matters.

Save 2+ Hours Daily

No more back-and-forth phone calls. Customers get instant prices 24/7.

More Bookings

Instant pricing means less friction. Visitors convert while they're interested.

Look Professional

Match the big cleaning companies. Interactive quotes build trust instantly.

No Coding Required

Describe what you need. Our AI builds your calculator in minutes.

Add a Quote Calculator in 3 Steps

1

Build Your Calculator

Use our AI assistant or pick a template. Set your prices, add-ons, and discounts. Preview in real-time.

2

Customize Your Look

Match your brand colors and add your logo. Mobile-friendly by default. Looks great everywhere.

3

Embed Anywhere

Copy 2 lines of code. Works on WordPress, Wix, Squarespace, Shopify, or any custom site.

Add to your website
<script src="https://formts.com/widget.js"></script>
<formts-widget link-id="your-calculator"></formts-widget>

Everything You Need to Automate Quotes

Real-Time Calculations

Prices update instantly as customers fill the form. No page reloads, no waiting.

Email Notifications

Get notified the moment someone requests a quote. Never miss a lead.

PDF Quotes

Auto-generate professional quote PDFs with your branding. Send instantly.

Conditional Logic

Show or hide options based on customer choices. Smart forms that adapt.

Works Everywhere

Embed on any website with 2 lines of code. WordPress, Wix, Squarespace, custom sites.

Mobile Optimized

Looks great on phones and tablets. Most of your customers browse on mobile.

Frequently Asked Questions

Do I need to know how to code?

No. Our AI assistant builds the calculator from your description. Just tell it what services you offer and your prices. You can also start from a template and customize it - no coding required.

Will it work on my Wix/WordPress/Squarespace site?

Yes. FormTs works on any website that allows custom HTML. You just paste 2 lines of code where you want the calculator to appear. Works on WordPress, Wix, Squarespace, Shopify, and any custom site.

Is the calculator really free?

Yes. Calculators that don't collect submissions are completely free - no limits, no credit card. If you want to collect customer details, the Free plan includes 100 submissions/month. Need more? Pro gives you 1,000/month.

Can I customize the prices and services?

Absolutely. You control everything - base prices, add-ons, discounts, minimum charges, rush fees, service areas. The calculator uses YOUR pricing formulas exactly as you define them.

How do I get notified when someone requests a quote?

All submissions appear in your dashboard instantly. Pro plan includes email notifications and webhooks to connect with Zapier, n8n, Make, or send data directly to your CRM.

What if I want to change the calculator later?

Edit anytime from your dashboard. Changes go live instantly - no need to touch your website code again. Update prices, add new services, or change the design whenever you want.

Ready to Stop Wasting Time on Quotes?

Join hundreds of cleaning business owners who automated their pricing. Start with a free template or let AI build your custom calculator.

No credit card required. Free plan available. Setup in under 10 minutes.