frappe-impl-clientscripts
Client Scripts — Implementation Workflows
Step-by-step workflows for building client-side form features. For exact API syntax, see frappe-syntax-clientscripts.
Version: v14/v15/v16 | Note: v13 renamed "Custom Script" to "Client Script"
Quick Decision: Client or Server?
MUST the logic ALWAYS execute (imports, API, Data Import)?
├── YES → Server Script or Controller
└── NO → What is the goal?
├── UI feedback / UX → Client Script
├── Show/hide fields → Client Script
├── Link filters → Client Script
├── Data validation → BOTH (client for UX, server for integrity)
└── Calculations → Client for display, server for critical
Rule: ALWAYS use Client Scripts for UX. ALWAYS back critical logic with server-side validation.
Workflow 1: Create a Client Script via UI
- Navigate to Setup > Client Script (or type "New Client Script" in awesomebar)
- Select the target DocType
- ALWAYS set Enabled checkbox
- Write script using the
frappe.ui.form.onpattern - Save — script is active immediately (no restart needed)
- Open target DocType form → test behavior
- Open browser DevTools Console (F12) for debugging
When to migrate to custom app: ALWAYS migrate when the script exceeds 50 lines, needs version control, or must be deployed across environments.
Workflow 2: Choose the Right Event
WHAT DO YOU WANT?
├── Set link filters → setup (once, earliest lifecycle)
├── Add custom buttons → refresh (re-added after each render)
├── Show/hide fields → refresh + {fieldname} (BOTH needed)
├── Validate before save → validate (frappe.throw stops save)
├── Action after save → after_save
├── Calculate on change → {fieldname} handler
├── Child row added → {tablename}_add
├── Child row removed → {tablename}_remove
├── Child field changed → Child DocType: {fieldname}
├── One-time init → setup or onload
└── After full DOM render → onload_post_render
See references/decision-tree.md for complete event timing matrix.
Workflow 3: Field Visibility Toggle
Goal: Show "delivery_date" only when "requires_delivery" is checked.
Step 1: Implement BOTH refresh and fieldname events:
frappe.ui.form.on('Sales Order', {
refresh(frm) {
frm.trigger('requires_delivery'); // Set initial state
},
requires_delivery(frm) {
frm.toggle_display('delivery_date', frm.doc.requires_delivery);
frm.toggle_reqd('delivery_date', frm.doc.requires_delivery);
}
});
Why both? refresh sets state on form load. {fieldname} responds to user interaction. NEVER use only one — the form will show wrong state on load or on change.
Workflow 4: Cascading Link Filters
Goal: Filter "city" based on selected "country".
frappe.ui.form.on('Customer', {
setup(frm) {
// ALWAYS set filters in setup — ensures consistency
frm.set_query('city', () => ({
filters: { country: frm.doc.country || '' }
}));
},
country(frm) {
frm.set_value('city', ''); // ALWAYS clear dependent field
}
});
Rule: ALWAYS put set_query in setup. ALWAYS clear child fields when parent changes.
Workflow 5: Calculated Fields (Child Table)
Goal: Calculate row amounts and document totals.
frappe.ui.form.on('Invoice Item', {
qty(frm, cdt, cdn) { calculate_row(frm, cdt, cdn); },
rate(frm, cdt, cdn) { calculate_row(frm, cdt, cdn); },
amount(frm) { calculate_totals(frm); }
});
frappe.ui.form.on('Invoice', {
items_remove(frm) { calculate_totals(frm); }
});
function calculate_row(frm, cdt, cdn) {
let row = frappe.get_doc(cdt, cdn);
frappe.model.set_value(cdt, cdn, 'amount',
flt(row.qty) * flt(row.rate));
}
function calculate_totals(frm) {
let total = (frm.doc.items || []).reduce(
(sum, row) => sum + flt(row.amount), 0);
frm.set_value('grand_total', flt(total, 2));
}
Rules:
- ALWAYS use
flt()for numeric operations (handles null/undefined) - ALWAYS handle
items_remove— totals must recalculate on row deletion - NEVER call
refresh_fieldafterset_value— it triggers automatically
Workflow 6: Server Calls: Which Method to Use
NEED TO CALL THE SERVER?
├── Fetch a single value?
│ └── frappe.db.get_value(doctype, name, fields)
│ Returns: Promise — lightweight, no whitelist needed
│
├── Call a document's controller method?
│ └── frm.call(method, args)
│ Requires: @frappe.whitelist() on controller method
│ Auto-includes: doctype, docname, doc context
│
├── Call a standalone whitelisted function?
│ └── frappe.call({method: 'dotted.path', args: {}})
│ Requires: @frappe.whitelist() decorator
│ Returns: Promise with r.message
│
└── Need Promise-only (no callback)?
└── frappe.xcall('dotted.path', args)
Same as frappe.call but returns clean Promise
Example — frm.call:
frm.call('calculate_taxes').then(r => {
frm.reload_doc(); // Refresh after server-side changes
});
Example — frappe.xcall:
let result = await frappe.xcall(
'myapp.api.check_credit', { customer: frm.doc.customer });
Workflow 7: Custom Button Implementation
frappe.ui.form.on('Sales Order', {
refresh(frm) {
// ALWAYS check conditions before adding buttons
if (!frm.is_new() && frm.doc.docstatus === 1) {
frm.add_custom_button(__('Create Invoice'), () => {
create_invoice(frm);
}, __('Create')); // Group label
}
}
});
Rules:
- ALWAYS add buttons in
refresh— they are cleared on each render - ALWAYS check
frm.is_new()— buttons on unsaved docs cause errors - ALWAYS wrap button labels in
__()for translation - NEVER add buttons in
setuporonload— UI not ready
Workflow 8: Async Validation with Server Check
frappe.ui.form.on('Sales Order', {
async validate(frm) {
if (!frm.doc.customer || !frm.doc.grand_total) return;
let r = await frappe.call({
method: 'myapp.api.check_credit',
args: {
customer: frm.doc.customer,
amount: frm.doc.grand_total
}
});
if (r.message && !r.message.allowed) {
frappe.throw(__('Credit limit exceeded. Available: {0}',
[r.message.available]));
}
}
});
Rules:
- ALWAYS use
async/awaitfor server calls invalidate - ALWAYS use
frappe.throw()to stop save —msgprintdoes NOT stop it - NEVER put slow server calls in
validatewithout user expectation
Workflow 9: Debugging in Browser
- Open F12 DevTools > Console
- Add
console.log(frm.doc)in your event handler - Use
cur_frmin Console to inspect current form state - Check Network tab for failed
frappe.callrequests - Use
frappe.ui.form.handlersto see registered event handlers
Debug pattern:
frappe.ui.form.on('MyDocType', {
my_field(frm) {
console.log('Field changed:', frm.doc.my_field);
// ... actual logic
}
});
Workflow 10: Migrate Client Script to Custom App
- Create JS file:
myapp/myapp/public/js/sales_order.js - Move script content to the file (keep
frappe.ui.form.onwrapper) - Register in
hooks.py:doctype_js = { "Sales Order": "public/js/sales_order.js" } - Run
bench build(orbench watchfor development) - Delete the Client Script document from Setup
- Test on the form — behavior must be identical
ALWAYS migrate when: version control needed, multi-environment deployment, script > 50 lines, team collaboration required.
Performance Rules
| Rule | Why |
|---|---|
set_query in setup only |
Prevents re-registration on every refresh |
Batch set_value calls |
frm.set_value({a: 1, b: 2}) — one update, not two |
| Cache server responses | Store in frm._cache_key to avoid repeat calls |
| NEVER query in loops | Fetch all data once, build lookup map |
Use frappe.db.get_value |
Lighter than frappe.call for simple lookups |
Related Skills
frappe-syntax-clientscripts— Exact API syntax and method signaturesfrappe-errors-clientscripts— Error handling and common pitfallsfrappe-syntax-whitelisted— Server methods callable from clientfrappe-core-database—frappe.db.*client-side APIfrappe-impl-serverscripts— When to move logic server-side
See references/decision-tree.md for event selection. See references/workflows.md for extended patterns. See references/examples.md for 10+ complete examples.