Product teams should be thinking differently about documents.Read blog post.
Anvil Logo

Embedding workflow webforms in your app

Author headshot
By Ben Ogle

Learn how to embed your Workflow webforms in an iframe: collect user information, fill PDFs, and gather e-signatures, all from within your app.

Back to all articles
Post hero image

A central feature of Anvil is Workflows. Workflows allow you to collect data from your users via a simple webform, automatically fill PDFs with the user's data, then gather signatures from relevant people.

For example, say we wanted our client to sign an NDA. We can give our client an easy webform to fill:

Workflow webform nda example

When the client is finished filling the webform, it will fill the NDA with their information, then ask them to sign:

embedded Workflow e signature

You can embed this entire process in an iframe on your app or website. That way, you control the experience; your user does not need to leave your site to fill and sign our hypothetical NDA.

In this post, I'll show you how to

  • Embed the webform filling and signature UIs in your app or website
  • Start Workflows
  • Have your app notified when actions take place. For example, when the user submits a page, or completes the Workflow
  • Download the documents when the user is finished

Embedding the form in an iframe

First, sign up for Anvil, then enable embedding for your app. Once enabled, embedding the webform UI in an iframe is easy. All you need to do is point the src attribute at your Workflow's webform URL.


And you will see the form inside your own app:

Workflow webform nda example

Starting a Workflow

There are two ways to start a Workflow submission for one of your users:

  1. Embedding a special URL to kick off a new submission
  2. Creating a Workflow submission via the API

Option 1: Embed a special URL

Embedding a form URL without an ID will create a new Workflow submission (a WeldData). This approach is the easiest way to get a Workflow submission started and seeded with your own data.

In our case, our iframe URL would look like this:

  src='{"name":"Sally Jones"}'

Note the d param, it will seed the new Workflow submission with data from your system. See the Workflow UI URLs docs for all the options available.

Option 2: Create a Workflow submission via the API

An alternative to creating a Workflow submission with a special URL is starting the Workflow with the GraphQL API, then embedding the resultant webform URL. This approach is a little more complex, but gives you full control over the submission.

Often customers with complex Workflows use the API method to start Workflows because it allows for more flexibility. With the API method, you can specify a specific webhookURL for the Workflow submission, seed multiple webforms in a Workflow, and more.

To create a Workflow submission, use the forgeSubmit mutation. Our node API client supports the forgeSubmit mutation out of the box:

// Uses the Anvil client, see
const result = await anvilClient.forgeSubmit({
  variables: {
    forgeEid: forge.eid, // your webform's eid
    complete: false,
    isTest: false,
    // Seed the submission with data here...
    payload: {
      name: Sally Jones,

// You will embed the resulting `weldData.continueURL` in the iframe
const responseData =
const embedURL = responseData.weldData.continueURL

Then in the HTML, you would use embedURL in the iframe's src attribute:


See the forgeSubmit docs for more information on using forgeSubmit. The basic mutation call used above to start a Workflow is as follows:

mutation ForgeSubmit(
  $forgeEid: String!,
  $weldDataEid: String,
  $submissionEid: String,
  $payload: JSON!,
  $complete: Boolean,
  $isTest: Boolean,
  $webhookURL: String,
) {
  forgeSubmit (
    forgeEid: $forgeEid,
    weldDataEid: $weldDataEid,
    submissionEid: $submissionEid,
    payload: $payload,
    complete: $complete,
    isTest: $isTest,
    webhookURL: $webhookURL,
  ) {
    weldData {

Anvil React components

We've created an AnvilEmbedFrame React component to aid embedding Anvil into your app.

import AnvilEmbedFrame from '@anvilco/anvil-embed-frame'

  onEvent={(event) => console.log('Event object:', event)}

Getting notified

Your app can be notified when a user takes an action in two ways:

  1. JavaScript events from the iframe to the parent frame
  2. Webhook notifications

Events from the iframe

For some actions, the iframe will emit events that your parent frame can capture. These events will help you know when the user is finished filling a page, finished with the webform, and finished signing.

On the page that renders the iframe element, you can subscribe to the window's message event. The events emitted by the iframe should provide you with all ids you need to query our system for the user's submitted data.

window.addEventListener('message', ({ origin, data: eventData }) => {
  if (origin !== '') return
  // eventData will be an object in one of the formats
  // {
  //   action: 'forgeSubmitPage',
  //   organizationSlug: 'my-org'
  //   weldDataEid: 'bdwIX5TUiLBzldgdwzL2',
  //   forgeEid: 'wIX5TUiLBzldgdwzL2bd',
  //   submissionEid: 'X5TUiLBzldgdwzL2bdwI',
  //   pageId: 'somePageId',
  // }
  // OR
  // {
  //   action: 'forgeComplete',
  //   organizationSlug: 'my-org'
  //   weldDataEid: 'bdwIX5TUiLBzldgdwzL2',
  //   forgeEid: 'wIX5TUiLBzldgdwzL2bd',
  //   submissionEid: 'X5TUiLBzldgdwzL2bdwI',
  // }

Events from AnvilEmbedFrame

The AnvilEmbedFrame React component will emit the events described above in the component's onEvent handler:

import AnvilEmbedFrame from '@anvilco/anvil-embed-frame'

  onEvent={(eventData) => {
    // eventData is the same as described above!
    // {
    //   action: 'forgeSubmitPage', 'forgeComplete', etc.,
    //   ...others,
    // }

Webhook notifications

On the server side, you can receive webhook notifications for several actions—Anvil will POST to an endpoint on your server. You can set up a global webhook URL in your organization's settings, or you can set a per-object webhook URL for each Workflow submission. In either case, your webhook URLs will receive the same event payloads.

The most important actions for Workflow embedding are:

  • forgeComplete - A user completes a webform
  • signerComplete - A signer finishes signing all of their documents
  • weldComplete - The Workflow submission is completely finished, all webforms have been completed, and all signers have finished signing

See the webhooks guide for more information on handling webhooks.

Downloading the documents

When all forms have been filled, and all signers have signed, it's time to download the documents. Download them in zip format with the following URL:

If you are using one of our API clients, this is easy with the downloadDocuments() function. Here's an example using our NodeJS client:

const weldDataEid =
const zipResponse = await anvilClient.downloadDocuments(weldDataEid)

See the full details of downloading documents in the Workflow API docs.

All done!

That's it! Now you have all the tools to embed a Workflow's form in your app or website, get notified when something happens, and download the finished documents.

Note that if you are already collecting your client's information with your own webforms, you can use Etch e-sign to fill PDFs with your data, create signature packets, and embed the e-signing UI in your app, all in one flow.

Further reading

If you have questions please do not hesitate to contact us at:

Sign up for a live demo

Request a 30-minute live demo today and we'll get in touch shortly. During the meeting our Sales team will help you find the right solution, including:
  • Simplifying data gathering
  • Streamlining document preparation
  • Requesting e-signatures
  • Building and scaling your business
Want to try Anvil first?Sign up for free
Want to try Anvil first?Sign up for free