Gadget provides high-level, super-automatic components for assembling working user interfaces very quickly called autocomponents. Autocomponents are React components that implement forms and tables that can read and write to Gadget data in your app's database. Autocomponents use a design system under the hood which means they look and feel like the rest of your app.
When to use autocomponents
Autocomponents are intended to be high-quality interfaces for common tasks. While autocomponents offer a remarkably high degree of customizability, they are designed to help you build fast rather than build perfect. They are not intended to be a replacement for fully custom or artisanally designed user interfaces. If you need fine-grained control over your UX, Gadget recommends using the headless hooks that power autocomponents with head-ful design elements on top!
Installing autocomponents
Because autocomponents use a design system, you need to install the design system's dependencies to use them.
Currently, Gadget supports the following design systems:
Autocomponents are only available in @gadgetinc/react version 0.16.0 and later. To upgrade to the latest version, run the following in
the Gadget command palette:
Upgrade @gadgetinc/react to >0.16.0
yarnadd @gadgetinc/react
Installing Shopify Polaris autocomponents
If you did not select the Shopify app type when creating your app, you will need to install Shopify Polaris and set up the Polaris provider and stylesheet in your app.
Install the following dependencies:
install the Polaris library and the latest version of @gadgetinc/react
yarnadd @shopify/polaris @shopify/polaris-icons
Now import the Polaris provider and required style.css file. Add the following imports to the top of your web/components/App.jsx file:
web/components/App.jsx
React
import{AppProvider}from"@shopify/polaris";
import"@shopify/polaris/build/esm/styles.css";
import{AppProvider}from"@shopify/polaris";
import"@shopify/polaris/build/esm/styles.css";
Then replace your Layout component in the same file with the following code to make use of the Polaris provider:
<AutoForm /> renders a form that calls one of your app's backend API actions. On form submit, <AutoForm /> can call one of your model actions such as create, update, or a custom action. Global actions can also be run by <AutoForm />.
<AutoForm /> renders the correct HTML input element for each backend field and includes support for form validation, relationship fields, error handling, autocomplete, and file uploads.
For example, if you have a backend widget model with a name and inventoryCount field, you can render a form to create a new widget like so:
web/components/example.jsx
React
1// Render an AutoForm using Shopify's Polaris design system
9{/** put these two auto inputs in a wrapper component to align them horizontally */}
10<AutoInputfield="name"/>
11<AutoInputfield="inventoryCount"/>
12</InlineStack>
13{/** render a submit button */}
14<AutoSubmit/>
15</AutoForm>
16);
17}
Using <AutoForm /> without specifying the included fields or explicitly passing children will render
a form with all fields from the model or all params for an action. This means that adding a new field to a model or params to a global
action will also add an input for that field to the form.
Make sure you test any default <AutoForm /> components after making changes to your models or actions to ensure they still work as expected.
Import AutoForm
Import Shopify Polaris <AutoForm> components from @gadgetinc/react/auto/polaris:
import the AutoForm components into your frontend files
By default, <AutoForm /> will render input components to set or edit data for each field on a model record, or each global action parameter. Form inputs can be removed, re-ordered, or grouped.
To include only some of the fields of a model or action in your form, pass the include option with a list of fields to render:
You cannot have include and exclude defined for the same form.
<AutoForm /> does not currently support custom params with the object or array types.
AutoForm rich text fields
The auto-generated input for rich text fields in AutoForms require an extra dependency to be installed:
install the rich text editor dependency
yarnadd -D @mdxeditor/editor
Then import the stylesheet into your app:
web/components/App.jsx
React
import"@mdxeditor/editor/style.css";
import"@mdxeditor/editor/style.css";
Custom form components
<AutoForm /> can be passed an explicit set of children components or HTML elements to render instead of the default auto-generated components. This allows full control of field order and layout, and provides the ability to mix and match autocomponents with custom inputs.
To begin customizing the children, open the <AutoForm /> tag and pass the children components or HTML elements you'd like to render, like so:
add custom components or HTML elements to an AutoForm
React
<AutoFormaction={api.widget.create}>
<h1>My form title</h1>
{/* more form elements*/}
</AutoForm>
<AutoFormaction={api.widget.create}>
<h1>My form title</h1>
{/* more form elements*/}
</AutoForm>
To keep your form looking like the fully-automatic version of an <AutoForm/>, replace each field with an <AutoInput field="..."/> element and add an <AutoSubmit/> element:
render only the name and inventoryCount fields with AutoInput children
React
<AutoFormaction={api.widget.create}>
<AutoInputfield="name"/>
<AutoInputfield="inventoryCount"/>
<AutoSubmit/>
</AutoForm>
<AutoFormaction={api.widget.create}>
<AutoInputfield="name"/>
<AutoInputfield="inventoryCount"/>
<AutoSubmit/>
</AutoForm>
Custom form title
<AutoForm /> has a title prop that can be used to override the default form title. This is useful when you want a custom form title but want to use the auto-generated input components for a form:
<AutoInput /> renders an automatic input for the provided field, selecting the right kind of control, formatting, and experience for inputting data for a given field type.
<AutoForm/>'s automatic input selection uses <AutoInput/> under the hood, so if you want to take over and customize the layout of your form while using the same input components, you can keep using a fully automatic version of each field with <AutoInput/>.
For example, you can convert the form from the previous section to use explicit children like so:
AutoForm vs manually built AutoForm using AutoInput and AutoSubmit
The field prop for each <AutoInput/> specifies the model field to accept input for. You can also add HTML elements or layout components, such as the Polaris FormLayout component:
render an AutoForm using a @shopify/polaris FormLayout component
React
1()=>(
2<AutoFormaction={api.widget.create}>
3<FormLayout>
4<FormLayout.Group>
5<AutoInputfield="name"/>
6<AutoInputfield="inventoryCount"/>
7</FormLayout.Group>
8</FormLayout>
9<AutoSubmit/>
10</AutoForm>
11);
1()=>(
2<AutoFormaction={api.widget.create}>
3<FormLayout>
4<FormLayout.Group>
5<AutoInputfield="name"/>
6<AutoInputfield="inventoryCount"/>
7</FormLayout.Group>
8</FormLayout>
9<AutoSubmit/>
10</AutoForm>
11);
Customizing Shopify Polaris <AutoInput />
If you would like to have more control over the input's appearance, you can use the individual input types directly:
manually select the input component that is used
React
1// before
2()=>(
3<AutoFormaction={api.widget.create}>
4<AutoInputfield="name"/>
5<AutoInputfield="inventoryCount"/>
6<AutoSubmit/>
7</AutoForm>
8);
9
10// after
11()=>(
12<AutoFormaction={api.widget.create}>
13<AutoStringInputfield="name"/>
14<AutoNumberInputfield="inventoryCount"/>
15<AutoSubmit/>
16</AutoForm>
17);
1// before
2()=>(
3<AutoFormaction={api.widget.create}>
4<AutoInputfield="name"/>
5<AutoInputfield="inventoryCount"/>
6<AutoSubmit/>
7</AutoForm>
8);
9
10// after
11()=>(
12<AutoFormaction={api.widget.create}>
13<AutoStringInputfield="name"/>
14<AutoNumberInputfield="inventoryCount"/>
15<AutoSubmit/>
16</AutoForm>
17);
These inputs map 1:1 to the corresponding Polaris input and can accept any parameters that the Polaris input could:
example using Polaris params on an AutoStringInput
An existing record can be automatically fetched by passing the record id into the findBy prop. You need to make sure your current user role has read access to the model you are trying to fetch.
render a form for updating a widget with id=123
React
<AutoFormaction={api.widget.update}findBy="123"/>
<AutoFormaction={api.widget.update}findBy="123"/>
To fetch an existing record by something other than id, pass the findBy prop a set of conditions to find by:
render a form for updating a widget with slug="foobar"
When using a findBy for a field other than id, it requires a by-field record finder like .findBySlug to exist for your model, which is
generated by adding a uniqueness validation to a field.
Call custom model actions on submit
<AutoForm /> can also be used to call custom model actions. Appropriate input components will be rendered for any custom params defined for the action. These custom actions require a record to be passed in, so you must pass the findBy prop to fetch the record to run the action on.
For example, if you have a custom widget.reconfigure action that has state: string and value: number params defined, the default <AutoForm> will render with a text input for name and a number input for value.
To fetch an existing record by something other than id, pass the findBy prop a set of conditions to find by:
render a form that runs a custom action for widget record with name="foobar"
<AutoForm /> can also be used to call global actions. Appropriate input components will be rendered for any custom params defined for the action.
example of a global action called on form submit
React
<AutoFormaction={api.sendEmail}/>
<AutoFormaction={api.sendEmail}/>
Setting initial values for create forms
Forms for create actions will automatically populate with default values set for fields in your backend Gadget models.
For example, if you have set the default inventoryCount for the widget model to 0 in the model schema page, the form will start with the inventoryCount field set to 0.
If you'd like to pre-populate the form with specific values when creating new records, you can also pass the defaultValues prop to set explicit initial values for the form:
set default form values explicitly
React
1<AutoForm
2action={api.widget.create}
3defaultValues={{
4 widget:{ name:"foobar", inventoryCount:0},
5}}
6/>
1<AutoForm
2action={api.widget.create}
3defaultValues={{
4widget:{name:"foobar",inventoryCount:0},
5}}
6/>
defaultValues must be passed in the fully-qualified form that includes the model name, like widget.name, instead of just name. This wrapper object allows the form to store other state beyond just the model data, and matches your app's API format under the hood.
Validation is automatically applied to all <AutoForm/> inputs. The default validation rules are set using the field validation rules set on a model's schema.
Validating custom params
For custom parameters on model or global actions, you can manually validate params passed into the action from an <AutoForm />. If a value is invalid, you can return an InvalidRecordError with a message to display to the user.
This validation is done in your actions, not in the frontend.
Here's a sample action that validates a custom params object:
7thrownewInvalidRecordError("count param not valid",[
8{apiIdentifier:"customUpload",message:"count must be between 0 and 100"},
9]);
10}
11
12// ... custom action code
13};
14
15exportconst params ={
16count:{type:"number"},
17};
You can call this action from an <AutoForm /> and the validation error will be displayed:
calling the action with custom params
React
<AutoFormaction={api.customAction}/>
<AutoFormaction={api.customAction}/>
To learn about setting validations on fields, see the field validation guide.
<AutoForm /> does not support custom client-side validation.
Manual form control using react-hook-form
If you need to manually control the form state, you can use the react-hook-form hooks directly. This is useful if you need to integrate with form state that is not provided by <AutoForm />, or if you need to manually control the form state for some other reason.
For example, I want to use Shopify's AppBridge resourcePicker to select a product, and then save the selected product ID to the form state. I can use the useFormContext hook from react-hook-form to get access to the form state APIs and manually set the selected product ID:
5// import useFormContext to get access to react-hook-form API
6import{ useFormContext }from"react-hook-form";
7
8exportdefaultfunction(){
9return(
10<AutoFormaction={api.saveProductOffer}>
11<ResourcePicker/>
12<AutoInputfield="message"/>
13<AutoSubmit/>
14</AutoForm>
15);
16}
17
18// custom component that selects a product using an external resource picker
19constResourcePicker=()=>{
20const shopify =useAppBridge();
21
22// use setValue to manually manage form state for the selected product
23const{ setValue }=useFormContext();
24
25return(
26<Button
27onClick={async()=>{
28const selection =await shopify.resourcePicker({
29type:"product",
30multiple:false,
31action:"select",
32});
33
34// save the selected productId to the form state
35if(selection && selection.length){
36setValue("productId", selection[0].id);
37}
38}}
39>
40 Select product
41</Button>
42);
43};
Running code after form submission
The <AutoForm> also has an onSuccess callback hook that allows you to run code after the form has been successfully submitted. This can be useful for things like redirecting the user to a new page, showing a success message, or updating the UI in some way.
<AutoButton /> renders a button that when clicked calls one of your app's backend API actions with variables passed as a prop. You can use <AutoButton /> for model actions to create, update, or delete records, or to call a global action.
<AutoButton /> renders a button from the design system you're using, so it fits right in with the rest of your application and doesn't require any additional styling.
To render an <AutoButton/>, pass the action prop with the backend API action you want to call.
render a button that when clicked calls the create action
React
<AutoButtonaction={api.widget.create}/>
// when clicked, will run api.widget.create()
<AutoButtonaction={api.widget.create}/>
// when clicked, will run api.widget.create()
When clicked, this button will run the action, and show a success or error toast when it completes. It will also show a nice loading spinner while the action is running.
Often, you'll need to pass variables to your action. Pass them in the same style as you would pass them to your app's JS client:
render a button that when clicked calls the create action
<AutoButton/> accepts all the props from the design system you're using, so you can control its look and feel in the same way you would control the <Button/> component from the underlying design system.
For example, to set a Polaris <AutoButton>'s label, you can pass label as children:
<AutoButton/> has onSuccess and onError callback props that are triggered when the action succeeds or fails. This can be useful for things like redirecting the user to a new page, showing a success message, or updating the UI in some way.
For example, you can run code after a button is clicked like so:
run code after successful action
React
1()=>(
2<AutoButton
3action={api.widget.create}
4onSuccess={(result)=>{
5const{ data }= result;
6// onSuccess is passed the same object that calling an action
7// with `useAction` would return, a `{data, fetching, error}` object
You can also pass custom error handlers with onError, which will be passed an Error object describing the error:
run code after failed action
React
1()=>{
2const[error, setError]=useState(null);
3
4return(
5<AutoButton
6action={api.widget.create}
7onError={(err:{ message:string})=>{
8// onError is passed an Error object
9console.log("failed to create record:", err.message);
10setError(err.message);
11}}
12/>
13);
14};
1()=>{
2const[error, setError]=useState(null);
3
4return(
5<AutoButton
6action={api.widget.create}
7onError={(err:{message: string })=>{
8// onError is passed an Error object
9console.log("failed to create record:", err.message);
10setError(err.message);
11}}
12/>
13);
14};
When onSuccess or onError is passed, the default behavior of rendering a toast in the selected design system is overridden. If you want to maintain that behavior, you need to trigger it yourself.
For example, if you want to show a Shopify Polaris toast after the button is clicked, you can call the toast function:
React
1// Show a toast after the button is clicked
2()=>(
3<AutoButton
4action={api.widget.create}
5onSuccess={()=>{
6 shopify.toast.show(`New widget created.`);
7}}
8/>
9);
1// Show a toast after the button is clicked
2()=>(
3<AutoButton
4action={api.widget.create}
5onSuccess={()=>{
6 shopify.toast.show(`New widget created.`);
7}}
8/>
9);
<AutoTable />
<AutoTable /> components generate data tables for your React frontends and are pre-configured to read and display data from your data models.
<AutoTable /> reads each record, and renders the right kind of cell. This autocomponent supports pagination, filtering, searching, and sorting out of the box.
simple AutoTable example: build a data table for a customer model
Using <AutoTable /> without specifying the included columns will render a table with all fields from the
model. This means that adding a new field to a model will also add a column to the table.
Make sure that you test any default <AutoTable /> components after making changes to your models to ensure they still work as expected.
Import AutoTable
Import Shopify Polaris <AutoTable> components from @gadgetinc/react/auto/polaris:
import the AutoTable component into your frontend files
AutoTable uses the useFindMany hook to fetch data from the backend under the hood, so AutoTable has the same filter prop as useFindMany. The sort prop from useFindMany is also available, but has been renamed to initialSort since users could choose a different column to sort by after the table is rendered.
AutoTable provides a search bar that filters records based on the query. It uses the same full-text search feature found in the API client. It is enabled by default, but can be disabled by passing searchable={false} to the AutoTable component.
AutoTable supports params from the underlying components in the design system you're using, so you can control its look and feel in the same way you would when building with the system's native table component.
You can specify which columns to display. Pass an array of their API identifiers to the columns prop. The columns are displayed in the order in which they are listed.
Related fields (fields that are of type has one, belongs to, or has many) are not rendered by default. You can render them using dot notation in the columns prop, using the same syntax as your app's GraphQL queries.
For has one or belongs to relationships, you can directly access fields on the related model:
show email of the customer that the purchase belongs to
11"customer.email",// for "has one" or "belongs to" relationships
12]}
13/>
14);
15};
PurchaseTable renders like so:
For example, CustomerTable displays the records associated with the customer model. Each customerhas manypurchases. Because GraphQL query syntax is used, edges.node goes between the related model name and the field name, like so: purchases.edges.node.purchaseLocation.
show all purchase locations associated with each customer
By default, columns for field types that are able to be sorted alphanumerically will be sortable in an AutoTable. A list of sortable fields can be found in your API documentation.
If you want to manually set columns that are sortable in an AutoTable you can use the sortable boolean in a custom column configuration:
13// code will be sortable if the field type is sortable
14"code",
15]}
16/>
17);
18};
Realtime query support for <AutoTable />
By default, an AutoTable does not re-render when a record is created/updated/deleted on the backend.
If you want to keep your frontend data table up to date with the data in your database, AutoTable components support realtime queries. Anytime a record is updated in the model powering the AutoTable, the AutoTable will automatically re-render to reflect the changes.
You can enable realtime query support on your AutoTable with the live prop:
React
// This component will rerender every time a customer is added, removed or updated
<AutoTablemodel={api.customer}live/>
// This component will rerender every time a customer is added, removed or updated
<AutoTablemodel={api.customer}live/>
Custom onClick
The onClick prop is a callback that is fired when the row is clicked. Including onClick overrides the default click behavior, which is to select the row.
Note that even when onClick is included as a prop, rows can still be selected by pressing directly on the checkbox:
The row parameter is an object that describes a row the user selected. Say I have a row with columns "First name", "Last name" and "Email", (row) => console.log(row) prints this object to the console:
You can trigger bulk actions from AutoTable out of the box. Upon selecting a row, a bulk action can be run by selecting the three dots, then selecting the name of the bulk action. The action is run for each selected record in the table.
Users can trigger actions that have delete and customaction types. AutoTable cannot trigger create and update action types. Instead, use AutoForm to create or update a record.
By default, all delete and custom actions are included in the table. You can include actions by passing their action API identifiers:
React
// myCustomAction is the API identifier of the action
<AutoTableactions={["myCustomAction"]}/>
// myCustomAction is the API identifier of the action
<AutoTableactions={["myCustomAction"]}/>
Actions can also be excluded:
React
// myOtherCustomAction is the API identifier of the action