The SmarkForm Constructor

📖 Table of Contents

Constructor Syntax

const form = new SmarkForm(element, options);
  • element — a DOM element (typically <form>, <div>, or any container) to enhance. The root element does not need a data-smark attribute — it is enhanced automatically. All descendants with data-smark are recursively processed.
  • options (optional) — a plain object that configures the form. See below for how these options work.

How Constructor Options Work

Most options you pass to the constructor are passed through to the root form component — they behave as if you had set them via data-smark on the root element. Only a few are truly constructor-only (no data-smark equivalent).

Pass-through options

// These configure the root form component, not the constructor itself:
const form = new SmarkForm(element, {
  value: { name: "Alice" },
  exportEmpties: false,
  focus_on_click: false,
});

Pass-through options are fully documented on the component type page they belong to. See:

Constructor-only options

// These are handled by the constructor and have no data-smark equivalent:
const form2 = new SmarkForm(element, {
  customActions: { /* … */ },
  on_click(ev) { /* … */ },
  allowExternalMixins: "allow",
});

See Constructor-only Options below.

Merging rules

If the root element also carries a data-smark attribute, both sources are merged. The constructor options take precedence.

<form data-smark='{"exportEmpties":true}'>
  <!-- … -->
</form>
// exportEmpties from data-smark is overridden:
const form = new SmarkForm(document.querySelector("form"), {
  exportEmpties: false, // wins
});

Options not specified in either source keep their documented defaults.


Constructor-only Options

These options can only be set via the constructor.

customActions

Defines additional actions beyond SmarkForm’s built-in ones. Triggers in the HTML can invoke them by name.

const form = new SmarkForm(element, {
  customActions: {
    async sendEmail(data, options) {
      const formData = await form.export();
      await fetch("/api/send", {
        method: "POST",
        body: JSON.stringify(formData),
      });
    },
  },
});
<button data-smark='{"action":"sendEmail"}'>Send Email</button>

Each custom action follows the async actionName(data, options) signature and participates in the standard action lifecycle (BeforeAction_<name>, AfterAction_<name>).

on* event handlers

A declarative shorthand for attaching event listeners at construction time. The naming follows the three listener levels:

Pattern Scope
on_<event> Bubbles if event permits
onLocal_<event> Target phase only
onAll_<event> Always bubbles
onBeforeAction_<action> BeforeAction hook
const form = new SmarkForm(element, {
  on_click(ev) { console.log("clicked", ev.target); },
  onLocal_AfterAction_export(ev) { console.log("exported", ev.data); },
  onBeforeAction_import(ev) { ev.preventDefault(); },
});

See: Event handlers via options for more examples.

Mixin security policies

SmarkForm mixin types can load external templates and execute scripts. Four constructor options control the security policy:

  • allowExternalMixins — fetch templates from external URLs
  • allowLocalMixinScripts — execute <script> blocks in local templates
  • allowSameOriginMixinScripts — execute same-origin external scripts
  • allowCrossOriginMixinScripts — execute cross-origin external scripts

Each accepts "block" (default, safer) or "allow". See Mixin security options for full documentation and examples.


How the Component Tree is Built

When you call new SmarkForm(element, options), the constructor creates a root form component from the given element and then builds the component tree recursively:

  1. The root form scans its direct children for data-smark attributes and enhances each one into its corresponding component type.
  2. Components that can contain children (form and list types) then scan their own direct children and repeat the process.
  3. Any component without data-smark is ignored — only descendants with the attribute are enhanced.

This means the tree depth matches your HTML nesting: a form with a nested list, which in turn has nested fields, produces precisely that three-level component structure.

Scope boundaries

Each form and list component only sees the data-smark children that directly belong to it — it does not reach into nested forms or lists. Those nested containers are responsible for their own children.

<div data-smark='{"type":"form","name":"parent"}'>
  <!-- The root form sees this input as a direct child -->
  <input name="name" data-smark>

  <!-- The root form sees this nested form, but NOT its children -->
  <div data-smark='{"type":"form","name":"address"}'>
    <!-- The nested form sees these children -->
    <input name="street" data-smark>
    <input name="city" data-smark>
  </div>
</div>

Rendering order

Rendering is asynchronous and proceeds outward-in: child components render before their parents are fully resolved. The rendered Promise signals when the entire tree has finished rendering, at which point every component is ready for interaction.


The rendered Promise

SmarkForm rendering is asynchronous. The rendered property returns a Promise that resolves once the full component tree has been rendered:

await form.rendered;
const field = form.find("/name"); // safe now

Methods like find(), export(), and import() depend on the form being fully rendered.