5.3.0 • Published 10 months ago

@typeform/embed v5.3.0

Weekly downloads
48,758
License
MIT
Repository
github
Last release
10 months ago

🛠 Typeform Vanilla Embed Library

Typeform/embed is the core embed library that lets you embed typeforms to your website using vanilla JavaScript.

Installation

As NPM package

Requirements:

  • node >= 18
  • yarn or npm

Install using your favourite package manager:

yarn add @typeform/embed

or

npm install --save @typeform/embed

Import the lib, CSS and create your embed:

import { createWidget } from '@typeform/embed'
import '@typeform/embed/build/css/widget.css'
createWidget('<form-id>', { container: document.querySelector('#form') })

From CDN

As HTML, the CSS is imported automatically. Place this code where you want to display your form.

<div data-tf-widget="<form-id>"></div>
<script src="//embed.typeform.com/next/embed.js"></script>

Via JavaScript for more control and specific integration.

<button id="button">open form</button>
<script src="//embed.typeform.com/next/embed.js"></script>
<link rel="stylesheet" href="//embed.typeform.com/next/css/popup.css" />
<script>
  const { open, close, toggle, refresh } = window.tf.createPopup('<form-id>')
  document.querySelector('#button').onclick = toggle
</script>

How to get form id of your form?

You can find <form-id> from the public URL of your form:

  • https://form.typeform.com/to/<form-id>

Or from admin panel URL:

  • https://admin.typeform.com/form/<form-id>/*

Limitations

For security purposes we prevent embedding typeorms in unsecure pages (via CSP headers). You can embed your typeform on pages served over HTTPS or via HTTP on localhost. You can also embed in wrapped progressive web apps.

Configuration

Embed types

Widget

<div id="form"></div>
<script>
  const { refresh, unmount } = createWidget('<form-id>', {
    container: document.querySelector('#form'),
    ...options,
  })
</script>

The createWidget method returns 2 functions:

  • refresh - reloads the form
  • unmount - unmounts the form (you should use this when you implement this lib in React manually)

Modal windows: popup, slider, sidetab, popover

  • popup: createPopup('<form-id>', options)
  • slider: createSlider('<form-id>', options)
  • sidetab: createSidetab('<form-id>', options)
  • popover: createPopover('<form-id>', options)
<button id="button">open form</button>
<script>
  const { open, close, toggle, refresh } = createPopup('<form-id>')
  document.querySelector('#button').onclick = toggle
</script>

Each of the create* methods for modal windows return 4 functions:

  • open - open the modal window (popup, slider, sidetab or popover) and display the form
  • close - close the modal window and hide the form
  • toggle - open when closed, close when opened
  • refresh - reloads the form
  • unmount - unmounts the form (you should use this when you implement this lib in React manually)

Closing and opening a typeform in modal window will restart the progress from the beginning. However answers will be saved in browsers local storage.

Options

options is an object with optional properties:

nametypedescriptiondefault
containerHTMLElementspecify element to place the embed into, only for widget, requiredcurrent element when embedding as HTML, otherwise undefined
positionstringslider position: right or leftright
sizenumbersize of the popup in percentage (desktop only, opens in fullscreen on mobile devices)100 (100% size, fullscreen popup)
widthnumber / stringwidth of the embed - number in pixels or string including units (for popup you can specify size instead)undefined
heightnumber / stringheight of the embed - number in pixels or string including units, supported by all embeds except slider (for popup you can specify size instead)undefined
hiddenobjecthidden fields to be passed to the form in URL hashundefined
trackingobjecttracking parameters to be passed to the form in URL query stringundefined
sourcestringdomain name of the site using the SDKdomain name from window.location
mediumstringname of the plugin built on top of the SDK"embed-sdk"
mediumVersionstringversion of the plugin built on top of the SDK"next"
transitiveSearchParamsstring[] / booleansearch parameters to be forwarded from host page to form, if true will forward all host page search parametersundefined
hideFooterbooleanhide form progress bar and navigation buttonsfalse
hideHeadersbooleanhide header that appears when you have a question group, or a long questionfalse
domainstringdomain name of the environment the SDK should run against"https://form.typeform.com"
opacitynumberform background opacity, number from 0 (fully transparent) 100 (fully opaque)100
autoFocusbooleanenable form auto focus when loadedfalse
openstringopen embed based on user action (see below)undefined
openValuenumberbased on open (see below)undefined
preventReopenOnClosebooleanprevent automatically re-opening the typeformfalse
enableSandboxbooleanenable sandbox mode (disables submissions and tracking, the responseId in callbacks will have value of "__sandbox")false
buttonTextstringcustomize the button text (sidetab only)"Launch me"
buttonColorstringcustomize the button background color (sidetab only)#3a7685
buttonTextColorstringcustomize the button text color (sidetab only)white or black (based on background color)
buttonTextSizeinteger / stringcustomize the button text size - number in pixels or string including units (sidetab only)24px
buttonWidthinteger / stringcustomize the button width - number in pixels or string including units (sidetab only)auto
buttonHeightinteger / stringcustomize the button height - number in pixels or string including units (sidetab only)48px
buttonAligntop / center / bottomcustomize button position relative to form iframe (sidetab only)center
topinteger / stringcustomize iframe position from top - number in pixels or string including units (sidetab only)50%
bottominteger / stringcustomize iframe position from bottom - number in pixels or string including units, can be used only when top is not set (sidetab only)undefined
customIconstringcustomize the message icon (popover, sidetab) more infoundefined
tooltipstringdisplay tooltip text next to the button (popover only)undefined
notificationDaysnumberdisplay red notification dot, hide for given number of days since popover is open (popover only)undefined
autoClosenumber / booleantime (ms) until the embedded typeform will automatically close after a respondent clicks the Submit button. (all embeds except widget)undefined
onReadyfunctionfires when the form is loadedundefined
onStartedfunctionfires on the "submission start" event, contains responseId in the payloadundefined
onSubmitfunctionfires when user submits the formundefined
onClosefunctionfires when the form is closed (when opened in modal window)undefined
onQuestionChangedfunctionfires when user navigates between form questionsundefined
onHeightChangedfunctionfires when form question height changes (eg. on navigation between questions or on error message)undefined
onEndingButtonClickfunctionfires when button on ending screen is clicked, disables button redirect functionalityundefined
onDuplicateDetectedfunctionfires when the respondent reaches the quota of responses defined in the duplicate prevention settingundefined
autoResizestring / booleanresize form to always fit the displayed question height, avoid scrollbars in the form (inline widget only), set min and max height separated by coma, eg. "200,600"false
shareGaInstancestring / booleanshares Google Analytics instance of the host page with embedded typeform, you can provide your Google Analytics ID to specify which instance to share (if you have more than one in your page)false
inlineOnMobilebooleanremoves placeholder welcome screen in mobile and makes form show inline instead of fullscreenfalse
iframePropsobjectHTML attributes to be passed directly to the iframe with typeformundefined
buttonPropsobjectHTML attributes to be passed directly to the button created by embed SDK (only for popover and sidetab)undefined
lazybooleanenable lazy loading (for widget only), typeform starts loading when user scrolls to it, see demofalse
keepSessionbooleanpreserve form state when modal window is closed (and re-opened)false
redirectTargetstringtarget for typeforms with redirect, valid values are _self, _top, _blank or _parent (see docs on anchor target)_parent
disableScrollbooleandisable navigation between questions via scrolling and swipingfalse
hubspotbooleanenable HubSpot source tracking - for details see article Set up source tracking for HubSpotfalse
fullScreenbooleanenable full screen view, set <body> size, resize on screen resize - also when browser navbars are displayed on mobilefalse
preselectobjectpreselect answer to the first question (more info in help center)undefined
respectOpenModalsall / samedo not open if there already is a modal with typeform open (same - same form, all - any form)undefined
noScrollbarsbooleando not render scrollbars — useful when used along autoResizeundefined

Options in plain HTML embed

  • to embed via HTML without writing JavaScript code, use data-tf-widget="<form-id>" for widget embed (see example above)
  • define options as data attributes with data-tf- prefix and dashes in name (eg. autoFocus becomes data-tf-auto-focus)
  • set a boolean property to true by omitting attribute value, (eg. <div ... data-tf-disable-footer></div>
  • pass function name for callbacks, eg. data-tf-on-ready="myReadyFunction" if this function is available on global scope (eg. window)
  • to pass string[] use comma-separated string, eg. transitiveSearchParams: ['foo', 'bar'] becomes data-tf-transitive-search-params="foo,bar"
  • to pass object pass comma-separated key=value pairs, eg. hidden: { foo: "f", bar: "b" } becomes data-tf-hidden="foo=f,bar=b"
    • Note: since commas , are used as delimiter for each value you will need to escape them with backward slash, eg. data-tf-hidden="foo=foo\,bar". In JavaScript you don't need to escape it.

Custom Launch Options

Properties open and openValue apply only to embed types that are opened by user action (all except widget). They define when to automatically open the typeform.

  • on page load
    • open: 'load'
    • openValue leave undefined (not used)
  • when user tries to leave the page
    • open: 'exit'
    • openValue specify the sensitivity threshold
    • To detect user is trying to exit the page we detect upwards mouse movement in top part of the website. The threshold defines height of this area. Useful when you have navigation in top part of your website and mouse movement in that area does not necessarily indicate exit intent.
  • when a user scrolls the page
    • open: 'scroll'
    • openValue percentage of page scrolled (0 - 100) to open the form
  • after time elapsed
    • open: 'time'
    • openValue number of milliseconds to wait before opening the form

For details see behavioral demo.

Share Google Analytics Instance

You can use shareGaInstance: true (or data-tf-share-ga-instance) attribute if both your page and your typeform are using Google Analytics. This will make sure the session is shared and Google Analytics will track only 1 user when they visit you page with an embedded typeform.

If you have more than 1 Google Analytics tracking codes in your website you can provide an ID to specify which tracker to use, eg:

<div data-tf-widget="<form-id>" data-tf-share-ga-instance="UA-XXXXXX-XX"></div>

or

createPopup('<form-id>', { container, shareGaInstance: 'UA-XXXXXX-XX' })

Callbacks

You can listen to form events by providing callback methods:

<button id="btn">click</button>
<script src="//embed.typeform.com/next/embed.js"></script>
<link rel="stylesheet" href="//embed.typeform.com/next/css/widget.css" />
<script>
  const { open } = window.tf.createPopup('<form-id>', {
    onReady: ({ formId, isClosed }) => {
      console.log(`Form ${formId} is ready. ${isClosed ? 'It is closed for new responses.' : 'It is open to new responses.'}`)
    },
    onStarted: ({ formId, responseId }) => {
      console.log(`Form ${formId} started with response ID ${responseId}`)
    },
    onQuestionChanged: ({ formId, ref }) => {
      console.log(`Question in form ${formId} changed to ${ref}`)
    },
    onHeightChanged: ({ formId, ref, height }) => {
      console.log(`Question ${ref} in form ${formId} has height ${height}px now`)
    },
    onSubmit: ({ formId, responseId }) => {
      console.log(`Form ${formId} submitted, response id: ${responseId}`)
      // to retrieve the response use `responseId` (you have to do it server-side)
      // more details: https://developer.typeform.com/responses/
    },
    onClose: ({ formId }) => {
      console.log(`Modal window with form ${formId} was closed`)
    }
    onEndingButtonClick: ({ formId, ref }) => {
      console.log(`Ending button clicked in form ${formId}`)

      // for plans with "Redirect from ending screen" feature you also receive `ref`:
      console.log(`Ending button clicked in end screen ${ref}`)
    },
    onDuplicateDetected: ({ formId }) => {
      console.log(`User reached the quote of responses for form ${formId}`)
    }
  })
  document.querySelector('#btn').click = () => {
    open()
  }
</script>

Callback method receive payload object from the form. Each payload contains form ID to identify which form sent the event (see chaining typeforms below):

  • onReady
  • onStarted
    • formId (string)
    • responseId (string)
  • onQuestionChanged
    • formId (string)
    • ref (string) identifies currently displayed question
  • onHeightChanged
    • formId (string)
    • ref (string) identifies currently displayed question
    • height (number) current height of currently displayed question
  • onSubmit
    • formId (string)
    • responseId (string) identifies the response, can be retrieved via Responses API
  • onClose
    • no payload
  • onEndingButtonClick
    • formId (string)
    • ref (string) identifies the end screen (Note: this is available for plans with "Redirect from ending screen" feature only.)
  • onDuplicateDetected
    • formId (string)

See callbacks example in demo package.

Custom icon

Custom icon provided string supports:

  • URL (used as an img src)
  • Text and Emojis
  • HTML Markup

Redirect Target

You can supply a target for typeform redirect (on submit or via ending). It works the same as target for anchor HTML element:

  • _parent (default), opens in parent page
  • _self opens in the same embedded iframe as your typeform
  • _top opens in current browser tab (same as _parent unless there are multiple nested iframes)
  • _blank opens in new tab, however it might be blocked by popup blockers.

⚠️ Warning: Target _blank is not working in Safari (both desktop and mobile) and triggers a popup warning in Chrome on Android. It works in Chrome and Firefox on desktop.

If you set target to _self and also enable autoClose option the iframe with your redirect will be closed before users are able to interact with it.

Positioning and overlapping

All embeds that are intended to be displayed over existing content in the website have z-index set to 10001. If you want to display content over your typeform you need to make sure it has higher z-index. However if you want your typeform to display over other content in your website you need to set its z-index to a value of 10000 or lower.

This is related to all embeds:

  • popup
  • slider
  • sidetab
  • popover
  • widget - on mobile devices widget opens in fullscreen modal window (unless inlineOnMobile is set)

Chaining typeforms

You can chain multiple typeforms inside an embed. You need to setup a redirect to another typeform:

  • make sure to use URL with typeform.com domain in case you have a custom domain set up
  • set redirectTarget / data-tf-redirect-target to _self to make the redirect inside the embed iframe

When you chain multiple typeforms they will be all displayed inside the embed and all embed options and callbacks will be preserved. You can use formId in the callback payload to identify which form is currently displayed.

Loading and reloading embedded forms

When the library loads it will initialize all HTML embed codes already present in the page. However sometimes you might want to add HTML snippet to your page later and initialize it after it was added.

To load new snippets use:

window.tf.load()

If you need to reload all snippets in the page:

window.tf.reload()

You can see an example of this in reload-event.html.

Examples

You can find examples for specific use-cases in our demos:

Local setup and development

Fork and clone this Github repo: https://github.com/Typeform/embed

Requirements:

  • node >= 18
  • yarn

Install dependencies:

yarn

We recommend you work in a branch:

git checkout -b cool-new-feature

Build, watch for changes and start a demo server too (using demo-nextjs)

yarn demo

Build and watch for changes:

yarn dev

Run unit tests:

yarn test

Run functional tests via Cypress:

yarn cy:run   # run in background (headless)
yarn cy:open  # open cypress UI

Run visual tests via Cypress and VRT:

yarn cy:visual    # run in background (headless)
yarn cy:open:vrt  # open cypress UI (with support for VRT)

Note: You need access to our self-hosted Visual Regression Tracker (aka VRT). Copy vrt.example.json to vrt.json and provide apiKey to run visual tests locally.

See details on contributing to this repo.

5.3.0

10 months ago

5.2.0

11 months ago

5.1.0

1 year ago

5.0.0

1 year ago

4.10.1

1 year ago

4.10.2

1 year ago

4.10.3

1 year ago

4.10.0

2 years ago

4.9.0

2 years ago

4.8.1

2 years ago

4.8.0

2 years ago

4.7.0

2 years ago

4.6.0

2 years ago

4.5.0

2 years ago

4.4.1

2 years ago

4.4.0

2 years ago

4.0.0

2 years ago

3.0.1

2 years ago

3.0.0

2 years ago

2.16.1

2 years ago

2.16.0

2 years ago

2.15.0

2 years ago

2.15.1

2 years ago

2.14.0

2 years ago

4.3.2

2 years ago

4.3.1

2 years ago

4.3.0

2 years ago

2.13.2

2 years ago

2.13.1

2 years ago

4.2.0

2 years ago

4.1.0

2 years ago

4.1.1

2 years ago

2.13.0

2 years ago

2.12.1

2 years ago

2.11.0

2 years ago

2.12.0

2 years ago

2.10.0

3 years ago

2.9.2

3 years ago

2.9.1

3 years ago

2.9.0

3 years ago

2.8.0

3 years ago

2.4.1

3 years ago

2.7.0

3 years ago

2.7.1

3 years ago

2.6.1

3 years ago

2.6.0

3 years ago

2.5.0

3 years ago

2.5.1

3 years ago

2.4.0

3 years ago

2.3.0

3 years ago

2.3.1

3 years ago

2.2.0

3 years ago

2.1.0

3 years ago

2.0.1

3 years ago

2.0.0

3 years ago

1.37.1

3 years ago

1.38.0

3 years ago

1.36.1

3 years ago

1.37.0

3 years ago

1.36.0

3 years ago

1.34.2

4 years ago

1.35.1

4 years ago

1.35.0

4 years ago

1.34.1

4 years ago

1.34.0

4 years ago

1.33.0

4 years ago

1.32.2

4 years ago

1.29.0

4 years ago

1.32.0

4 years ago

1.32.1

4 years ago

1.27.2

4 years ago

1.27.0

4 years ago

1.27.1

4 years ago

1.30.0

4 years ago

1.28.1

4 years ago

1.28.2

4 years ago

1.28.0

4 years ago

1.31.1

4 years ago

1.31.0

4 years ago

1.22.0

4 years ago

1.22.1

4 years ago

1.23.0

4 years ago

1.23.1

4 years ago

1.24.1

4 years ago

1.24.0

4 years ago

1.25.0

4 years ago

1.26.0

4 years ago

1.26.3

4 years ago

1.26.1

4 years ago

1.26.2

4 years ago

1.21.0

4 years ago

1.21.1

4 years ago

1.20.2

4 years ago

1.20.1

4 years ago

1.20.0

4 years ago

1.19.1

4 years ago

1.19.0

4 years ago

1.18.1

4 years ago

1.18.0

4 years ago

1.17.10

4 years ago

1.17.9

4 years ago

1.17.8

4 years ago

1.17.7

4 years ago

0.36.4

4 years ago

0.36.5

4 years ago

1.17.6

4 years ago

1.17.5

4 years ago

1.17.4

4 years ago

1.17.3

4 years ago

1.17.2

4 years ago

1.17.1

4 years ago

1.16.1

4 years ago

1.16.0

4 years ago

1.17.0

4 years ago

1.15.2

4 years ago

1.15.0

4 years ago

1.15.1

4 years ago

1.14.0

4 years ago

1.12.1

4 years ago

1.13.0

4 years ago

0.36.3

4 years ago

1.12.0

4 years ago

1.11.0

4 years ago

0.36.2

4 years ago

0.36.1

4 years ago

1.10.1

4 years ago

1.10.0

4 years ago

1.9.0

4 years ago

1.8.2

4 years ago

1.8.1

4 years ago

1.8.0

4 years ago

1.9.2

4 years ago

1.7.0

4 years ago

1.2.0

4 years ago

1.6.1

4 years ago

1.6.0

4 years ago

1.2.1

4 years ago

0.36.0

4 years ago

1.1.6

5 years ago

1.5.1

4 years ago

1.5.0

4 years ago

1.4.0

4 years ago

1.3.0

4 years ago

0.35.3

5 years ago

1.1.5

5 years ago

0.35.2

5 years ago

1.1.4

5 years ago

1.1.3

5 years ago

1.1.2

5 years ago

0.35.1

5 years ago

0.35.0

5 years ago

1.1.1

5 years ago

1.1.0

5 years ago

1.0.3

5 years ago

1.0.2

5 years ago

0.34.8

5 years ago

1.0.1

5 years ago

1.0.0

5 years ago

0.34.7

5 years ago

1.0.0-next.16

5 years ago

1.0.0-next.15

5 years ago

1.0.0-next.14

5 years ago

1.0.0-next.13

5 years ago

1.0.0-next.12

5 years ago

1.0.0-next.11

5 years ago

1.0.0-next.10

5 years ago

1.0.0-next.8

5 years ago

1.0.0-next.9

5 years ago

1.0.0-next.7

5 years ago

1.0.0-next.5

5 years ago

1.0.0-next.6

5 years ago

1.0.0-next.4

5 years ago

1.0.0-next.2

5 years ago

1.0.0-next.3

5 years ago

0.34.6

5 years ago

0.34.5

5 years ago

0.34.4

5 years ago

0.34.3

5 years ago

0.34.2

5 years ago

0.34.1

5 years ago

0.34.0

5 years ago

0.33.2

5 years ago

0.32.0

5 years ago

0.31.1

5 years ago

0.31.0

5 years ago

0.30.0

5 years ago

0.29.1

5 years ago

0.29.0

5 years ago

0.28.0

5 years ago

0.27.1

5 years ago

0.26.0

5 years ago

0.27.0

5 years ago

0.25.6

5 years ago

0.25.5

5 years ago

0.25.4

5 years ago

0.25.3

5 years ago

0.25.2

5 years ago

0.25.1

5 years ago

0.25.0

5 years ago

0.24.4

5 years ago

0.24.3

5 years ago

0.24.2

5 years ago

0.24.1

5 years ago

0.24.0

5 years ago

0.23.2

5 years ago

0.23.1

5 years ago

0.23.0

5 years ago

0.22.2

5 years ago

0.22.1

5 years ago

0.22.0

5 years ago

0.21.4

5 years ago

0.21.1

5 years ago

0.21.0

5 years ago

0.20.0

5 years ago

0.19.0

5 years ago

0.18.0

5 years ago

0.17.0

5 years ago

0.16.0

6 years ago

0.15.3

6 years ago

0.15.2

6 years ago

0.15.1

6 years ago

0.15.0

6 years ago

0.14.1

6 years ago

0.14.0

6 years ago

0.13.1

6 years ago

0.13.0

6 years ago

0.12.2

6 years ago

0.12.1

6 years ago

0.12.0

6 years ago

0.11.0

6 years ago

0.10.0

6 years ago

0.9.5

6 years ago

0.9.4

6 years ago

0.9.3

6 years ago

0.9.2

6 years ago

0.9.1

6 years ago

0.9.0

6 years ago

0.8.1

7 years ago

0.8.0

7 years ago

0.7.1

7 years ago

0.7.0

7 years ago

0.6.1

7 years ago

0.5.12

7 years ago

0.5.11

7 years ago

0.5.10

7 years ago

0.5.9

7 years ago

0.5.8

8 years ago

0.5.7

8 years ago

0.5.6

8 years ago

0.5.5

8 years ago

0.5.4

8 years ago

0.5.3

8 years ago

0.5.2

8 years ago

0.5.1

8 years ago

0.5.0

8 years ago

0.0.0

8 years ago

0.4.2

8 years ago

0.4.1

8 years ago

0.4.0

8 years ago

0.1.7

8 years ago

0.1.6

8 years ago

0.1.5

8 years ago

0.1.4

8 years ago

0.1.3

8 years ago

0.1.0

8 years ago

0.0.1-28

9 years ago

0.0.1-27

9 years ago

0.0.1-26

9 years ago

0.0.1-25

9 years ago

0.0.1-24

9 years ago

0.0.1-23

9 years ago

0.0.1-22

9 years ago

0.0.1-21

9 years ago

0.0.1-20

9 years ago

0.0.1-19

9 years ago

0.0.1-18

9 years ago

0.0.1-17

9 years ago

0.0.1-16

9 years ago

0.0.1-15

9 years ago

0.0.1-14

9 years ago

0.0.1-13

9 years ago

0.0.1-12

9 years ago

0.0.1-11

9 years ago

0.0.1-10

9 years ago

0.0.1-9

9 years ago

0.0.1-8

9 years ago

0.0.1-7

9 years ago

0.0.1-6

9 years ago

0.0.1-5

9 years ago

0.0.1-4

9 years ago

0.0.1-2

9 years ago

0.0.1-1

9 years ago

0.0.1-0

9 years ago