Frappe_Claude_Skill_Package frappe-impl-clientscripts
git clone https://github.com/OpenAEC-Foundation/Frappe_Claude_Skill_Package
T=$(mktemp -d) && git clone --depth=1 https://github.com/OpenAEC-Foundation/Frappe_Claude_Skill_Package "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/source/impl/frappe-impl-clientscripts" ~/.claude/skills/openaec-foundation-frappe-claude-skill-package-frappe-impl-clientscripts && rm -rf "$T"
skills/source/impl/frappe-impl-clientscripts/SKILL.mdClient 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
patternfrappe.ui.form.on - 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
for numeric operations (handles null/undefined)flt() - ALWAYS handle
— totals must recalculate on row deletionitems_remove - NEVER call
afterrefresh_field
— it triggers automaticallyset_value
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
— they are cleared on each renderrefresh - ALWAYS check
— buttons on unsaved docs cause errorsfrm.is_new() - ALWAYS wrap button labels in
for translation__() - NEVER add buttons in
orsetup
— UI not readyonload
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
for server calls inasync/awaitvalidate - ALWAYS use
to stop save —frappe.throw()
does NOT stop itmsgprint - NEVER put slow server calls in
without user expectationvalidate
Workflow 9: Debugging in Browser
- Open F12 DevTools > Console
- Add
in your event handlerconsole.log(frm.doc) - Use
in Console to inspect current form statecur_frm - Check Network tab for failed
requestsfrappe.call - Use
to see registered event handlersfrappe.ui.form.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
wrapper)frappe.ui.form.on - Register in
:hooks.pydoctype_js = { "Sales Order": "public/js/sales_order.js" } - Run
(orbench build
for development)bench watch - 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 |
|---|---|
in only | Prevents re-registration on every refresh |
Batch calls | — one update, not two |
| Cache server responses | Store in to avoid repeat calls |
| NEVER query in loops | Fetch all data once, build lookup map |
Use | Lighter than for simple lookups |
Related Skills
— Exact API syntax and method signaturesfrappe-syntax-clientscripts
— Error handling and common pitfallsfrappe-errors-clientscripts
— Server methods callable from clientfrappe-syntax-whitelisted
—frappe-core-database
client-side APIfrappe.db.*
— When to move logic server-sidefrappe-impl-serverscripts
See references/decision-tree.md for event selection. See references/workflows.md for extended patterns. See references/examples.md for 10+ complete examples.