Compare commits
150 Commits
@formbrick
...
integratio
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ec051c0de0 | ||
|
|
5679c38029 | ||
|
|
a815270784 | ||
|
|
2d97e9c797 | ||
|
|
5c90862137 | ||
|
|
206926a0a9 | ||
|
|
4e0fe7e6fb | ||
|
|
9230ce558f | ||
|
|
78d8a0604f | ||
|
|
e05cfaba5f | ||
|
|
0d74921233 | ||
|
|
60c7713aa0 | ||
|
|
c62f041819 | ||
|
|
7782196822 | ||
|
|
f964319ddb | ||
|
|
e0c17407e3 | ||
|
|
7f25bbc008 | ||
|
|
ae530d710b | ||
|
|
08bdc7208e | ||
|
|
224ba2ea22 | ||
|
|
8a4ceae38c | ||
|
|
d91e1cc7ea | ||
|
|
8896cbcd87 | ||
|
|
5e1822c9e6 | ||
|
|
1b69560e50 | ||
|
|
b28a4e72d8 | ||
|
|
8694c371af | ||
|
|
359da760f7 | ||
|
|
044080fee9 | ||
|
|
8a7d498a26 | ||
|
|
82f916d86b | ||
|
|
6ac48a26bb | ||
|
|
ab22c0297e | ||
|
|
1ce02edc1b | ||
|
|
d36de1e54f | ||
|
|
15c91b798d | ||
|
|
12d995465a | ||
|
|
f9861cf772 | ||
|
|
8857c971d6 | ||
|
|
7ea79df145 | ||
|
|
ac12ddaafb | ||
|
|
9dc47c6a27 | ||
|
|
e0e4a637e2 | ||
|
|
b9259116e1 | ||
|
|
83ffd7a371 | ||
|
|
b3d772c463 | ||
|
|
12d600093e | ||
|
|
51cec0184f | ||
|
|
4c85fcb3cd | ||
|
|
782b3e0974 | ||
|
|
fee2517009 | ||
|
|
2e16e046f1 | ||
|
|
b275cce7ad | ||
|
|
45f02fd3c2 | ||
|
|
368df47035 | ||
|
|
2ce759c023 | ||
|
|
ddc06b19bf | ||
|
|
17410ba14c | ||
|
|
9b34833bdd | ||
|
|
89c614fafb | ||
|
|
6f552886d0 | ||
|
|
6a7b66aaaa | ||
|
|
f28bb9b82a | ||
|
|
8dd67ec484 | ||
|
|
aa43d0a94c | ||
|
|
d648762f4f | ||
|
|
f8c0021346 | ||
|
|
8fd78bc08f | ||
|
|
95ed9b87de | ||
|
|
ab5f18d2c0 | ||
|
|
cd4b6fdae0 | ||
|
|
b11a7cc3ec | ||
|
|
2e5ed00414 | ||
|
|
557e912309 | ||
|
|
e2aba0cd4a | ||
|
|
7c3c6652d4 | ||
|
|
cbf11de352 | ||
|
|
72f7946bcc | ||
|
|
413a3a92cb | ||
|
|
ee8edbd547 | ||
|
|
663fa0124f | ||
|
|
81234c4bde | ||
|
|
3103760611 | ||
|
|
d8b6b95ed5 | ||
|
|
acc6674ec5 | ||
|
|
dd0d296c6a | ||
|
|
f5110fe9c1 | ||
|
|
8244a5fa48 | ||
|
|
59936e54a0 | ||
|
|
5468287f9b | ||
|
|
22e55677ae | ||
|
|
8d422eeda0 | ||
|
|
62dbd9e121 | ||
|
|
c950c96934 | ||
|
|
1fa12d473c | ||
|
|
b9def78d2e | ||
|
|
626356be55 | ||
|
|
85f5425d89 | ||
|
|
d2c703ef60 | ||
|
|
4e8e6390b1 | ||
|
|
9271e375af | ||
|
|
35a9685b71 | ||
|
|
723ea558fa | ||
|
|
8a4a635ee3 | ||
|
|
1a30e9fd11 | ||
|
|
dc8e1c764b | ||
|
|
48e9148728 | ||
|
|
25525e0b03 | ||
|
|
9720c0ecba | ||
|
|
33cbe7cf22 | ||
|
|
4b0eef9c2e | ||
|
|
6e08a94da7 | ||
|
|
c8f621cea2 | ||
|
|
6436ec6416 | ||
|
|
e7c3d9abee | ||
|
|
c8bc942eb4 | ||
|
|
d4fcaa54ba | ||
|
|
2118f881f6 | ||
|
|
05884ead56 | ||
|
|
e53e04ca05 | ||
|
|
32b2cd9ef3 | ||
|
|
f734a76588 | ||
|
|
d71b1ee052 | ||
|
|
6b1d4a249a | ||
|
|
6b69d7c9af | ||
|
|
ff4f4be69c | ||
|
|
a7f9e8d8eb | ||
|
|
163732cea0 | ||
|
|
c35a57d2ca | ||
|
|
b40ddbf47b | ||
|
|
6be825184a | ||
|
|
4782195ca5 | ||
|
|
1280daafe3 | ||
|
|
c05433f4f9 | ||
|
|
88567fb056 | ||
|
|
7c09b66d53 | ||
|
|
31853411f3 | ||
|
|
f036e83894 | ||
|
|
24a3af210a | ||
|
|
4f8a94bfe7 | ||
|
|
45fbdd58af | ||
|
|
b28f6f4bb2 | ||
|
|
b691a74369 | ||
|
|
d0c9af8e67 | ||
|
|
92cc5eaae4 | ||
|
|
ca31d1c3e8 | ||
|
|
e4df544634 | ||
|
|
bdd2d611d2 | ||
|
|
af95d0c3dc | ||
|
|
2dcf675fbe |
@@ -1,5 +1,5 @@
|
|||||||
# [Choice] Node.js version (use -bullseye variants on local arm64/Apple Silicon): 18, 16, 14, 18-bullseye, 16-bullseye, 14-bullseye, 18-buster, 16-buster, 14-buster
|
# [Choice] Node.js version (use -bullseye variants on local arm64/Apple Silicon): 18, 16, 14, 18-bullseye, 16-bullseye, 14-bullseye, 18-buster, 16-buster, 14-buster
|
||||||
ARG VARIANT=18-bullseye
|
ARG VARIANT=20
|
||||||
FROM mcr.microsoft.com/vscode/devcontainers/javascript-node:0-${VARIANT}
|
FROM mcr.microsoft.com/vscode/devcontainers/javascript-node:0-${VARIANT}
|
||||||
|
|
||||||
# [Optional] Uncomment this section to install additional OS packages.
|
# [Optional] Uncomment this section to install additional OS packages.
|
||||||
@@ -13,4 +13,4 @@ FROM mcr.microsoft.com/vscode/devcontainers/javascript-node:0-${VARIANT}
|
|||||||
# [Optional] Uncomment if you want to install more global node modules
|
# [Optional] Uncomment if you want to install more global node modules
|
||||||
# RUN su node -c "npm install -g <your-package-list-here>"
|
# RUN su node -c "npm install -g <your-package-list-here>"
|
||||||
|
|
||||||
RUN su node -c "npm install -g pnpm"
|
RUN su node -c "npm install -g pnpm"
|
||||||
|
|||||||
@@ -2,29 +2,27 @@
|
|||||||
// https://github.com/microsoft/vscode-dev-containers/tree/v0.245.2/containers/javascript-node-postgres
|
// https://github.com/microsoft/vscode-dev-containers/tree/v0.245.2/containers/javascript-node-postgres
|
||||||
// Update the VARIANT arg in docker-compose.yml to pick a Node.js version
|
// Update the VARIANT arg in docker-compose.yml to pick a Node.js version
|
||||||
{
|
{
|
||||||
"name": "Node.js & PostgreSQL",
|
"name": "Node.js & PostgreSQL",
|
||||||
"dockerComposeFile": "docker-compose.yml",
|
"dockerComposeFile": "docker-compose.yml",
|
||||||
"service": "app",
|
"service": "app",
|
||||||
"workspaceFolder": "/workspace",
|
"workspaceFolder": "/workspace",
|
||||||
|
|
||||||
// Configure tool-specific properties.
|
// Configure tool-specific properties.
|
||||||
"customizations": {
|
"customizations": {
|
||||||
// Configure properties specific to VS Code.
|
// Configure properties specific to VS Code.
|
||||||
"vscode": {
|
"vscode": {
|
||||||
// Add the IDs of extensions you want installed when the container is created.
|
// Add the IDs of extensions you want installed when the container is created.
|
||||||
"extensions": [
|
"extensions": ["dbaeumer.vscode-eslint"]
|
||||||
"dbaeumer.vscode-eslint"
|
}
|
||||||
]
|
},
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// Use 'forwardPorts' to make a list of ports inside the container available locally.
|
// Use 'forwardPorts' to make a list of ports inside the container available locally.
|
||||||
// This can be used to network with other containers or with the host.
|
// This can be used to network with other containers or with the host.
|
||||||
"forwardPorts": [3000, 5432, 8025],
|
"forwardPorts": [3000, 5432, 8025],
|
||||||
|
|
||||||
// Use 'postCreateCommand' to run commands after the container is created.
|
// Use 'postCreateCommand' to run commands after the container is created.
|
||||||
"postCreateCommand": "pnpm install",
|
"postCreateCommand": "cp .env.example .env && sed -i '/^ENCRYPTION_KEY=/c\\ENCRYPTION_KEY='$(openssl rand -hex 32) .env && sed -i '/^NEXTAUTH_SECRET=/c\\NEXTAUTH_SECRET='$(openssl rand -hex 32) .env && pnpm install && pnpm db:migrate:dev",
|
||||||
|
|
||||||
// Comment out to connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root.
|
// Comment out to connect as root instead. More info: https://aka.ms/vscode-remote/containers/non-root.
|
||||||
"remoteUser": "node"
|
"remoteUser": "node"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,10 +6,10 @@ services:
|
|||||||
context: .
|
context: .
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
args:
|
args:
|
||||||
# Update 'VARIANT' to pick an LTS version of Node.js: 18, 16, 14.
|
# Update 'VARIANT' to pick an LTS version of Node.js: 20, 18, 16, 14.
|
||||||
# Append -bullseye or -buster to pin to an OS version.
|
# Append -bullseye or -buster to pin to an OS version.
|
||||||
# Use -bullseye variants on local arm64/Apple Silicon.
|
# Use -bullseye variants on local arm64/Apple Silicon.
|
||||||
VARIANT: "18"
|
VARIANT: "20"
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
- ..:/workspace:cached
|
- ..:/workspace:cached
|
||||||
@@ -33,7 +33,7 @@ services:
|
|||||||
environment:
|
environment:
|
||||||
POSTGRES_PASSWORD: postgres
|
POSTGRES_PASSWORD: postgres
|
||||||
POSTGRES_USER: postgres
|
POSTGRES_USER: postgres
|
||||||
POSTGRES_DB: postgres
|
POSTGRES_DB: formbricks
|
||||||
# Add "forwardPorts": ["5432"] to **devcontainer.json** to forward PostgreSQL locally.
|
# Add "forwardPorts": ["5432"] to **devcontainer.json** to forward PostgreSQL locally.
|
||||||
# (Adding the "ports" property to this file will not forward from a Codespace.)
|
# (Adding the "ports" property to this file will not forward from a Codespace.)
|
||||||
|
|
||||||
|
|||||||
19
.env.example
@@ -1,4 +1,3 @@
|
|||||||
/*
|
|
||||||
########################################################################
|
########################################################################
|
||||||
# ------------ MANDATORY (CHANGE ACCORDING TO YOUR SETUP) ------------#
|
# ------------ MANDATORY (CHANGE ACCORDING TO YOUR SETUP) ------------#
|
||||||
########################################################################
|
########################################################################
|
||||||
@@ -64,10 +63,10 @@ SMTP_PASSWORD=smtpPassword
|
|||||||
#####################
|
#####################
|
||||||
|
|
||||||
# Email Verification. If you enable Email Verification you have to setup SMTP-Settings, too.
|
# Email Verification. If you enable Email Verification you have to setup SMTP-Settings, too.
|
||||||
# EMAIL_VERIFICATION_DISABLED=1
|
EMAIL_VERIFICATION_DISABLED=1
|
||||||
|
|
||||||
# Password Reset. If you enable Password Reset functionality you have to setup SMTP-Settings, too.
|
# Password Reset. If you enable Password Reset functionality you have to setup SMTP-Settings, too.
|
||||||
# PASSWORD_RESET_DISABLED=1
|
PASSWORD_RESET_DISABLED=1
|
||||||
|
|
||||||
# Signup. Disable the ability for new users to create an account.
|
# Signup. Disable the ability for new users to create an account.
|
||||||
# SIGNUP_DISABLED=1
|
# SIGNUP_DISABLED=1
|
||||||
@@ -99,7 +98,6 @@ AZUREAD_AUTH_ENABLED=0
|
|||||||
AZUREAD_CLIENT_ID=
|
AZUREAD_CLIENT_ID=
|
||||||
AZUREAD_CLIENT_SECRET=
|
AZUREAD_CLIENT_SECRET=
|
||||||
AZUREAD_TENANT_ID=
|
AZUREAD_TENANT_ID=
|
||||||
AZURE_DIRECT_REDIRECT=0
|
|
||||||
|
|
||||||
# Cron Secret
|
# Cron Secret
|
||||||
CRON_SECRET=
|
CRON_SECRET=
|
||||||
@@ -107,6 +105,10 @@ CRON_SECRET=
|
|||||||
# Configure this when you want to ship JS & CSS files from a complete URL instead of the current domain
|
# Configure this when you want to ship JS & CSS files from a complete URL instead of the current domain
|
||||||
# ASSET_PREFIX_URL=
|
# ASSET_PREFIX_URL=
|
||||||
|
|
||||||
|
# Oauth credentials for Notion Integration
|
||||||
|
NOTION_OAUTH_CLIENT_ID=
|
||||||
|
NOTION_OAUTH_CLIENT_SECRET=
|
||||||
|
|
||||||
# Stripe Billing Variables
|
# Stripe Billing Variables
|
||||||
STRIPE_SECRET_KEY=
|
STRIPE_SECRET_KEY=
|
||||||
STRIPE_WEBHOOK_SECRET=
|
STRIPE_WEBHOOK_SECRET=
|
||||||
@@ -127,4 +129,11 @@ AIRTABLE_CLIENT_ID=
|
|||||||
# Enterprise License Key
|
# Enterprise License Key
|
||||||
ENTERPRISE_LICENSE_KEY=
|
ENTERPRISE_LICENSE_KEY=
|
||||||
|
|
||||||
*/
|
# Automatically assign new users to a specific team and role within that team
|
||||||
|
# Insert an existing team id or generate a valid CUID for a new one at https://www.getuniqueid.com/cuid (e.g. cjld2cjxh0000qzrmn831i7rn)
|
||||||
|
# (Role Management is an Enterprise feature)
|
||||||
|
# DEFAULT_TEAM_ID=
|
||||||
|
# DEFAULT_TEAM_ROLE=admin
|
||||||
|
|
||||||
|
# set to 1 to skip onboarding for new users
|
||||||
|
# ONBOARDING_DISABLED=1
|
||||||
|
|||||||
@@ -4,8 +4,8 @@ on:
|
|||||||
# "Scheduled workflows run on the latest commit on the default or base branch."
|
# "Scheduled workflows run on the latest commit on the default or base branch."
|
||||||
# — https://docs.github.com/en/actions/learn-github-actions/events-that-trigger-workflows#schedule
|
# — https://docs.github.com/en/actions/learn-github-actions/events-that-trigger-workflows#schedule
|
||||||
schedule:
|
schedule:
|
||||||
# This will run the job at 23:00 UTC every day of every month.
|
# This will run the job at 22:00 UTC every day of every month.
|
||||||
- cron: "0 21 * * *"
|
- cron: "0 22 * * *"
|
||||||
jobs:
|
jobs:
|
||||||
cron-reportUsageToStripe:
|
cron-reportUsageToStripe:
|
||||||
env:
|
env:
|
||||||
@@ -19,4 +19,5 @@ jobs:
|
|||||||
curl ${{ env.APP_URL }}/api/cron/report-usage \
|
curl ${{ env.APP_URL }}/api/cron/report-usage \
|
||||||
-X POST \
|
-X POST \
|
||||||
-H 'x-api-key: ${{ env.CRON_SECRET }}' \
|
-H 'x-api-key: ${{ env.CRON_SECRET }}' \
|
||||||
|
-H 'Cache-Control: no-cache' \
|
||||||
--fail
|
--fail
|
||||||
|
|||||||
2
.github/workflows/lint.yml
vendored
@@ -25,7 +25,7 @@ jobs:
|
|||||||
- name: create .env
|
- name: create .env
|
||||||
run: cp .env.example .env
|
run: cp .env.example .env
|
||||||
|
|
||||||
- name: Generate Random NEXTAUTH_SECRET
|
- name: Generate Random ENCRYPTION_KEY
|
||||||
run: |
|
run: |
|
||||||
SECRET=$(openssl rand -hex 32)
|
SECRET=$(openssl rand -hex 32)
|
||||||
echo "ENCRYPTION_KEY=$SECRET" >> $GITHUB_ENV
|
echo "ENCRYPTION_KEY=$SECRET" >> $GITHUB_ENV
|
||||||
|
|||||||
40
.github/workflows/playwright.yml
vendored
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
name: E2E Tests
|
||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
name: Run E2E Tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 60
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: 20
|
||||||
|
|
||||||
|
- name: Install Docker Compose
|
||||||
|
run: sudo apt-get update && sudo apt-get install -y docker-compose
|
||||||
|
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm install -g pnpm && pnpm install
|
||||||
|
|
||||||
|
- name: Build Formricks JS package
|
||||||
|
run: pnpm build --filter=js
|
||||||
|
|
||||||
|
- name: Build Formbricks Image & Run
|
||||||
|
run: docker-compose up -d
|
||||||
|
|
||||||
|
- name: Install Playwright Browsers
|
||||||
|
run: pnpm exec playwright install --with-deps
|
||||||
|
|
||||||
|
- name: Run Playwright tests
|
||||||
|
run: pnpm test:e2e
|
||||||
|
|
||||||
|
- uses: actions/upload-artifact@v3
|
||||||
|
if: always()
|
||||||
|
with:
|
||||||
|
name: playwright-report
|
||||||
|
path: playwright-report/
|
||||||
|
retention-days: 30
|
||||||
7
.github/workflows/pr.yml
vendored
@@ -27,8 +27,13 @@ jobs:
|
|||||||
uses: ./.github/workflows/build-web.yml
|
uses: ./.github/workflows/build-web.yml
|
||||||
secrets: inherit
|
secrets: inherit
|
||||||
|
|
||||||
|
e2e-test:
|
||||||
|
name: Run E2E Tests
|
||||||
|
uses: ./.github/workflows/playwright.yml
|
||||||
|
secrets: inherit
|
||||||
|
|
||||||
required:
|
required:
|
||||||
needs: [lint, test, build]
|
needs: [lint, test, build, e2e-test]
|
||||||
if: always()
|
if: always()
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
|
|||||||
28
.github/workflows/release-docker-github.yml
vendored
@@ -6,6 +6,7 @@ name: Docker
|
|||||||
# documentation.
|
# documentation.
|
||||||
|
|
||||||
on:
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
push:
|
push:
|
||||||
tags:
|
tags:
|
||||||
- "v*"
|
- "v*"
|
||||||
@@ -15,6 +16,9 @@ env:
|
|||||||
REGISTRY: ghcr.io
|
REGISTRY: ghcr.io
|
||||||
# github.repository as <account>/<repo>
|
# github.repository as <account>/<repo>
|
||||||
IMAGE_NAME: ${{ github.repository }}
|
IMAGE_NAME: ${{ github.repository }}
|
||||||
|
TURBO_TOKEN: ${{ secrets.TURBO_TOKEN }}
|
||||||
|
TURBO_TEAM: ${{ secrets.TURBO_TEAM }}
|
||||||
|
DATABASE_URL: "postgresql://postgres:postgres@localhost:5432/formbricks?schema=public"
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@@ -27,6 +31,16 @@ jobs:
|
|||||||
id-token: write
|
id-token: write
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
- name: Generate Random NEXTAUTH_SECRET
|
||||||
|
run: |
|
||||||
|
SECRET=$(openssl rand -hex 32)
|
||||||
|
echo "NEXTAUTH_SECRET=$SECRET" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Generate Random ENCRYPTION_KEY
|
||||||
|
run: |
|
||||||
|
SECRET=$(openssl rand -hex 32)
|
||||||
|
echo "ENCRYPTION_KEY=$SECRET" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
@@ -38,17 +52,22 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
cosign-release: "v2.1.1"
|
cosign-release: "v2.1.1"
|
||||||
|
|
||||||
|
# Add support for more platforms with QEMU (optional)
|
||||||
|
# https://github.com/docker/setup-qemu-action
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
|
||||||
# Set up BuildKit Docker container builder to be able to build
|
# Set up BuildKit Docker container builder to be able to build
|
||||||
# multi-platform images and export cache
|
# multi-platform images and export cache
|
||||||
# https://github.com/docker/setup-buildx-action
|
# https://github.com/docker/setup-buildx-action
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@f95db51fddba0c2d1ec667646a06c2ce06100226 # v3.0.0
|
uses: docker/setup-buildx-action@v3 # v3.0.0
|
||||||
|
|
||||||
# Login against a Docker registry except on PR
|
# Login against a Docker registry except on PR
|
||||||
# https://github.com/docker/login-action
|
# https://github.com/docker/login-action
|
||||||
- name: Log into registry ${{ env.REGISTRY }}
|
- name: Log into registry ${{ env.REGISTRY }}
|
||||||
if: github.event_name != 'pull_request'
|
if: github.event_name != 'pull_request'
|
||||||
uses: docker/login-action@343f7c4344506bcbf9b4de18042ae17996df046d # v3.0.0
|
uses: docker/login-action@v3 # v3.0.0
|
||||||
with:
|
with:
|
||||||
registry: ${{ env.REGISTRY }}
|
registry: ${{ env.REGISTRY }}
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
@@ -58,7 +77,7 @@ jobs:
|
|||||||
# https://github.com/docker/metadata-action
|
# https://github.com/docker/metadata-action
|
||||||
- name: Extract Docker metadata
|
- name: Extract Docker metadata
|
||||||
id: meta
|
id: meta
|
||||||
uses: docker/metadata-action@96383f45573cb7f253c731d3b3ab81c87ef81934 # v5.0.0
|
uses: docker/metadata-action@v5 # v5.0.0
|
||||||
with:
|
with:
|
||||||
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}
|
||||||
|
|
||||||
@@ -66,10 +85,11 @@ jobs:
|
|||||||
# https://github.com/docker/build-push-action
|
# https://github.com/docker/build-push-action
|
||||||
- name: Build and push Docker image
|
- name: Build and push Docker image
|
||||||
id: build-and-push
|
id: build-and-push
|
||||||
uses: docker/build-push-action@0565240e2d4ab88bba5387d719585280857ece09 # v5.0.0
|
uses: docker/build-push-action@v5 # v5.0.0
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
file: ./apps/web/Dockerfile
|
file: ./apps/web/Dockerfile
|
||||||
|
# platforms: linux/amd64,linux/arm64
|
||||||
push: ${{ github.event_name != 'pull_request' }}
|
push: ${{ github.event_name != 'pull_request' }}
|
||||||
tags: ${{ steps.meta.outputs.tags }}
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
|
|||||||
5
.github/workflows/test.yml
vendored
@@ -29,6 +29,11 @@ jobs:
|
|||||||
- name: create .env
|
- name: create .env
|
||||||
run: cp .env.example .env
|
run: cp .env.example .env
|
||||||
|
|
||||||
|
- name: Generate Random ENCRYPTION_KEY
|
||||||
|
run: |
|
||||||
|
SECRET=$(openssl rand -hex 32)
|
||||||
|
echo "ENCRYPTION_KEY=$SECRET" >> $GITHUB_ENV
|
||||||
|
|
||||||
- name: Build formbricks-js dependencies
|
- name: Build formbricks-js dependencies
|
||||||
run: pnpm build --filter=js
|
run: pnpm build --filter=js
|
||||||
|
|
||||||
|
|||||||
8
.gitignore
vendored
@@ -44,4 +44,10 @@ packages/database/zod
|
|||||||
# nixos stuff
|
# nixos stuff
|
||||||
.direnv
|
.direnv
|
||||||
|
|
||||||
Zone.Identifier
|
Zone.Identifier
|
||||||
|
|
||||||
|
# Playwright
|
||||||
|
/test-results/
|
||||||
|
/playwright-report/
|
||||||
|
/blob-report/
|
||||||
|
/playwright/.cache/
|
||||||
|
|||||||
@@ -1 +1,6 @@
|
|||||||
module.exports = require("./packages/prettier-config/prettier-preset");
|
const baseConfig = require("./packages/prettier-config/prettier-preset");
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
...baseConfig,
|
||||||
|
plugins: ["@trivago/prettier-plugin-sort-imports", "prettier-plugin-tailwindcss"],
|
||||||
|
};
|
||||||
|
|||||||
132
README.md
@@ -1,100 +1,128 @@
|
|||||||
<div id="top"></div>
|
<div id="top"></div>
|
||||||
<p align="center">
|
|
||||||
<a href="https://formbricks.com">
|
|
||||||
<img width="120" alt="Open Source Experience Management Solution Qualtrics Alternative Logo" src="https://github.com/formbricks/formbricks/assets/72809645/0086704f-bee7-4d38-9cc8-fa42ee59e004">
|
|
||||||
</a>
|
|
||||||
<h3 align="center">Formbricks</h3>
|
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
The Open Source Survey Toolbox
|
|
||||||
<br />
|
<a href="https://formbricks.com">
|
||||||
<a href="https://formbricks.com/">Website</a> | <a href="https://formbricks.com/discord">Join Discord community</a>
|
|
||||||
</p>
|
<img width="120" alt="Open Source Privacy First Experience Management Solution Qualtrics Alternative Logo" src="https://github.com/formbricks/formbricks/assets/72809645/0086704f-bee7-4d38-9cc8-fa42ee59e004">
|
||||||
|
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<h3 align="center">Formbricks</h3>
|
||||||
|
|
||||||
|
<p align="center">
|
||||||
|
Harvest user-insights, build irresistible experiences.
|
||||||
|
<br />
|
||||||
|
<a href="https://formbricks.com/">Website</a> | <a href="https://formbricks.com/discord">Join Discord community</a>
|
||||||
|
</p>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<a href="https://github.com/formbricks/formbricks/blob/main/LICENSE"><img src="https://img.shields.io/badge/License-AGPL-purple" alt="License"></a> <a href="https://formbricks.com/discord"><img src="https://img.shields.io/discord/979077669410979880?label=Discord&logo=discord&logoColor=%23fff" alt="Join Formbricks Discord"></a> <a href="https://github.com/formbricks/formbricks/stargazers"><img src="https://img.shields.io/github/stars/formbricks/formbricks?logo=github" alt="Github Stars"></a>
|
<a href="https://github.com/formbricks/formbricks/blob/main/LICENSE"><img src="https://img.shields.io/badge/License-AGPL-purple" alt="License"></a> <a href="https://formbricks.com/discord"><img src="https://img.shields.io/discord/979077669410979880?label=Discord&logo=discord&logoColor=%23fff" alt="Join Formbricks Discord"></a> <a href="https://github.com/formbricks/formbricks/stargazers"><img src="https://img.shields.io/github/stars/formbricks/formbricks?logo=github" alt="Github Stars"></a>
|
||||||
<a href="https://news.ycombinator.com/item?id=32303986"><img src="https://img.shields.io/badge/Hacker%20News-122-%23FF6600" alt="Hacker News"></a>
|
<a href="https://news.ycombinator.com/item?id=32303986"><img src="https://img.shields.io/badge/Hacker%20News-122-%23FF6600" alt="Hacker News"></a>
|
||||||
<a href="[https://www.producthunt.com/products/formbricks](https://www.producthunt.com/posts/formbricks)"><img src="https://img.shields.io/badge/Product%20Hunt-455-orange?logo=producthunt&logoColor=%23fff" alt="Product Hunt"></a>
|
<a href="[https://www.producthunt.com/products/formbricks](https://www.producthunt.com/posts/formbricks)"><img src="https://img.shields.io/badge/Product%20Hunt-455-orange?logo=producthunt&logoColor=%23fff" alt="Product Hunt"></a>
|
||||||
<a href="https://github.blog/2023-04-12-github-accelerator-our-first-cohort-and-whats-next/"><img src="https://img.shields.io/badge/2023-blue?logo=github&label=Github%20Accelerator" alt="Github Accelerator"></a>
|
<a href="https://github.blog/2023-04-12-github-accelerator-our-first-cohort-and-whats-next/"><img src="https://img.shields.io/badge/2023-blue?logo=github&label=Github%20Accelerator" alt="Github Accelerator"></a>
|
||||||
<a href="https://github.com/formbricks/formbricks/issues?q=is:issue+is:open+label:%22%F0%9F%99%8B%F0%9F%8F%BB%E2%80%8D%E2%99%82%EF%B8%8Fhelp+wanted%22"><img src="https://img.shields.io/badge/Help%20Wanted-Contribute-blue"></a>
|
<a href="https://github.com/formbricks/formbricks/issues?q=is:issue+is:open+label:%22%F0%9F%99%8B%F0%9F%8F%BB%E2%80%8D%E2%99%82%EF%B8%8Fhelp+wanted%22"><img src="https://img.shields.io/badge/Help%20Wanted-Contribute-blue"></a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<br/>
|
<br/>
|
||||||
|
|
||||||
<p align="center">
|
<div style="background-color:#f8fafc; border-radius:5px;">
|
||||||
|
<p align="center">
|
||||||
<i>Trusted by</i>
|
<i>Trusted by</i>
|
||||||
<a href="https://github.com/calcom/cal.com/"><img src="https://github.com/formbricks/formbricks/assets/675065/1a8763cf-f47e-4960-90f6-334f6dc12a17#gh-light-mode-only" height="20px"></a><a href="https://github.com/calcom/cal.com/"><img src="https://github.com/formbricks/formbricks/assets/72809645/9a031e8d-538f-4fdc-9338-b77e9a57d6ac#gh-dark-mode-only" height="20px"></a>
|
<a href="https://flixbus.com"><img src="https://github.com/formbricks/formbricks/assets/72809645/d6c91d89-7633-4845-ae1e-03bbd2ce0946" height="35px"></a>
|
||||||
<a href="https://github.com/CrowdDotDev/crowd.dev"><img src="https://github.com/formbricks/formbricks/assets/675065/59b1a4d4-25e4-4ef3-b0bf-4426446fbfd0#gh-light-mode-only" height="20px"></a><a href="https://github.com/CrowdDotDev/crowd.dev"><img src="https://github.com/formbricks/formbricks/assets/72809645/4bb4caf7-4b64-44c8-94bd-850606d181c1#gh-dark-mode-only" height="20px"></a>
|
<a href="https://github.com/calcom/cal.com/"><img src="https://github.com/formbricks/formbricks/assets/675065/1a8763cf-f47e-4960-90f6-334f6dc12a17#gh-light-mode-only" height="20px"></a>
|
||||||
<a href="https://clovyr.io/"><img src="https://github.com/formbricks/formbricks/assets/675065/9291c8df-9aac-423a-a430-a9a581240075" height="20px"></a>
|
<a href="https://github.com/CrowdDotDev/crowd.dev"><img src="https://github.com/formbricks/formbricks/assets/675065/59b1a4d4-25e4-4ef3-b0bf-4426446fbfd0#gh-light-mode-only" height="20px"></a>
|
||||||
<a href="https://neverinstall.com/"><img src="https://github.com/formbricks/formbricks/assets/675065/72e5e37b-8ef7-4340-b06e-f1d12a05330f#gh-light-mode-only" height="20px"></a><a href="https://neverinstall.com/"><img src="https://github.com/formbricks/formbricks/assets/72809645/9d9711dc-75e5-4084-b7fa-bbaf621064a8#gh-dark-mode-only" height="20px">
|
<a href="https://neverinstall.com/"><img src="https://github.com/formbricks/formbricks/assets/675065/72e5e37b-8ef7-4340-b06e-f1d12a05330f#gh-light-mode-only" height="20px"></a>
|
||||||
|
<a href="https://clovyr.io/"><img src="https://github.com/formbricks/formbricks/assets/675065/9291c8df-9aac-423a-a430-a9a581240075" height="20px"></a>
|
||||||
</p>
|
</p>
|
||||||
|
<div>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<a href="https://trendshift.io/repositories/2570" target="_blank"><img src="https://trendshift.io/api/badge/repositories/2570" alt="Trendshift Badge for formbricks/formbricks" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
<a href="https://trendshift.io/repositories/2570" target="_blank"><img src="https://trendshift.io/api/badge/repositories/2570" alt="Trendshift Badge for formbricks/formbricks" style="width: 250px; height: 55px;" width="250" height="55"/></a>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
## ✨ About Formbricks
|
## ✨ About Formbricks
|
||||||
|
|
||||||
<img width="1527" alt="formbricks-sneak" src="https://github-production-user-asset-6210df.s3.amazonaws.com/675065/249441967-ccb89ea3-82b4-4bf2-8d2c-528721ec313b.png">
|
<img width="1527" alt="formbricks-sneak" src="https://github-production-user-asset-6210df.s3.amazonaws.com/675065/249441967-ccb89ea3-82b4-4bf2-8d2c-528721ec313b.png">
|
||||||
|
|
||||||
Formbricks is your go-to solution for in-product micro-surveys that will supercharge your product experience. Use micro-surveys to target the right users at the right time without making surveys annoying.
|
Formbricks provides a free and open source surveying platform. Gather feedback at every point in the user journey with beautiful in-app, website, link and email surveys. Build on top of Formbricks or leverage prebuilt data analysis capabilities.
|
||||||
|
|
||||||
**Try it out in the cloud at [formbricks.com](https://formbricks.com)**
|
**Try it out in the cloud at [formbricks.com](https://app.formbricks.com/auth/signup)**
|
||||||
|
|
||||||
## 💪 Mission: Make customer-centric decisions based on data.
|
## 💪 Mission: Empower your team, craft an irresistible experience.
|
||||||
|
|
||||||
Formbricks is a powerful tool for creating in-product micro-surveys - and leverage a significantly higher conversion rate. It allows you to gather valuable insights from your users, enabling you to make data-driven decisions that enhance your product's user experience. With Formbricks, you can create surveys with our no-code editor, choose from a variety of templates, target specific user groups, and much more.
|
Formbricks is both a free and open source survey platform - and a privacy-first experience management platform. Use in-app, website, link and email surveys to gather user and customer insights at every point of their journey. Leverage Formbricks Insight Platform or build your own. Life's too short for mediocre UX.
|
||||||
|
|
||||||
### Table of Contents
|
### Table of Contents
|
||||||
|
|
||||||
- [Features](#features)
|
- [Features](#features)
|
||||||
|
|
||||||
- [Getting Started](#getting-started)
|
- [Getting Started](#getting-started)
|
||||||
- [Cloud Version](#cloud-version)
|
|
||||||
- [Self-hosted Version](#self-hosted-version)
|
- [Cloud Version](#cloud-version)
|
||||||
- [Development](#development)
|
|
||||||
|
- [Self-hosted Version](#self-hosted-version)
|
||||||
|
|
||||||
|
- [Development](#development)
|
||||||
|
|
||||||
- [Contribution](#contribution)
|
- [Contribution](#contribution)
|
||||||
|
|
||||||
- [Contact](#contact-us)
|
- [Contact](#contact-us)
|
||||||
|
|
||||||
- [License](#license)
|
- [License](#license)
|
||||||
|
|
||||||
- [Security](#security)
|
- [Security](#security)
|
||||||
|
|
||||||
<a id="features"></a>
|
<a id="features"></a>
|
||||||
|
|
||||||
### Features
|
### Features
|
||||||
|
|
||||||
- 📲 Create **in-product surveys** with our no-code editor with multiple question types.
|
- 📲 Create **conversion-optimized surveys** with our no-code editor with several question types.
|
||||||
|
|
||||||
- 📚 Choose from a variety of best-practice **templates**.
|
- 📚 Choose from a variety of best-practice **templates**.
|
||||||
|
|
||||||
- 👩🏻 Launch and **target your surveys to specific user groups** without changing your application code.
|
- 👩🏻 Launch and **target your surveys to specific user groups** without changing your application code.
|
||||||
|
|
||||||
- 🔗 Create shareable **link surveys**.
|
- 🔗 Create shareable **link surveys**.
|
||||||
|
|
||||||
- 👨👩👦 Invite your team members to **collaborate** on your surveys.
|
- 👨👩👦 Invite your team members to **collaborate** on your surveys.
|
||||||
- 🔌 Integrate Formbricks with **Slack, Posthog, Zapier, n8n and more**.
|
|
||||||
|
- 🔌 Integrate Formbricks with **Slack, Notion, Zapier, n8n and more**.
|
||||||
|
|
||||||
- 🔒 All **open source**, transparent and self-hostable.
|
- 🔒 All **open source**, transparent and self-hostable.
|
||||||
|
|
||||||
### Built on Open Source
|
### Built on Open Source
|
||||||
|
|
||||||
- 💻 [Typescript](https://www.typescriptlang.org/)
|
- 💻 [Typescript](https://www.typescriptlang.org/)
|
||||||
|
|
||||||
- 🚀 [Next.js](https://nextjs.org/)
|
- 🚀 [Next.js](https://nextjs.org/)
|
||||||
|
|
||||||
- ⚛️ [React](https://reactjs.org/)
|
- ⚛️ [React](https://reactjs.org/)
|
||||||
|
|
||||||
- 🎨 [TailwindCSS](https://tailwindcss.com/)
|
- 🎨 [TailwindCSS](https://tailwindcss.com/)
|
||||||
|
|
||||||
- 📚 [Prisma](https://prisma.io/)
|
- 📚 [Prisma](https://prisma.io/)
|
||||||
|
|
||||||
- 🔒 [Auth.js](https://authjs.dev/)
|
- 🔒 [Auth.js](https://authjs.dev/)
|
||||||
|
|
||||||
- 🧘♂️ [Zod](https://zod.dev/)
|
- 🧘♂️ [Zod](https://zod.dev/)
|
||||||
|
|
||||||
<a id="getting-started"></a>
|
<a id="getting-started"></a>
|
||||||
|
|
||||||
## 🚀 Getting started
|
## 🚀 Getting started
|
||||||
|
|
||||||
We've got several options depending on your need to help you quickly get started with Formbricks.
|
We've got several options depending on your need to help you quickly get started with Formbricks.
|
||||||
|
|
||||||
<a id="cloud-version"></a>
|
<a id="cloud-version"></a>
|
||||||
|
|
||||||
### ☁️ Cloud Version
|
### ☁️ Cloud Version
|
||||||
|
|
||||||
Formbricks has a hosted cloud offering with a generous free plan to get you up and running as quickly as possible. To get started, please visit [formbricks.com](https://formbricks.com).
|
Formbricks has a hosted cloud offering with a generous free plan to get you up and running as quickly as possible. To get started, please visit [formbricks.com](https://app.formbricks.com/auth/signup).
|
||||||
|
|
||||||
<a id="self-hosted-version"></a>
|
<a id="self-hosted-version"></a>
|
||||||
|
|
||||||
### 🐳 Self-hosted version
|
### 🐳 Self-hosting Formbricks
|
||||||
|
|
||||||
Formbricks is available Open-Source under AGPLv3 license. You can host Formbricks on your own servers using Docker without a subscription.
|
Formbricks is available Open-Source under AGPLv3 license. You can host Formbricks on your own servers using Docker without a subscription.
|
||||||
|
|
||||||
@@ -114,7 +142,7 @@ You can deploy Formbricks on [Railway](https://railway.app) using the button bel
|
|||||||
|
|
||||||
[](https://railway.app/new/template/PPDzCd)
|
[](https://railway.app/new/template/PPDzCd)
|
||||||
|
|
||||||
<a id="development"></a>
|
<a id="development"></a>
|
||||||
|
|
||||||
### 👨💻 Development
|
### 👨💻 Development
|
||||||
|
|
||||||
@@ -123,7 +151,9 @@ You can deploy Formbricks on [Railway](https://railway.app) using the button bel
|
|||||||
Here is what you need to be able to run Formbricks:
|
Here is what you need to be able to run Formbricks:
|
||||||
|
|
||||||
- [Node.js](https://nodejs.org/en) (Version: >=18.x)
|
- [Node.js](https://nodejs.org/en) (Version: >=18.x)
|
||||||
|
|
||||||
- [Pnpm](https://pnpm.io/)
|
- [Pnpm](https://pnpm.io/)
|
||||||
|
|
||||||
- [Docker](https://www.docker.com/) - to run PostgreSQL and MailHog
|
- [Docker](https://www.docker.com/) - to run PostgreSQL and MailHog
|
||||||
|
|
||||||
#### Local Setup
|
#### Local Setup
|
||||||
@@ -138,7 +168,7 @@ To get started locally, we've got a [guide to help you](https://formbricks.com/d
|
|||||||
|
|
||||||
[](https://gitpod.io/#https://github.com/formbricks/formbricks)
|
[](https://gitpod.io/#https://github.com/formbricks/formbricks)
|
||||||
|
|
||||||
<a id="contribution"></a>
|
<a id="contribution"></a>
|
||||||
|
|
||||||
## ✍️ Contribution
|
## ✍️ Contribution
|
||||||
|
|
||||||
@@ -147,35 +177,39 @@ We are very happy if you are interested in contributing to Formbricks 🤗
|
|||||||
Here are a few options:
|
Here are a few options:
|
||||||
|
|
||||||
- Star this repo.
|
- Star this repo.
|
||||||
|
|
||||||
- Create issues every time you feel something is missing or goes wrong.
|
- Create issues every time you feel something is missing or goes wrong.
|
||||||
|
|
||||||
- Upvote issues with 👍 reaction so we know what the demand for a particular issue is to prioritize it within the roadmap.
|
- Upvote issues with 👍 reaction so we know what the demand for a particular issue is to prioritize it within the roadmap.
|
||||||
|
|
||||||
Please check out [our contribution guide](https://formbricks.com/docs/contributing/introduction) and our [list of open issues](https://github.com/formbricks/formbricks/issues) for more information.
|
Please check out [our contribution guide](https://formbricks.com/docs/contributing/introduction) and our [list of open issues](https://github.com/formbricks/formbricks/issues) for more information.
|
||||||
|
|
||||||
## All Thanks To Our Contributors
|
## All Thanks To Our Contributors
|
||||||
|
|
||||||
<a href="https://github.com/formbricks/formbricks/graphs/contributors">
|
<a href="https://github.com/formbricks/formbricks/graphs/contributors">
|
||||||
<img src="https://contrib.rocks/image?repo=formbricks/formbricks" />
|
|
||||||
|
<img src="https://contrib.rocks/image?repo=formbricks/formbricks" />
|
||||||
|
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
<a id="contact-us"></a>
|
<a id="contact-us"></a>
|
||||||
|
|
||||||
## 📆 Contact us
|
## 📆 Contact us
|
||||||
|
|
||||||
Let's have a chat about your survey needs and get you started.
|
Let's have a chat about your survey needs and get you started.
|
||||||
|
|
||||||
<a href="https://cal.com/johannes/onboarding?utm_source=banner&utm_campaign=oss"><img alt="Book us with Cal.com" src="https://cal.com/book-with-cal-dark.svg" /></a>
|
<a href="https://cal.com/johannes/onboarding?utm_source=banner&utm_campaign=oss"><img alt="Book us with Cal.com" src="https://cal.com/book-with-cal-dark.svg" /></a>
|
||||||
|
|
||||||
<a id="license"></a>
|
<a id="license"></a>
|
||||||
|
|
||||||
## ⚖️ License
|
## ⚖️ License
|
||||||
|
|
||||||
Distributed under the AGPLv3 License. See [`LICENSE`](./LICENSE) for more information.
|
Distributed under the AGPLv3 License. See [`LICENSE`](./LICENSE) for more information.
|
||||||
|
|
||||||
<a id="security"></a>
|
<a id="security"></a>
|
||||||
|
|
||||||
## 🔒 Security
|
## 🔒 Security
|
||||||
|
|
||||||
We take security very seriously. If you come across any security vulnerabilities, please disclose them by sending an email to security@formbricks.com. We appreciate your help in making our platform as secure as possible and are committed to working with you to resolve any issues quickly and efficiently. See [`SECURITY.md`](./SECURITY.md) for more information.
|
We take security very seriously. If you come across any security vulnerabilities, please disclose them by sending an email to security@formbricks.com. We appreciate your help in making our platform as secure as possible and are committed to working with you to resolve any issues quickly and efficiently. See [`SECURITY.md`](./SECURITY.md) for more information.
|
||||||
|
|
||||||
<p align="right"><a href="#top">🔼 Back to top</a></p>
|
<p align="right"><a href="#top">🔼 Back to top</a></p>
|
||||||
|
|||||||
@@ -12,8 +12,8 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@formbricks/js": "workspace:*",
|
"@formbricks/js": "workspace:*",
|
||||||
"@heroicons/react": "^2.0.18",
|
"@heroicons/react": "^2.1.1",
|
||||||
"next": "14.0.0",
|
"next": "14.0.4",
|
||||||
"react": "18.2.0",
|
"react": "18.2.0",
|
||||||
"react-dom": "18.2.0"
|
"react-dom": "18.2.0"
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import type { AppProps } from "next/app";
|
import type { AppProps } from "next/app";
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
|
|
||||||
import "../styles/globals.css";
|
import "../styles/globals.css";
|
||||||
|
|
||||||
export default function App({ Component, pageProps }: AppProps) {
|
export default function App({ Component, pageProps }: AppProps) {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Html, Head, Main, NextScript } from "next/document";
|
import { Head, Html, Main, NextScript } from "next/document";
|
||||||
|
|
||||||
export default function Document() {
|
export default function Document() {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,8 +1,10 @@
|
|||||||
import formbricks from "@formbricks/js";
|
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { useEffect, useState } from "react";
|
|
||||||
import fbsetup from "../../public/fb-setup.png";
|
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
|
import formbricks from "@formbricks/js";
|
||||||
|
|
||||||
|
import fbsetup from "../../public/fb-setup.png";
|
||||||
|
|
||||||
declare const window: any;
|
declare const window: any;
|
||||||
|
|
||||||
@@ -22,11 +24,13 @@ export default function AppPage({}) {
|
|||||||
if (process.env.NEXT_PUBLIC_FORMBRICKS_ENVIRONMENT_ID && process.env.NEXT_PUBLIC_FORMBRICKS_API_HOST) {
|
if (process.env.NEXT_PUBLIC_FORMBRICKS_ENVIRONMENT_ID && process.env.NEXT_PUBLIC_FORMBRICKS_API_HOST) {
|
||||||
const isUserId = window.location.href.includes("userId=true");
|
const isUserId = window.location.href.includes("userId=true");
|
||||||
const userId = isUserId ? "THIS-IS-A-VERY-LONG-USER-ID-FOR-TESTING" : undefined;
|
const userId = isUserId ? "THIS-IS-A-VERY-LONG-USER-ID-FOR-TESTING" : undefined;
|
||||||
|
const attributes = isUserId ? { "Init Attribute 1": "eight", "Init Attribute 2": "two" } : undefined;
|
||||||
formbricks.init({
|
formbricks.init({
|
||||||
environmentId: process.env.NEXT_PUBLIC_FORMBRICKS_ENVIRONMENT_ID,
|
environmentId: process.env.NEXT_PUBLIC_FORMBRICKS_ENVIRONMENT_ID,
|
||||||
apiHost: process.env.NEXT_PUBLIC_FORMBRICKS_API_HOST,
|
apiHost: process.env.NEXT_PUBLIC_FORMBRICKS_API_HOST,
|
||||||
userId,
|
userId,
|
||||||
debug: true,
|
debug: true,
|
||||||
|
attributes,
|
||||||
});
|
});
|
||||||
window.formbricks = formbricks;
|
window.formbricks = formbricks;
|
||||||
}
|
}
|
||||||
@@ -70,7 +74,7 @@ export default function AppPage({}) {
|
|||||||
</p>
|
</p>
|
||||||
<Image src={fbsetup} alt="fb setup" className="mt-4 rounded" priority />
|
<Image src={fbsetup} alt="fb setup" className="mt-4 rounded" priority />
|
||||||
|
|
||||||
<div className="mt-4 flex-col items-start text-sm text-slate-700 dark:text-slate-300 sm:flex sm:items-center sm:text-base">
|
<div className="mt-4 flex-col items-start text-sm text-slate-700 sm:flex sm:items-center sm:text-base dark:text-slate-300">
|
||||||
<p className="mb-1 sm:mb-0 sm:mr-2">You're connected with env:</p>
|
<p className="mb-1 sm:mb-0 sm:mr-2">You're connected with env:</p>
|
||||||
<div className="flex items-center">
|
<div className="flex items-center">
|
||||||
<strong className="w-32 truncate sm:w-auto">
|
<strong className="w-32 truncate sm:w-auto">
|
||||||
|
|||||||
@@ -45,14 +45,17 @@ Adds an Actions for a given User by their User ID
|
|||||||
curl --location --request POST 'https://app.formbricks.com/api/v1/client/<environment-id>/actions' \
|
curl --location --request POST 'https://app.formbricks.com/api/v1/client/<environment-id>/actions' \
|
||||||
--data-raw '{
|
--data-raw '{
|
||||||
"userId": "1",
|
"userId": "1",
|
||||||
"name": "new_action_v2"
|
"name": "new_action_v2",
|
||||||
|
"properties":{}
|
||||||
|
|
||||||
}'
|
}'
|
||||||
```
|
```
|
||||||
|
|
||||||
```json {{ title: 'Example Request Body' }}
|
```json {{ title: 'Example Request Body' }}
|
||||||
{
|
{
|
||||||
"userId": "1",
|
"userId": "1",
|
||||||
"name": "new_action_v2"
|
"name": "new_action_v3",
|
||||||
|
"properties":{}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -65,12 +65,7 @@ This set of API can be used to
|
|||||||
```json {{title:'200 Success'}}
|
```json {{title:'200 Success'}}
|
||||||
{
|
{
|
||||||
"data": {
|
"data": {
|
||||||
"id": "clp83r8uy000ceyqcbld2ebwj",
|
"id": "clphzz6oo00083zdmc7e0nwzi"
|
||||||
"createdAt": "2023-11-21T08:57:23.866Z",
|
|
||||||
"updatedAt": "2023-11-21T08:57:23.866Z",
|
|
||||||
"surveyId": "cloqzeuu70000z8khcirufo60",
|
|
||||||
"responseId": null,
|
|
||||||
"personId": "cloo25v3e0000z8ptskh030jd"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
@@ -81,7 +76,7 @@ This set of API can be used to
|
|||||||
"message": "Fields are missing or incorrectly formatted",
|
"message": "Fields are missing or incorrectly formatted",
|
||||||
"details": {
|
"details": {
|
||||||
"surveyId": "Required"
|
"surveyId": "Required"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
</CodeGroup>
|
</CodeGroup>
|
||||||
@@ -129,14 +124,7 @@ This set of API can be used to
|
|||||||
|
|
||||||
```json {{title:'200 Success'}}
|
```json {{title:'200 Success'}}
|
||||||
{
|
{
|
||||||
"data": {
|
"data": {}
|
||||||
"id": "clp83r8uy000ceyqcbld2ebwj",
|
|
||||||
"createdAt": "2023-11-21T08:57:23.866Z",
|
|
||||||
"updatedAt": "2023-11-21T09:05:27.285Z",
|
|
||||||
"surveyId": "cloqzeuu70000z8khcirufo60",
|
|
||||||
"responseId": null,
|
|
||||||
"personId": "cloo25v3e0000z8ptskh030jd"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -16,7 +16,9 @@ Checkout the [API Key Setup](/docs/api/management/api-key-setup) - to generate,
|
|||||||
|
|
||||||
The Public Client API is designed for the JavaScript SDK and does not require authentication. It's primarily used for creating persons, sessions, and responses within the Formbricks platform. This API is ideal for client-side interactions, as it doesn't expose sensitive information.
|
The Public Client API is designed for the JavaScript SDK and does not require authentication. It's primarily used for creating persons, sessions, and responses within the Formbricks platform. This API is ideal for client-side interactions, as it doesn't expose sensitive information.
|
||||||
|
|
||||||
|
- [Actions API](/docs/api/client/actions) - Create actions for a person
|
||||||
- [Displays API](/docs/api/client/displays) - Mark Survey as Displayed or Responded for a Person
|
- [Displays API](/docs/api/client/displays) - Mark Survey as Displayed or Responded for a Person
|
||||||
|
- [People API](/docs/api/client/people) - Create & update people (e.g. attributes)
|
||||||
- [Responses API](/docs/api/client/responses) - Create & update responses for a survey
|
- [Responses API](/docs/api/client/responses) - Create & update responses for a survey
|
||||||
|
|
||||||
## Management API
|
## Management API
|
||||||
|
|||||||
@@ -26,12 +26,6 @@ This set of API can be used to
|
|||||||
Create User with your own User ID
|
Create User with your own User ID
|
||||||
|
|
||||||
### Mandatory Request Body JSON Keys
|
### Mandatory Request Body JSON Keys
|
||||||
<Properties>
|
|
||||||
<Property name="environmentId" type="string">
|
|
||||||
Environment to create a person in
|
|
||||||
</Property>
|
|
||||||
</Properties>
|
|
||||||
|
|
||||||
<Properties>
|
<Properties>
|
||||||
<Property name="userId" type="string">
|
<Property name="userId" type="string">
|
||||||
User ID which you would like to identify the person with
|
User ID which you would like to identify the person with
|
||||||
@@ -48,7 +42,6 @@ This set of API can be used to
|
|||||||
'https://app.formbricks.com/api/v1/client/<environment-id>/people' \
|
'https://app.formbricks.com/api/v1/client/<environment-id>/people' \
|
||||||
-H 'Content-Type: application/json' \
|
-H 'Content-Type: application/json' \
|
||||||
-d '{
|
-d '{
|
||||||
"environmentId":"clonzr6vc0009z8md7y06hipl",
|
|
||||||
"userId":"docs_user"
|
"userId":"docs_user"
|
||||||
}'
|
}'
|
||||||
```
|
```
|
||||||
@@ -60,15 +53,7 @@ This set of API can be used to
|
|||||||
```json {{title:'200 Success'}}
|
```json {{title:'200 Success'}}
|
||||||
{
|
{
|
||||||
"data": {
|
"data": {
|
||||||
"status": "success",
|
"userId": "docs_user"
|
||||||
"person": {
|
|
||||||
"id": "clp861gvu0000v7wzr950fqad",
|
|
||||||
"userId": "docs_user",
|
|
||||||
"attributes": {},
|
|
||||||
"environmentId": "clonzr6vc0009z8md7y06hipl",
|
|
||||||
"createdAt": "2023-11-21T10:01:20.058Z",
|
|
||||||
"updatedAt": "2023-11-21T10:01:20.058Z"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
@@ -126,16 +111,7 @@ This set of API can be used to
|
|||||||
|
|
||||||
```json {{title:'200 Success'}}
|
```json {{title:'200 Success'}}
|
||||||
{
|
{
|
||||||
"data": {
|
"data": {}
|
||||||
"id": "clp868htl0003v7wzmy977m15",
|
|
||||||
"userId": "new_docs_user",
|
|
||||||
"attributes": {
|
|
||||||
"welcome_to": "formbricks"
|
|
||||||
},
|
|
||||||
"environmentId": "clonzr6vc0009z8md7y06hipl",
|
|
||||||
"createdAt": "2023-11-21T10:06:47.865Z",
|
|
||||||
"updatedAt": "2023-11-21T10:06:47.865Z"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -95,29 +95,6 @@ Add a new response to a survey.
|
|||||||
{
|
{
|
||||||
"data": {
|
"data": {
|
||||||
"id": "clp84xdld0002px36fkgue5ka",
|
"id": "clp84xdld0002px36fkgue5ka",
|
||||||
"createdAt": "2023-11-21T09:30:09.553Z",
|
|
||||||
"updatedAt": "2023-11-21T09:30:09.553Z",
|
|
||||||
"surveyId": "cloqzeuu70000z8khcirufo60",
|
|
||||||
"finished": true,
|
|
||||||
"data": {
|
|
||||||
"clfqjny0v0003yzgscnog1j9i": 10,
|
|
||||||
"clfqjtn8n0070yzgs6jgx9rog": "I love Formbricks"
|
|
||||||
},
|
|
||||||
"meta": {
|
|
||||||
"userAgent": {}
|
|
||||||
},
|
|
||||||
"personAttributes": {},
|
|
||||||
"singleUseId": null,
|
|
||||||
"person": {
|
|
||||||
"id": "cloo25v3e0000z8ptskh030jd",
|
|
||||||
"userId": "1",
|
|
||||||
"attributes": {},
|
|
||||||
"environmentId": "clonzr6vc0009z8md7y06hipl",
|
|
||||||
"createdAt": "2023-11-07T08:17:23.114Z",
|
|
||||||
"updatedAt": "2023-11-07T08:17:23.114Z"
|
|
||||||
},
|
|
||||||
"tags": [],
|
|
||||||
"notes": []
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
@@ -196,32 +173,7 @@ Update an existing response in a survey.
|
|||||||
|
|
||||||
```json {{ title: '200 Success' }}
|
```json {{ title: '200 Success' }}
|
||||||
{
|
{
|
||||||
"data": {
|
"data": {}
|
||||||
"id": "clp84xdld0002px36fkgue5ka",
|
|
||||||
"createdAt": "2023-11-21T09:30:09.553Z",
|
|
||||||
"updatedAt": "2023-11-21T09:35:38.357Z",
|
|
||||||
"surveyId": "cloqzeuu70000z8khcirufo60",
|
|
||||||
"finished": false,
|
|
||||||
"data": {
|
|
||||||
"clfqjny0v0003yzgscnog1j9i": 10,
|
|
||||||
"clfqjtn8n0070yzgs6jgx9rog": "I love Formbricks"
|
|
||||||
},
|
|
||||||
"meta": {
|
|
||||||
"userAgent": {}
|
|
||||||
},
|
|
||||||
"personAttributes": {},
|
|
||||||
"singleUseId": null,
|
|
||||||
"person": {
|
|
||||||
"id": "cloo25v3e0000z8ptskh030jd",
|
|
||||||
"userId": "1",
|
|
||||||
"attributes": {},
|
|
||||||
"environmentId": "clonzr6vc0009z8md7y06hipl",
|
|
||||||
"createdAt": "2023-11-07T08:17:23.114Z",
|
|
||||||
"updatedAt": "2023-11-07T08:17:23.114Z"
|
|
||||||
},
|
|
||||||
"tags": [],
|
|
||||||
"notes": []
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,6 @@
|
|||||||
import { Fence } from "@/components/shared/Fence";
|
import { Fence } from "@/components/shared/Fence";
|
||||||
|
import {generateManagementApiMetadata} from "@/lib/utils"
|
||||||
export const metadata = {
|
export const metadata = generateManagementApiMetadata("Action Class",["Fetch","Create","Delete"])
|
||||||
title: "Formbricks People API: Fetch or Create Person Overview",
|
|
||||||
description:
|
|
||||||
"Dive into Formbricks' People API within the Public Client API suite, designed to work without authentication requirements. Seamlessly fetch or create a person by their userId and environmentId, optimizing client-side interactions while maintaining data privacy.",
|
|
||||||
};
|
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,7 @@
|
|||||||
import { Fence } from "@/components/shared/Fence";
|
import { Fence } from "@/components/shared/Fence";
|
||||||
|
import {generateManagementApiMetadata} from "@/lib/utils"
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = generateManagementApiMetadata("Attribute Class",["Fetch","Create","Delete"])
|
||||||
title: "Formbricks People API: Fetch or Create Person Overview",
|
|
||||||
description:
|
|
||||||
"Dive into Formbricks' People API within the Public Client API suite, designed to work without authentication requirements. Seamlessly fetch or create a person by their userId and environmentId, optimizing client-side interAttributes while maintaining data privacy.",
|
|
||||||
};
|
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import { Fence } from "@/components/shared/Fence";
|
import { Fence } from "@/components/shared/Fence";
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = {
|
||||||
title: "Formbricks People API: Fetch or Create Person Overview",
|
title: "Formbricks Me API: Fetch your environment details",
|
||||||
description:
|
description:
|
||||||
"Dive into Formbricks' People API within the Public Client API suite, designed to work without authentication requirements. Seamlessly fetch or create a person by their userId and environmentId, optimizing client-side interactions while maintaining data privacy.",
|
"Dive into Formbricks' Me API within the Public Client API suite. Seamlessly fetch your own current environment details.",
|
||||||
};
|
};
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|||||||
@@ -1,10 +1,8 @@
|
|||||||
import { Fence } from "@/components/shared/Fence";
|
import { Fence } from "@/components/shared/Fence";
|
||||||
|
import {generateManagementApiMetadata} from "@/lib/utils"
|
||||||
|
|
||||||
|
export const metadata = generateManagementApiMetadata("People",["Fetch","Delete"])
|
||||||
|
|
||||||
export const metadata = {
|
|
||||||
title: "Formbricks People API: Fetch or Create Person Overview",
|
|
||||||
description:
|
|
||||||
"Dive into Formbricks' People API within the Public Client API suite, designed to work without authentication requirements. Seamlessly fetch or create a person by their userId and environmentId, optimizing client-side interactions while maintaining data privacy.",
|
|
||||||
};
|
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,7 @@
|
|||||||
import { Fence } from "@/components/shared/Fence";
|
import { Fence } from "@/components/shared/Fence";
|
||||||
|
import {generateManagementApiMetadata} from "@/lib/utils"
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = generateManagementApiMetadata("Responses",["Fetch","Delete"])
|
||||||
title: "Formbricks Responses API Documentation - Manage Your Survey Data Seamlessly",
|
|
||||||
description:
|
|
||||||
"Unlock the full potential of Formbricks' Responses API. From fetching to updating survey responses, our comprehensive guide helps you integrate and manage survey data efficiently without compromising security. Ideal for client-side interactions.",
|
|
||||||
};
|
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|
||||||
@@ -221,7 +218,7 @@ This set of API can be used to
|
|||||||
<CodeGroup title="Request" tag="DELETE" label="/api/v1/client/responses/<response-id>">
|
<CodeGroup title="Request" tag="DELETE" label="/api/v1/client/responses/<response-id>">
|
||||||
|
|
||||||
```bash {{ title: 'cURL' }}
|
```bash {{ title: 'cURL' }}
|
||||||
curl -X DELETE https://app.formbricks.com/api/v1/management/resposnes/<response-id> \
|
curl -X DELETE https://app.formbricks.com/api/v1/management/responses/<response-id> \
|
||||||
--header 'x-api-key: <your-api-key>'
|
--header 'x-api-key: <your-api-key>'
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,7 @@
|
|||||||
import { Fence } from "@/components/shared/Fence";
|
import { Fence } from "@/components/shared/Fence";
|
||||||
|
import {generateManagementApiMetadata} from "@/lib/utils"
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = generateManagementApiMetadata("Surveys",["Fetch","Create","Update","Delete"])
|
||||||
title: "Formbricks Surveys API Documentation - How to Retrieve All Surveys",
|
|
||||||
description:
|
|
||||||
"Explore the comprehensive guide to the Formbricks Surveys API. Learn how to effectively retrieve all the surveys in your environment with the necessary headers and API key setup. Includes sample request and response formats.",
|
|
||||||
};
|
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
export const metadata = {
|
import {generateManagementApiMetadata} from "@/lib/utils"
|
||||||
title: "Formbricks Webhook API Documentation - List, Retrieve, Create, and Delete Webhooks",
|
|
||||||
description:
|
export const metadata = generateManagementApiMetadata("Webhook",["Fetch","Create","Delete"])
|
||||||
"Explore the comprehensive guide to the Formbricks Webhooks API. This is all you need to interact and play with the Formbricks Webhooks and integrate them into any third party app of your choice",
|
|
||||||
};
|
|
||||||
|
|
||||||
#### Management API
|
#### Management API
|
||||||
|
|
||||||
|
|||||||
@@ -10,9 +10,31 @@ export const metadata = {
|
|||||||
|
|
||||||
One way to send attributes to Formbricks is in your code. In Formbricks, there are two special attributes for [user identification](/docs/attributes/identify-users)(user ID & email) and custom attributes. An example:
|
One way to send attributes to Formbricks is in your code. In Formbricks, there are two special attributes for [user identification](/docs/attributes/identify-users)(user ID & email) and custom attributes. An example:
|
||||||
|
|
||||||
## Setting Custom User Attributes
|
## Setting during Initialization
|
||||||
|
|
||||||
|
It's recommended to set custom user attributes directly during the initialization of Formbricks for better user identification.
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Set custom attributes during initialization">
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
formbricks.init({
|
||||||
|
environmentId: "<environment-id>",
|
||||||
|
apiHost: "<api-host>",
|
||||||
|
userId: "<user_id>",
|
||||||
|
attributes: {
|
||||||
|
plan: "free",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
## Setting independently
|
||||||
|
|
||||||
|
You can use the setAttribute function to set any custom attribute for the user (e.g. name, plan, etc.) anywhere in the user journey. Formbricks maintains a state of the current user inside the browser and makes sure attributes aren't sent to the backend twice.
|
||||||
|
|
||||||
You can use the setAttribute function to set any custom attribute for the user (e.g. name, plan, etc.):
|
|
||||||
<Col>
|
<Col>
|
||||||
<CodeGroup title="Setting Plan to Pro">
|
<CodeGroup title="Setting Plan to Pro">
|
||||||
|
|
||||||
|
|||||||
@@ -29,6 +29,29 @@ formbricks.init({
|
|||||||
|
|
||||||
</CodeGroup>
|
</CodeGroup>
|
||||||
</Col>
|
</Col>
|
||||||
|
|
||||||
|
## Enhanced Initialization with User Attributes
|
||||||
|
|
||||||
|
In addition to setting the `userId`, Formbricks allows you to set user attributes right at the initialization. This ensures that your user data is seamlessly integrated from the start. Here's how you can include user attributes in the `init()` function:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Enhanced Initialization with User Attributes">
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
formbricks.init({
|
||||||
|
environmentId: "<environment-id>",
|
||||||
|
apiHost: "<api-host>",
|
||||||
|
userId: "<user_id>",
|
||||||
|
attributes: {
|
||||||
|
// your custom attributes
|
||||||
|
Plan: "premium",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
## Setting User Email
|
## Setting User Email
|
||||||
|
|
||||||
The `userId` is the main identifier used in Formbricks and user identification is only enabled when it is set. In addition to the userId you can also set attributes that describes the user better. The email address can be set using the setEmail function:
|
The `userId` is the main identifier used in Formbricks and user identification is only enabled when it is set. In addition to the userId you can also set attributes that describes the user better. The email address can be set using the setEmail function:
|
||||||
|
|||||||
@@ -1,21 +1,22 @@
|
|||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import GitpodPorts from "./gitpod/ports.webp";
|
|
||||||
import GitpodAuth from "./gitpod/auth.webp";
|
import GitpodAuth from "./gitpod/auth.webp";
|
||||||
import GitpodNewWorkspace from "./gitpod/new-workspace.webp";
|
import GitpodNewWorkspace from "./gitpod/new-workspace.webp";
|
||||||
|
import GitpodPorts from "./gitpod/ports.webp";
|
||||||
import GitpodPreparing from "./gitpod/preparing.webp";
|
import GitpodPreparing from "./gitpod/preparing.webp";
|
||||||
import GitpodRunning from "./gitpod/running.webp";
|
import GitpodRunning from "./gitpod/running.webp";
|
||||||
|
|
||||||
import GithubCodespaceNew from "./github-codespaces/new.webp";
|
|
||||||
import GithubCodespaceLoading from "./github-codespaces/loading.webp";
|
|
||||||
import GithubCodespaceEnvFile from "./github-codespaces/env.webp";
|
import GithubCodespaceEnvFile from "./github-codespaces/env.webp";
|
||||||
import GithubCodespaceTerminal from "./github-codespaces/terminal.webp";
|
import GithubCodespaceLoading from "./github-codespaces/loading.webp";
|
||||||
import GithubCodespaceRun from "./github-codespaces/run.webp";
|
import GithubCodespaceNew from "./github-codespaces/new.webp";
|
||||||
import GithubCodespacePorts from "./github-codespaces/ports.webp";
|
import GithubCodespacePorts from "./github-codespaces/ports.webp";
|
||||||
|
import GithubCodespaceRun from "./github-codespaces/run.webp";
|
||||||
|
import GithubCodespaceTerminal from "./github-codespaces/terminal.webp";
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = {
|
||||||
title: "Formbricks Development Setup: Complete Guide to Local Environment Configuration for Dev",
|
title: "Formbricks Development Setup: Complete Guide to Local Environment Configuration for Dev",
|
||||||
description:
|
description:
|
||||||
"Step-by-step guide to setting up a development environment for Formbricks. We officially support Gitpod and Github Codespaces for quick setup.",
|
"Step-by-step guide to setting up a development environment for Formbricks. We officially support Gitpod and Github Codespaces for quick setup. Our advanced users can also setup Formbricks locally on their machine.",
|
||||||
};
|
};
|
||||||
|
|
||||||
#### Contributing
|
#### Contributing
|
||||||
@@ -25,28 +26,29 @@ export const metadata = {
|
|||||||
We currently officially support the below methods to set up your development environment for Formbricks.
|
We currently officially support the below methods to set up your development environment for Formbricks.
|
||||||
|
|
||||||
<Note>
|
<Note>
|
||||||
Both the below cloud IDEs have a **generous free tier** to explore and develop! But make sure to
|
Both the below cloud IDEs have a **generous free tier** to explore and develop! But make sure to not overuse
|
||||||
not overuse the machines as Formbricks will not be responsible for any charges incurred.
|
the machines as Formbricks will not be responsible for any charges incurred.
|
||||||
</Note>
|
</Note>
|
||||||
|
|
||||||
### [GitPod](#gitpod)
|
### [GitPod](#gitpod)
|
||||||
This will open a fully configured workspace in your browser with all the necessary dependencies already installed. Click the button below to open this project in Gitpod:
|
|
||||||
|
This will open a fully configured workspace in your browser with all the necessary dependencies already installed. Click the button below to open this project in Gitpod. For a detailed guide, visit the [Gitpod Setup Guide](#gitpod-guide) section below.
|
||||||
|
|
||||||
[](https://gitpod.io/#https://Github.com/formbricks/formbricks)
|
[](https://gitpod.io/#https://Github.com/formbricks/formbricks)
|
||||||
|
|
||||||
For a detailed guide, visit the [Gitpod Setup Guide](#gitpod-guide) section below.
|
|
||||||
|
|
||||||
### [Github Codespaces](#Github-codespaces)
|
### [Github Codespaces](#Github-codespaces)
|
||||||
This will open a Github VSCode Interface on the cloud for you. This setup will have the Formbricks codebase and all the dependencies installed. Click the button below to configure your instance and open the project in Github Codespaces:
|
|
||||||
|
This will open a Github VSCode Interface on the cloud for you. This setup will have the Formbricks codebase and all the dependencies installed. Click the button below to configure your instance and open the project in Github Codespaces. For a detailed guide, visit the [Github Codespaces Setup Guide](#github-codespaces-guide) section below.
|
||||||
|
|
||||||
[](https://Github.com/codespaces/new?machine=standardLinux32gb&repo=500289888&ref=main&devcontainer_path=.devcontainer%2Fdevcontainer.json&location=EastUs2)
|
[](https://Github.com/codespaces/new?machine=standardLinux32gb&repo=500289888&ref=main&devcontainer_path=.devcontainer%2Fdevcontainer.json&location=EastUs2)
|
||||||
|
|
||||||
|
### [Local Machine](#local-machine-setup)
|
||||||
|
|
||||||
For a detailed guide, visit the [Github Codespaces Setup Guide](#github-codespaces-guide) section below.
|
This will install the Formbricks codebase and all the dependencies on your local machine. Note that this method is recommended **only for advanced users**. If you're an advanced user, access the steps for [Local Machine Setup here](#local-machine-setup).
|
||||||
|
|
||||||
<Note>
|
<Note>
|
||||||
For a smooth experience, we suggest the above recommended methods.
|
For a smooth experience, we suggest the above cloud IDE methods. Assistance with setup issues on your local
|
||||||
Assistance with setup issues on your local machine may be limited due to varying factors like OS and permissions.
|
machine may be limited due to varying factors like OS and permissions.
|
||||||
</Note>
|
</Note>
|
||||||
|
|
||||||
## Gitpod Guide
|
## Gitpod Guide
|
||||||
@@ -54,16 +56,16 @@ Assistance with setup issues on your local machine may be limited due to varying
|
|||||||
**Building custom image for the workspace:**
|
**Building custom image for the workspace:**
|
||||||
- This includes : Installing `yq` and `turbo` globally before the workspace starts. This is accomplished within the `.gitpod.Dockerfile` along with starting upon a base custom image building on [workspace-full](https://hub.docker.com/r/gitpod/workspace-full/dockerfile).
|
- This includes : Installing `yq` and `turbo` globally before the workspace starts. This is accomplished within the `.gitpod.Dockerfile` along with starting upon a base custom image building on [workspace-full](https://hub.docker.com/r/gitpod/workspace-full/dockerfile).
|
||||||
|
|
||||||
**Initialization of Formbricks:**
|
**Initialization of Formbricks:**
|
||||||
- During the prebuilds phase, we initialize Formbricks by performing the following tasks:
|
- During the prebuilds phase, we initialize Formbricks by performing the following tasks:
|
||||||
1. Setting up environment variables.
|
1. Setting up environment variables.
|
||||||
2. Installing monorepo dependencies.
|
2. Installing monorepo dependencies.
|
||||||
3. Installing Docker images by extracting them from the `packages/database/docker-compose.yml` file.
|
3. Installing Docker images by extracting them from the `packages/database/docker-compose.yml` file.
|
||||||
4. Building the @formbricks/js component.
|
4. Building the @formbricks/js component.
|
||||||
- When the workspace starts:
|
- When the workspace starts:
|
||||||
1. Wait for the web and demo apps to launch on Gitpod. This automatically opens the `apps/demo/.env` file. Utilize dynamic localhost URLs (e.g., `localhost:3000` for signup and `localhost:8025` for email confirmation) to configure `NEXT_PUBLIC_FORMBRICKS_ENVIRONMENT_ID`. After creating your account and finding the `ID` in the URL at `localhost:3000`, replace `YOUR_ENVIRONMENT_ID` in the `.env` file located in `app/demo`.
|
1. Wait for the web and demo apps to launch on Gitpod. This automatically opens the `apps/demo/.env` file. Utilize dynamic localhost URLs (e.g., `localhost:3000` for signup and `localhost:8025` for email confirmation) to configure `NEXT_PUBLIC_FORMBRICKS_ENVIRONMENT_ID`. After creating your account and finding the `ID` in the URL at `localhost:3000`, replace `YOUR_ENVIRONMENT_ID` in the `.env` file located in `app/demo`.
|
||||||
|
|
||||||
**Web Component Initialization:**
|
**Web Component Initialization:**
|
||||||
- we initialize the @formbricks/web component during prebuilds. This involves:
|
- we initialize the @formbricks/web component during prebuilds. This involves:
|
||||||
1. Installing build dependencies for the `@formbricks/web#go` task from turbo.json in prebuilds to save time.
|
1. Installing build dependencies for the `@formbricks/web#go` task from turbo.json in prebuilds to save time.
|
||||||
2. Starting PostgreSQL and Mailhog containers for running migrations in prebuilds.
|
2. Starting PostgreSQL and Mailhog containers for running migrations in prebuilds.
|
||||||
@@ -73,95 +75,131 @@ Assistance with setup issues on your local machine may be limited due to varying
|
|||||||
2. Replacing `NEXT_PUBLIC_WEBAPP_URL` and `NEXTAUTH_URL` to take in Gitpod URL's ports when running on VSCode browser.
|
2. Replacing `NEXT_PUBLIC_WEBAPP_URL` and `NEXTAUTH_URL` to take in Gitpod URL's ports when running on VSCode browser.
|
||||||
3. Starting the `@formbricks/web` dev environment.
|
3. Starting the `@formbricks/web` dev environment.
|
||||||
|
|
||||||
**Demo Component Initialization:**
|
**Demo Component Initialization:**
|
||||||
- Similar to the web component, the demo component is also initialized during prebuilds. This includes:
|
- Similar to the web component, the demo component is also initialized during prebuilds. This includes:
|
||||||
1. Installing build dependencies for the `formbricks/demo#go` task from turbo.json in prebuilds to save time.
|
1. Installing build dependencies for the `formbricks/demo#go` task from turbo.json in prebuilds to save time.
|
||||||
2. Caching hits and replaying builds from the `@formbricks/js` component.
|
2. Caching hits and replaying builds from the `@formbricks/js` component.
|
||||||
- When the workspace starts:
|
- When the workspace starts:
|
||||||
1. Initializing environment variables.
|
1. Initializing environment variables.
|
||||||
2. Replaces `NEXT_PUBLIC_FORMBRICKS_API_HOST` to take in Gitpod URL's ports when running on VSCode browser.
|
2. Replaces `NEXT_PUBLIC_FORMBRICKS_API_HOST` to take in Gitpod URL's ports when running on VSCode browser.
|
||||||
3. Starting the `@formbricks/demo` dev environment.
|
3. Starting the `@formbricks/demo` dev environment.
|
||||||
|
|
||||||
**Github Prebuilds Configuration:**
|
**Github Prebuilds Configuration:**
|
||||||
- This configures Github Prebuilds for the master branch, pull requests, and adding comments. This helps automate the prebuild process for the specified branches and actions.
|
- This configures Github Prebuilds for the master branch, pull requests, and adding comments. This helps automate the prebuild process for the specified branches and actions.
|
||||||
|
|
||||||
**VSCode Extensions:**
|
**VSCode Extensions:**
|
||||||
- This includes a list of VSCode extensions that are added to the configuration when using Gitpod. These extensions can enhance the development experience within Gitpod.
|
- This includes a list of VSCode extensions that are added to the configuration when using Gitpod. These extensions can enhance the development experience within Gitpod.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### 1. Browser Redirection
|
### 1. Browser Redirection
|
||||||
|
|
||||||
After clicking the one-click setup button, Gitpod will open a new tab or window. Please ensure that your browser allows redirection to successfully access the services:
|
After clicking the one-click setup button, Gitpod will open a new tab or window. Please ensure that your browser allows redirection to successfully access the services:
|
||||||
|
|
||||||
### 2. Authorizing in Gitpod
|
### 2. Authorizing in Gitpod
|
||||||
<Image src={GitpodAuth} alt="Gitpod Auth Page" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
|
||||||
- This is the Gitpod Authentication Page. It appears when you click the "Open in GitPod" button and Gitpod needs to authenticate your access to the workspace. Click on 'Continue With Github' to authorize your GitPod session.
|
<Image src={GitpodAuth} alt="Gitpod Auth Page" quality="100" className="max-w-full rounded-lg sm:max-w-3xl" />-
|
||||||
|
This is the Gitpod Authentication Page. It appears when you click the "Open in GitPod" button and Gitpod needs
|
||||||
|
to authenticate your access to the workspace. Click on 'Continue With Github' to authorize your GitPod session.
|
||||||
|
|
||||||
### 3. Creating a New Workspace
|
### 3. Creating a New Workspace
|
||||||
<Image src={GitpodNewWorkspace} alt="Gitpod New workspace Page" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
|
||||||
- After authentication, Gitpod asks to create a new workspace for you. This page displays the configurations of your workspace.
|
|
||||||
- You can use either choose either VS Code Browser or VS Code Desktop editor with the 'Standard Class' for your workspace class.
|
|
||||||
- If you opt for the VS Code Desktop, follow the following steps
|
|
||||||
1. Gitpod will prompt you to grant access to the VSCode app. Once approved, install the GitPod extension from the VSCode Marketplace and follow the prompts to authorize the integration.
|
|
||||||
2. Change the `WEBAPP_URL` and the `NEXTAUTH_URL` to `https://localhost:3000`
|
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={GitpodNewWorkspace}
|
||||||
|
alt="Gitpod New workspace Page"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
- After authentication, Gitpod asks to create a new workspace for you. This page displays the configurations of
|
||||||
|
your workspace. - You can use either choose either VS Code Browser or VS Code Desktop editor with the 'Standard
|
||||||
|
Class' for your workspace class. - If you opt for the VS Code Desktop, follow the following steps 1. Gitpod will
|
||||||
|
prompt you to grant access to the VSCode app. Once approved, install the GitPod extension from the VSCode Marketplace
|
||||||
|
and follow the prompts to authorize the integration. 2. Change the `WEBAPP_URL` and the `NEXTAUTH_URL` to `https://localhost:3000`
|
||||||
|
|
||||||
### 4. Gitpod preparing the created Workspace
|
### 4. Gitpod preparing the created Workspace
|
||||||
<Image src={GitpodPreparing} alt="Gitpod Preparing workspace Page" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
|
||||||
- Gitpod is preparing your workspace with all the necessary dependencies and configurations. You will see this page while Gitpod sets up your development environment.
|
<Image
|
||||||
|
src={GitpodPreparing}
|
||||||
|
alt="Gitpod Preparing workspace Page"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
- Gitpod is preparing your workspace with all the necessary dependencies and configurations. You will see this
|
||||||
|
page while Gitpod sets up your development environment.
|
||||||
|
|
||||||
### 5. Gitpod running the Workspace
|
### 5. Gitpod running the Workspace
|
||||||
<Image src={GitpodRunning} alt="Gitpod Running Workspace Page" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
|
||||||
- Once the workspace is fully prepared, voila, it enters the running state. You can start working on your project in this environment.
|
<Image
|
||||||
|
src={GitpodRunning}
|
||||||
|
alt="Gitpod Running Workspace Page"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
- Once the workspace is fully prepared, voila, it enters the running state. You can start working on your project
|
||||||
|
in this environment.
|
||||||
|
|
||||||
### Ports and Services
|
### Ports and Services
|
||||||
|
|
||||||
Here are the ports and corresponding URLs for the services within your Gitpod environment:
|
Here are the ports and corresponding URLs for the services within your Gitpod environment:
|
||||||
|
|
||||||
- **Port 3000**:
|
- **Port 3000**:
|
||||||
- **Service**: Demo App
|
|
||||||
- **Description**: This port hosts the demo application of your project. You can access and interact with your application's demo by navigating to this port.
|
- **Service**: Demo App
|
||||||
|
- **Description**: This port hosts the demo application of your project. You can access and interact with your application's demo by navigating to this port.
|
||||||
|
|
||||||
- **Port 3001**:
|
- **Port 3001**:
|
||||||
- **Service**: Formbricks website
|
|
||||||
- **Description**: This port hosts the [Formbricks](https://formbricks.com) website, which contains documents, pricing, blogs, best practices, and concierge service.
|
- **Service**: Formbricks website
|
||||||
|
- **Description**: This port hosts the [Formbricks](https://formbricks.com) website, which contains documents, pricing, blogs, best practices, and concierge service.
|
||||||
|
|
||||||
- **Port 3002**:
|
- **Port 3002**:
|
||||||
- **Service**: Formbricks In-product Survey Demo App
|
|
||||||
- **Description**: This app helps you test your in-app surveys. You can create and test user actions, create and update user attributes, etc.
|
- **Service**: Formbricks In-product Survey Demo App
|
||||||
|
- **Description**: This app helps you test your in-app surveys. You can create and test user actions, create and update user attributes, etc.
|
||||||
|
|
||||||
- **Port 5432**:
|
- **Port 5432**:
|
||||||
- **Service**: PostgreSQL Database Server
|
|
||||||
- **Description**: The PostgreSQL DB is hosted on this port.
|
- **Service**: PostgreSQL Database Server
|
||||||
|
- **Description**: The PostgreSQL DB is hosted on this port.
|
||||||
|
|
||||||
- **Port 1025**:
|
- **Port 1025**:
|
||||||
- **Service**: SMPT server
|
|
||||||
- **Description**: SMTP Server for sending and receiving email messages. This server is responsible for handling email communication.
|
- **Service**: SMPT server
|
||||||
|
- **Description**: SMTP Server for sending and receiving email messages. This server is responsible for handling email communication.
|
||||||
|
|
||||||
- **Port 8025**:
|
- **Port 8025**:
|
||||||
- **Service**: Mailhog
|
- **Service**: Mailhog
|
||||||
|
|
||||||
### Accessing port URLs
|
### Accessing port URLs
|
||||||
1. **Direct URL Composition**:
|
|
||||||
- You can access the dedicated port URL by pre-pending the port number to the workspace URL.
|
|
||||||
- For example, if you want to access port 3000, you can use the URL format: `3000-yourworkspace.ws-eu45.gitpod.io`.
|
|
||||||
|
|
||||||
2. **Using [gp CLI](https://www.gitpod.io/docs/references/gitpod-cli)**:
|
1. **Direct URL Composition**:
|
||||||
- Gitpod provides a convenient command, `gp url`, to quickly retrieve the URL for a specific port.
|
|
||||||
- Simply use the command followed by the desired port number. For example, to get the URL for port 3000, run: `gp url 3000`.
|
|
||||||
|
|
||||||
3. **Listing All Open Port URLs**:
|
- You can access the dedicated port URL by pre-pending the port number to the workspace URL.
|
||||||
- If you prefer to see a list of all open port URLs at once, you can use the `gp ports list` command.
|
- For example, if you want to access port 3000, you can use the URL format: `3000-yourworkspace.ws-eu45.gitpod.io`.
|
||||||
- Running this command will display a list of ports along with their corresponding URLs.
|
|
||||||
|
|
||||||
4. **Viewing All Ports in Panel**:
|
2. **Using [gp CLI](https://www.gitpod.io/docs/references/gitpod-cli)**:
|
||||||
- Gitpod also offers a user-friendly 'Ports' tab in the Gitpod panel.
|
|
||||||
- Click on the 'Ports' tab to view a list of all open ports and their respective URLs.
|
|
||||||
|
|
||||||
<Image src={GitpodPorts} alt="Gitpod Ports tab" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
- Gitpod provides a convenient command, `gp url`, to quickly retrieve the URL for a specific port.
|
||||||
|
- Simply use the command followed by the desired port number. For example, to get the URL for port 3000, run: `gp url 3000`.
|
||||||
|
|
||||||
These URLs and port numbers represent various services and endpoints within your Gitpod environment. You can access and interact with these services by the Port URL for the respective service.
|
3. **Listing All Open Port URLs**:
|
||||||
|
|
||||||
|
- If you prefer to see a list of all open port URLs at once, you can use the `gp ports list` command.
|
||||||
|
- Running this command will display a list of ports along with their corresponding URLs.
|
||||||
|
|
||||||
|
4. **Viewing All Ports in Panel**:
|
||||||
|
|
||||||
|
- Gitpod also offers a user-friendly 'Ports' tab in the Gitpod panel.
|
||||||
|
- Click on the 'Ports' tab to view a list of all open ports and their respective URLs.
|
||||||
|
|
||||||
|
{" "}
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={GitpodPorts}
|
||||||
|
alt="Gitpod Ports tab"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
These URLs and port numbers represent various services and endpoints within your Gitpod environment. You can access and interact with these services by the Port URL for the respective service.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -169,21 +207,41 @@ Here are the ports and corresponding URLs for the services within your Gitpod en
|
|||||||
|
|
||||||
1. After clicking the one-click setup button, you will be redirected to the Github Codespaces page. Review the configuration and click on the 'Create Codespace' button to create a new Codespace.
|
1. After clicking the one-click setup button, you will be redirected to the Github Codespaces page. Review the configuration and click on the 'Create Codespace' button to create a new Codespace.
|
||||||
|
|
||||||
<Image src={GithubCodespaceNew} alt="New Github Codespace" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
<Image
|
||||||
|
src={GithubCodespaceNew}
|
||||||
|
alt="New Github Codespace"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
2. This will start loading the Codespace. Keep in mind this might take a few minutes to complete depending on your internet connection and the instance availability.
|
2. This will start loading the Codespace. Keep in mind this might take a few minutes to complete depending on your internet connection and the instance availability.
|
||||||
|
|
||||||
<Image src={GithubCodespaceLoading} alt="Loading Github Codespace" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
<Image
|
||||||
|
src={GithubCodespaceLoading}
|
||||||
|
alt="Loading Github Codespace"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
3. Once the Codespace is loaded, you will be redirected to the VSCode editor. You can start working on your project in this environment.
|
3. Once the Codespace is loaded, you will be redirected to the VSCode editor. You can start working on your project in this environment.
|
||||||
|
|
||||||
4. Make the changes you want to, and now, to run the app, we first need to configure the .env file. Copy the .env.example and edit the variables as mentioned in the file itself.
|
4. Make the changes you want to, and now, to run the app, we first need to configure the .env file. Copy the .env.example and edit the variables as mentioned in the file itself.
|
||||||
|
|
||||||
<Image src={GithubCodespaceEnvFile} alt="Github Codespace Env File" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
<Image
|
||||||
|
src={GithubCodespaceEnvFile}
|
||||||
|
alt="Github Codespace Env File"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
5. Once you have configured the .env, it's now time to run the app and see the changes. Lets open the terminal first
|
5. Once you have configured the .env, it's now time to run the app and see the changes. Lets open the terminal first
|
||||||
|
|
||||||
<Image src={GithubCodespaceTerminal} alt="Github Codespace Open Terminal" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
<Image
|
||||||
|
src={GithubCodespaceTerminal}
|
||||||
|
alt="Github Codespace Open Terminal"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
6. Now, run the following command to run the app
|
6. Now, run the following command to run the app
|
||||||
|
|
||||||
@@ -197,7 +255,12 @@ pnpm dev
|
|||||||
</CodeGroup>
|
</CodeGroup>
|
||||||
</Col>
|
</Col>
|
||||||
|
|
||||||
<Image src={GithubCodespaceRun} alt="Run on Github Codespace" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
<Image
|
||||||
|
src={GithubCodespaceRun}
|
||||||
|
alt="Run on Github Codespace"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
7. Monitor the logs in the terminal and once you see the following, you are good to go!
|
7. Monitor the logs in the terminal and once you see the following, you are good to go!
|
||||||
|
|
||||||
@@ -210,7 +273,7 @@ pnpm dev
|
|||||||
@formbricks/web:dev: - Environments: .env
|
@formbricks/web:dev: - Environments: .env
|
||||||
@formbricks/web:dev: - Experiments (use at your own risk):
|
@formbricks/web:dev: - Experiments (use at your own risk):
|
||||||
@formbricks/web:dev: · serverActions
|
@formbricks/web:dev: · serverActions
|
||||||
@formbricks/web:dev:
|
@formbricks/web:dev:
|
||||||
@formbricks/web:dev: ✓ Ready in 9.4s
|
@formbricks/web:dev: ✓ Ready in 9.4s
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -219,10 +282,123 @@ pnpm dev
|
|||||||
|
|
||||||
8. Right next to the Terminal, you will see a **Ports** tab, click on it to see the ports and their respective URLs. Now access the Forwarded Address for port 3000 and you should be able to visit your Formbricks App!
|
8. Right next to the Terminal, you will see a **Ports** tab, click on it to see the ports and their respective URLs. Now access the Forwarded Address for port 3000 and you should be able to visit your Formbricks App!
|
||||||
|
|
||||||
<Image src={GithubCodespacePorts} alt="Github Codespace Ports" quality="100" className="rounded-lg max-w-full sm:max-w-3xl" />
|
<Image
|
||||||
|
src={GithubCodespacePorts}
|
||||||
|
alt="Github Codespace Ports"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
Now make the changes you want to and see them live in action!
|
Now make the changes you want to and see them live in action!
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
Still can’t figure it out? Join our [Discord](https://discord.com/invite/3YFcABF2Ts)!
|
## Local Machine Setup
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
The below only works for **Mac**, **Linux** & **WSL2** on Windows (not on pure Windows)!
|
||||||
|
|
||||||
|
This method is recommended **only for advanced users** & we won't be able to provide official support for this.
|
||||||
|
|
||||||
|
</Note>
|
||||||
|
|
||||||
|
To get the project running locally on your machine you need to have the following development tools installed:
|
||||||
|
|
||||||
|
- Node.JS (we recommend v20)
|
||||||
|
- [pnpm](https://pnpm.io/)
|
||||||
|
- [Docker](https://www.docker.com/) (to run PostgreSQL / MailHog)
|
||||||
|
|
||||||
|
1. Clone the project & move into the directory:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Git clone Formbricks monorepo">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/formbricks/formbricks && cd formbricks
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
2. Install Node.JS packages via pnpm. Don't have pnpm? Get it [here](https://pnpm.io/installation)
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Install dependencies via pnpm">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm install
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
3. Create a `.env` file based on `.env.example`. It's already preset to work with the local development setup but you can also change values if needed.
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Define environment variables">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cp .env.example .env
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
4. Generate & set some secret values mandatory for the ENCRYPTION_KEY & NEXTAUTH_SECRET in the .env file. You can use the following command to generate the random string of required length:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Set value of ENCRYPTION_KEY">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
sed -i '/^ENCRYPTION_KEY=/c\ENCRYPTION_KEY='$(openssl rand -hex 32) .env
|
||||||
|
sed -i '/^NEXTAUTH_SECRET=/c\NEXTAUTH_SECRET='$(openssl rand -hex 32) .env
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
5. Make sure you have [`Docker`](https://docs.docker.com/compose/) & [`docker-compose`](https://docs.docker.com/compose/) installed and running on your machine. Then run the following command to start the Formbricks dev setup:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Start Formbricks Dev Setup">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm go
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
This starts the Formbricks main app (plus all its dependencies) as well as the following services using Docker:
|
||||||
|
|
||||||
|
- a `postgres` container for hosting your database,
|
||||||
|
- a `mailhog` container that acts as a mock SMTP server and shows received mails in a web UI (forwarded to your host's `localhost:8025`)
|
||||||
|
- Demo App at [http://localhost:3002](http://localhost:3002)
|
||||||
|
- Landing Page at [http://localhost:3001](http://localhost:3001)
|
||||||
|
|
||||||
|
**You can now access the Formbricks app on [http://localhost:3000](http://localhost:3000)**. You will be automatically redirected to the login. To use your local installation of formbricks, create a new account.
|
||||||
|
|
||||||
|
{" "}
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
A fresh setup does not have a default account. Please create a new account and proceed accordingly.
|
||||||
|
</Note>
|
||||||
|
|
||||||
|
For viewing the emails sent by the system, you can access mailhog at [http://localhost:8025](http://localhost:8025)
|
||||||
|
|
||||||
|
### Build
|
||||||
|
|
||||||
|
To build all apps and packages and check for build errors, run the following command:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Build Formbricks stack">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
pnpm build
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
Still can’t figure it out? Join our [Discord](https://discord.com/invite/3YFcABF2Ts)!
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ All you need to do is copy a `<script>` tag to your HTML head, and that’s abou
|
|||||||
```html {{ title: 'index.html' }}
|
```html {{ title: 'index.html' }}
|
||||||
<!-- START Formbricks Surveys -->
|
<!-- START Formbricks Surveys -->
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
!function(){var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src="https://unpkg.com/@formbricks/js@^1.2.0/dist/index.umd.js";var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e),setTimeout(function(){window.formbricks.init({environmentId: "<your-environment-id>", apiHost: "<api-host>"})},500)}();
|
!function(){var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src="https://unpkg.com/@formbricks/js@^1.4.0/dist/index.umd.js";var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e),setTimeout(function(){window.formbricks.init({environmentId: "<your-environment-id>", apiHost: "<api-host>"})},500)}();
|
||||||
</script>
|
</script>
|
||||||
<!-- END Formbricks Surveys -->
|
<!-- END Formbricks Surveys -->
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -10,8 +10,9 @@ import DeleteConnection from "./delete-connection.webp";
|
|||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = {
|
||||||
title: "n8n Setup",
|
title: "Google Sheets",
|
||||||
description: "Wire up Formbricks with n8n and 350+ other apps",
|
description:
|
||||||
|
"The Google Sheets integration allows you to automatically send responses to a Google Sheet of your choice.",
|
||||||
};
|
};
|
||||||
|
|
||||||
#### Integrations
|
#### Integrations
|
||||||
@@ -62,7 +63,7 @@ Before the next step, make sure that you have a Formbricks Survey with at least
|
|||||||
|
|
||||||
</Note>
|
</Note>
|
||||||
|
|
||||||
6. Now click on the "Link New Sheet" button to link a new Google Sheet with Formbricks and a modal will open up.
|
5. Now click on the "Link New Sheet" button to link a new Google Sheet with Formbricks and a modal will open up.
|
||||||
|
|
||||||
<Image
|
<Image
|
||||||
src={LinkSurveyWithSheet}
|
src={LinkSurveyWithSheet}
|
||||||
@@ -71,17 +72,16 @@ Before the next step, make sure that you have a Formbricks Survey with at least
|
|||||||
className="max-w-full rounded-lg sm:max-w-3xl"
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
7. Select the Google Sheet you want to link with Formbricks and the Survey. On doing so, you will be asked with what questions' responses you want to feed in the Google Sheet. Select the questions and click on the "Link Sheet" button.
|
6. Select the Google Sheet you want to link with Formbricks and the Survey. On doing so, you will be asked with what questions' responses you want to feed in the Google Sheet. Select the questions and click on the "Link Sheet" button.
|
||||||
|
|
||||||
<Image
|
<Image
|
||||||
src={LinkWithQuestions}
|
src={LinkWithQuestions}
|
||||||
alt="Select question to link with Google Sheet"
|
alt="Select question to link with Google Sheet"
|
||||||
quality="100"
|
quality="100"
|
||||||
className="max-w-full rounded-lg sm:max-w-3xl"
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
8. On submitting, the modal will close and you will see the linked Google Sheet in the list of linked Google Sheets.
|
7. On submitting, the modal will close and you will see the linked Google Sheet in the list of linked Google Sheets.
|
||||||
|
|
||||||
<Image
|
<Image
|
||||||
src={ListLinkedSurveys}
|
src={ListLinkedSurveys}
|
||||||
|
|||||||
|
After Width: | Height: | Size: 653 KiB |
|
After Width: | Height: | Size: 498 KiB |
|
After Width: | Height: | Size: 386 KiB |
|
After Width: | Height: | Size: 836 KiB |
|
After Width: | Height: | Size: 1.1 MiB |
|
After Width: | Height: | Size: 384 KiB |
|
After Width: | Height: | Size: 462 KiB |
126
apps/formbricks-com/app/docs/integrations/notion/page.mdx
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
import IntegrationsTab from "./images/integrations-tab.png";
|
||||||
|
import ConnectWithNotion from "./images/connect-with-notion.png";
|
||||||
|
import NotionConnected from "./images/notion-connected.png";
|
||||||
|
import LinkSurveyWithDatabase from "./images/link-survey-with-database.png";
|
||||||
|
import LinkWithDatabases from "./images/link-with-databases.png";
|
||||||
|
import ListLinkedDatabases from "./images/list-linked-databases.png";
|
||||||
|
import DeleteConnection from "./images/delete-connection.png";
|
||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
|
export const metadata = {
|
||||||
|
title: "Notion",
|
||||||
|
description:
|
||||||
|
"The notion integration allows you to automatically send responses to a Notion database of your choice.",
|
||||||
|
};
|
||||||
|
|
||||||
|
#### Integrations
|
||||||
|
|
||||||
|
# Notion
|
||||||
|
|
||||||
|
The notion integration allows you to automatically send responses to a Notion database of your choice.
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
This feature is enabled by default in Formbricks Cloud but needs to be self-configured when running a
|
||||||
|
self-hosted version of Formbricks.
|
||||||
|
</Note>
|
||||||
|
|
||||||
|
## Formbricks Cloud
|
||||||
|
|
||||||
|
1. Go to the Integrations tab in your [Formbricks Cloud dashboard](https://app.formbricks.com/) and click on the "Connect" button under Notion integration.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={IntegrationsTab}
|
||||||
|
alt="Formbricks Integrations Tab"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
2. Now click on the "Connect with Notion" button to authenticate yourself with Notion.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={ConnectWithNotion}
|
||||||
|
alt="Connect Formbricks with your Notion account"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
3. You will now be taken to the Notion OAuth page where you can select the Notion account you want to use for the integration
|
||||||
|
|
||||||
|
4. Once you have selected the account and databases and completed the authentication and authorization process, you will be taken back to Formbricks Cloud and see the connected status as below:
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={NotionConnected}
|
||||||
|
alt="Formbricks is now connected with Notion"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
Before the next step, make sure that you have a Formbricks Survey with at least one question and a Notion
|
||||||
|
database in the Notion account you integrated.
|
||||||
|
</Note>
|
||||||
|
|
||||||
|
5. Now click on the "Link New Database" button to link a new Notion database with Formbricks and a modal will open up.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={LinkSurveyWithDatabase}
|
||||||
|
alt="Link Formbricks with a Notion database"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
6. Select the Notion database you want to link with Formbricks and the Survey. On doing so, you will be asked to map formbricks' survey questions with selected databases' column. Complete the mapping and click on the "Link Database" button.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={LinkWithDatabases}
|
||||||
|
alt="Question to notion database column mapping"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
7. On submitting, the modal will close and you will see the linked Notion database in the list of linked Notion databases.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={ListLinkedDatabases}
|
||||||
|
alt="List of linked notion databases"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
Congratulations! You have successfully linked a Notion database with Formbricks. Now whenever a response is submitted for the linked Survey, it will be automatically added to the linked Notion database.
|
||||||
|
|
||||||
|
## Setup in self-hosted Formbricks
|
||||||
|
|
||||||
|
Enabling the Notion Integration in a self-hosted environment requires a setup using Notion account and changing the environment variables of your Formbricks instance.
|
||||||
|
|
||||||
|
1. Sign up for a [Notion](https://www.notion.so/) account, if you don't have one already.
|
||||||
|
2. Go to the [my integrations](https://www.notion.so/my-integrations) page and click on **New integration**.
|
||||||
|
3. Fill up the basic information like **Name**, **Logo** and click on **Submit**.
|
||||||
|
4. Now, click on **Distribution** tab on the sidebar. A text will appear which will ask you to make the integration public. Click on that toggle button. A form will appear below the text.
|
||||||
|
5. Now provide it the details such as requested. Under **Redirect URIs** field:
|
||||||
|
- If you are running formbricks locally, you can enter `http://localhost:3000/api/v1/integrations/notion/callback`.
|
||||||
|
- Or, you can enter `https://<your-public-facing-url>/api/v1/integrations/notion/callback`
|
||||||
|
6. Once you've filled all the necessary details, click on **Submit**.
|
||||||
|
7. A screen will appear which will have **Client ID**, **Client secret** and **Authorization URL**. Copy them and set them as the environment variables in your Formbricks instance as:
|
||||||
|
- `NOTION_OAUTH_CLIENT_ID` - OAuth Client ID
|
||||||
|
- `NOTION_OAUTH_CLIENT_SECRET` - OAuth Client Secret
|
||||||
|
|
||||||
|
Voila! You have successfully enabled the Notion integration in your self-hosted Formbricks instance. Now you can follow the steps mentioned in the [Formbricks Cloud](#formbricks-cloud) section to link a Notion database with Formbricks.
|
||||||
|
|
||||||
|
## Remove Integration with Notion Account
|
||||||
|
|
||||||
|
To remove the integration with Notion Account,
|
||||||
|
|
||||||
|
1. Visit the Integrations tab in your Formbricks Cloud dashboard.
|
||||||
|
2. Select "Manage" button in the Notion card.
|
||||||
|
3. Click on the "Connected with `<your-workspace-name-here`> Workspace" just before the "Link new Database" button.
|
||||||
|
4. It will now ask for a confirmation to remove the integration. Click on the "Delete" button to remove the integration. You can always come back and connect again with the same Notion Account.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={DeleteConnection}
|
||||||
|
alt="Delete Notion Integration with Formbricks"
|
||||||
|
quality="100"
|
||||||
|
className="max-w-full rounded-lg sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
Still struggling or something not working as expected? [Join our Discord!](https://formbricks.com/discord) and we'd be glad to assist you!
|
||||||
|
After Width: | Height: | Size: 27 KiB |
|
After Width: | Height: | Size: 37 KiB |
|
After Width: | Height: | Size: 36 KiB |
@@ -1,3 +1,11 @@
|
|||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
|
import FormBuilder from "./form-builder.webp";
|
||||||
|
import Targeting from "./targeting.webp";
|
||||||
|
import Trigger from "./trigger.webp";
|
||||||
|
import integrations from "./integrations.webp";
|
||||||
|
import Analytics from "./analytics.webp";
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = {
|
||||||
title: "Inside Look: Formbricks In-Product Micro-Surveys",
|
title: "Inside Look: Formbricks In-Product Micro-Surveys",
|
||||||
description:
|
description:
|
||||||
@@ -21,14 +29,49 @@ Formbricks is a powerful platform designed to help you create and manage in-prod
|
|||||||
|
|
||||||
The Form Builder is where you create and customize your micro-surveys. With its intuitive drag-and-drop interface, you can easily add different question types, set response options, and apply your branding to the survey forms. The Form Builder allows you to preview your survey in real-time, ensuring it looks and feels perfect for your users.
|
The Form Builder is where you create and customize your micro-surveys. With its intuitive drag-and-drop interface, you can easily add different question types, set response options, and apply your branding to the survey forms. The Form Builder allows you to preview your survey in real-time, ensuring it looks and feels perfect for your users.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={FormBuilder}
|
||||||
|
alt="Create & Customize Surveys No Code with Formbricks"
|
||||||
|
quality="100"
|
||||||
|
className="rounded-lg max-w-full sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
## Targeting & Triggers
|
## Targeting & Triggers
|
||||||
|
|
||||||
Formbricks offers fine-grained user targeting and event-based triggers to help you display your surveys to the most relevant audience. Using the platform, you can define user segments based on attributes and behaviors, and set up triggers to show your surveys at specific moments within your product. This ensures that you're capturing the most accurate and valuable feedback possible.
|
Formbricks offers fine-grained user targeting and event-based triggers to help you display your surveys to the most relevant audience. Using the platform, you can define user segments based on attributes and behaviors, and set up triggers to show your surveys at specific moments within your product. This ensures that you're capturing the most accurate and valuable feedback possible.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={Targeting}
|
||||||
|
alt="Targeting & Triggers with Formbricks"
|
||||||
|
quality="100"
|
||||||
|
className="rounded-lg w-full sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={Trigger}
|
||||||
|
alt="Targeting & Triggers with Formbricks"
|
||||||
|
quality="100"
|
||||||
|
className="rounded-lg max-w-full sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
## Integration
|
## Integration
|
||||||
|
|
||||||
Integrating Formbricks into your web or mobile application is a breeze. With SDKs for popular web frameworks like React, and an HTML snippet for non-framework based websites, you can quickly add Formbricks to your project. The provided code snippets make it easy to initialize the Formbricks widget and configure it to communicate with your backend.
|
Integrating Formbricks into your web or mobile application is a breeze. With SDKs for popular web frameworks like React, and an HTML snippet for non-framework based websites, you can quickly add Formbricks to your project. The provided code snippets make it easy to initialize the Formbricks widget and configure it to communicate with your backend.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={integrations}
|
||||||
|
alt="Integrations with Formbricks"
|
||||||
|
quality="100"
|
||||||
|
className="rounded-lg max-w-full sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
|
||||||
## Analytics & Insights
|
## Analytics & Insights
|
||||||
|
|
||||||
Formbricks provides powerful analytics and insights to help you understand user responses and make data-driven decisions. The platform aggregates survey results and presents them in an easy-to-understand format, enabling you to identify trends, spot issues, and uncover opportunities for improvement. With Formbricks, you're always one step ahead in understanding your users and optimizing your product experience.
|
Formbricks provides powerful analytics and insights to help you understand user responses and make data-driven decisions. The platform aggregates survey results and presents them in an easy-to-understand format, enabling you to identify trends, spot issues, and uncover opportunities for improvement. With Formbricks, you're always one step ahead in understanding your users and optimizing your product experience.
|
||||||
|
|
||||||
|
<Image
|
||||||
|
src={Analytics}
|
||||||
|
alt="Analytics & Insights with Formbricks"
|
||||||
|
quality="100"
|
||||||
|
className="rounded-lg max-w-full sm:max-w-3xl"
|
||||||
|
/>
|
||||||
|
After Width: | Height: | Size: 16 KiB |
|
After Width: | Height: | Size: 35 KiB |
@@ -1,5 +1,4 @@
|
|||||||
import { GettingStarted } from "@/components/docs/GettingStarted";
|
import { GettingStarted } from "@/components/docs/GettingStarted";
|
||||||
import BestPractices from "@/components/docs/BestPractices";
|
|
||||||
import { HeroPattern } from "@/components/docs/HeroPattern";
|
import { HeroPattern } from "@/components/docs/HeroPattern";
|
||||||
import { Button } from "@/components/docs/Button";
|
import { Button } from "@/components/docs/Button";
|
||||||
|
|
||||||
@@ -9,10 +8,7 @@ export const metadata = {
|
|||||||
"Enhance your product with Formbricks – the leading open-source solution for in-product micro-surveys. Dive deep into user research, amplify product-market fit, and uncover the 'why' behind your analytics.",
|
"Enhance your product with Formbricks – the leading open-source solution for in-product micro-surveys. Dive deep into user research, amplify product-market fit, and uncover the 'why' behind your analytics.",
|
||||||
};
|
};
|
||||||
|
|
||||||
export const sections = [
|
export const sections = [];
|
||||||
{ title: "Getting Started", id: "getting-started" },
|
|
||||||
{ title: "Best Practices", id: "best-practices" },
|
|
||||||
];
|
|
||||||
|
|
||||||
<HeroPattern />
|
<HeroPattern />
|
||||||
|
|
||||||
@@ -20,7 +16,7 @@ export const sections = [
|
|||||||
|
|
||||||
Welcome to Formbricks, your go-to solution for in-product micro-surveys that will supercharge your product experience! 🚀 {{ className: 'lead' }}
|
Welcome to Formbricks, your go-to solution for in-product micro-surveys that will supercharge your product experience! 🚀 {{ className: 'lead' }}
|
||||||
|
|
||||||
<div className="mb-16 mt-6 flex gap-3">
|
<div className="mb-16 mt-6 flex gap-3" id="why-formbricks">
|
||||||
<Button href="/docs/getting-started/quickstart-in-app-survey" arrow="right" children="Quickstart" />
|
<Button href="/docs/getting-started/quickstart-in-app-survey" arrow="right" children="Quickstart" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
import glob from "fast-glob";
|
|
||||||
|
|
||||||
import { Providers } from "@/app/providers";
|
import { Providers } from "@/app/providers";
|
||||||
import { Layout } from "@/components/docs/Layout";
|
import { Layout } from "@/components/docs/Layout";
|
||||||
|
|
||||||
import { type Section } from "@/components/docs/SectionProvider";
|
import { type Section } from "@/components/docs/SectionProvider";
|
||||||
|
import glob from "fast-glob";
|
||||||
import { type Metadata } from "next";
|
import { type Metadata } from "next";
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
|
|||||||
@@ -3,14 +3,14 @@ import Image from "next/image";
|
|||||||
import PeopleView from "./people-view.webp";
|
import PeopleView from "./people-view.webp";
|
||||||
|
|
||||||
export const metadata = {
|
export const metadata = {
|
||||||
title: "Effective User Identification in Formbricks Link Surveys",
|
title: "Effective Identify Users in Formbricks Link Surveys",
|
||||||
description:
|
description:
|
||||||
"Discover how to seamlessly connect responses from Formbricks link surveys to existing users in your database. Learn the intricacies of the userId URL parameter to enhance user tracking, profiling, and segmentation, ensuring more personalized interactions and data-driven decisions.",
|
"Discover how to seamlessly connect responses from Formbricks link surveys to existing users in your database. Learn the intricacies of the userId URL parameter to enhance user tracking, profiling, and segmentation, ensuring more personalized interactions and data-driven decisions.",
|
||||||
};
|
};
|
||||||
|
|
||||||
#### Link Surveys
|
#### Link Surveys
|
||||||
|
|
||||||
# User Identification in Link Surveys
|
# Identify Users in Link Surveys
|
||||||
|
|
||||||
Identifying users in link features lets you connect responses from link surveys with existing users in your Formbricks database.
|
Identifying users in link features lets you connect responses from link surveys with existing users in your Formbricks database.
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ export const metadata = {
|
|||||||
|
|
||||||
#### Self-Hosting
|
#### Self-Hosting
|
||||||
|
|
||||||
# Running Formbricks with Docker
|
# Advanced Setup
|
||||||
|
|
||||||
Quickly set up and start using Formbricks without getting into the technicalities of the build process. You'll be using an image that we've already built for you. The pre-built image is ready-to-run, and it only requires minimal configuration on your part. This approach is perfect for getting a functional instance of Formbricks up and running with minimal hassle. It's as easy as downloading the Docker image and firing up the container.
|
Quickly set up and start using Formbricks without getting into the technicalities of the build process. You'll be using an image that we've already built for you. The pre-built image is ready-to-run, and it only requires minimal configuration on your part. This approach is perfect for getting a functional instance of Formbricks up and running with minimal hassle. It's as easy as downloading the Docker image and firing up the container.
|
||||||
|
|
||||||
@@ -242,8 +242,10 @@ These variables can be provided at the runtime i.e. in your docker-compose file.
|
|||||||
| TELEMETRY_DISABLED | Disables telemetry if set to `1`. | optional | |
|
| TELEMETRY_DISABLED | Disables telemetry if set to `1`. | optional | |
|
||||||
| INSTANCE_ID | Instance ID for Formbricks Cloud to be sent to Telemetry. | optional | |
|
| INSTANCE_ID | Instance ID for Formbricks Cloud to be sent to Telemetry. | optional | |
|
||||||
| INTERNAL_SECRET | Internal Secret (Currently we overwrite the value with a random value). | optional | |
|
| INTERNAL_SECRET | Internal Secret (Currently we overwrite the value with a random value). | optional | |
|
||||||
| IS_FORMBRICKS_CLOUD | Uses Formbricks Cloud if set to `1` | optional | |
|
|
||||||
| DEFAULT_BRAND_COLOR | Default brand color for your app (Can be overwritten from the UI as well). | optional | `#64748b` |
|
| DEFAULT_BRAND_COLOR | Default brand color for your app (Can be overwritten from the UI as well). | optional | `#64748b` |
|
||||||
|
| DEFAULT_TEAM_ID | Automatically assign new users to a specific team when joining | optional | |
|
||||||
|
| DEFAULT_TEAM_ROLE | Role of the user in the default team. | optional | `admin` |
|
||||||
|
| ONBOARDING_DISABLED | Disables onboarding for new users if set to `1` | optional | |
|
||||||
|
|
||||||
## Build-time Variables
|
## Build-time Variables
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,13 @@
|
|||||||
export const metadata = {
|
export const metadata = {
|
||||||
title: "External auth providers",
|
title: "Configure Formbricks with External auth providers",
|
||||||
description:
|
description:
|
||||||
"Set up and integrate multiple external authentication providers with Formbricks. Our step-by-step guide covers Google OAuth and more, ensuring a seamless login experience for your users.",
|
"Set up and integrate multiple external authentication providers with Formbricks. Our step-by-step guide covers Google OAuth and more, ensuring a seamless login experience for your users.",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#### Self-Hosting
|
||||||
|
|
||||||
|
# Configure
|
||||||
|
|
||||||
## Google OAuth Authentication
|
## Google OAuth Authentication
|
||||||
|
|
||||||
Integrating Google OAuth with your Formbricks instance allows users to log in using their Google credentials, ensuring a secure and streamlined user experience. This guide will walk you through the process of setting up Google OAuth for your Formbricks instance.
|
Integrating Google OAuth with your Formbricks instance allows users to log in using their Google credentials, ensuring a secure and streamlined user experience. This guide will walk you through the process of setting up Google OAuth for your Formbricks instance.
|
||||||
@@ -35,17 +39,24 @@ Integrating Google OAuth with your Formbricks instance allows users to log in us
|
|||||||
- Select the application type **Web application** for your project and enter any additional information required.
|
- Select the application type **Web application** for your project and enter any additional information required.
|
||||||
- Ensure to specify authorized JavaScript origins and authorized redirect URIs.
|
- Ensure to specify authorized JavaScript origins and authorized redirect URIs.
|
||||||
|
|
||||||
```
|
<Col>
|
||||||
|
<CodeGroup title="Configuration URLs">
|
||||||
|
``` {{ title: 'Redirect & Origin URLs' }}
|
||||||
Authorized JavaScript origins: {WEBAPP_URL}
|
Authorized JavaScript origins: {WEBAPP_URL}
|
||||||
Authorized redirect URIs: {WEBAPP_URL}/api/auth/callback/google
|
Authorized redirect URIs: {WEBAPP_URL}/api/auth/callback/google
|
||||||
```
|
```
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
5. **Update Environment Variables in Docker**:
|
5. **Update Environment Variables in Docker**:
|
||||||
- To integrate the Google OAuth, you have two options: either update the environment variables in the docker-compose file or directly add them to the running container.
|
- To integrate the Google OAuth, you have two options: either update the environment variables in the docker-compose file or directly add them to the running container.
|
||||||
- In your Docker setup directory, open the `.env` file, and add or update the following lines with the `Client ID` and `Client Secret` obtained from Google Cloud Platform:
|
- In your Docker setup directory, open the `.env` file, and add or update the following lines with the `Client ID` and `Client Secret` obtained from Google Cloud Platform:
|
||||||
- Alternatively, you can add the environment variables directly to the running container using the following commands (replace `container_id` with your actual Docker container ID):
|
- Alternatively, you can add the environment variables directly to the running container using the following commands (replace `container_id` with your actual Docker container ID):
|
||||||
|
|
||||||
```
|
<Col>
|
||||||
|
<CodeGroup title="Set env vars">
|
||||||
|
|
||||||
|
```sh {{ title: 'Shell commands' }}
|
||||||
docker exec -it container_id /bin/bash
|
docker exec -it container_id /bin/bash
|
||||||
export GOOGLE_AUTH_ENABLED=1
|
export GOOGLE_AUTH_ENABLED=1
|
||||||
export GOOGLE_CLIENT_ID=your-client-id-here
|
export GOOGLE_CLIENT_ID=your-client-id-here
|
||||||
@@ -53,12 +64,15 @@ export GOOGLE_CLIENT_SECRET=your-client-secret-here
|
|||||||
exit
|
exit
|
||||||
```
|
```
|
||||||
|
|
||||||
```
|
```sh {{ title: 'env file' }}
|
||||||
GOOGLE_AUTH_ENABLED=1
|
GOOGLE_AUTH_ENABLED=1
|
||||||
GOOGLE_CLIENT_ID=your-client-id-here
|
GOOGLE_CLIENT_ID=your-client-id-here
|
||||||
GOOGLE_CLIENT_SECRET=your-client-secret-here
|
GOOGLE_CLIENT_SECRET=your-client-secret-here
|
||||||
```
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
6. **Restart Your Formbricks Instance**:
|
6. **Restart Your Formbricks Instance**:
|
||||||
- **Note:** Restarting your Docker containers may cause a brief period of downtime. Plan accordingly.
|
- **Note:** Restarting your Docker containers may cause a brief period of downtime. Plan accordingly.
|
||||||
- Once the environment variables have been updated, it's crucial to restart your Docker containers to apply the changes. This ensures that your Formbricks instance can utilize the new Google OAuth configuration for user authentication. Here's how you can do it:
|
- Once the environment variables have been updated, it's crucial to restart your Docker containers to apply the changes. This ensures that your Formbricks instance can utilize the new Google OAuth configuration for user authentication. Here's how you can do it:
|
||||||
|
|||||||
@@ -18,7 +18,6 @@ Formbricks v1.2 ships a lot of features targeting our Link Surveys. We have also
|
|||||||
| -------------------- | -------- | ------------------------------ | ----------------------------------------------------------- |
|
| -------------------- | -------- | ------------------------------ | ----------------------------------------------------------- |
|
||||||
| ENCRYPTION_KEY | true | `openssl rand -hex 32` | Needed for 2 Factor Authentication |
|
| ENCRYPTION_KEY | true | `openssl rand -hex 32` | Needed for 2 Factor Authentication |
|
||||||
| SHORT_URL_BASE | false | `<your-short-base-url>` | Needed if you want to enable shorter links for Link Surveys |
|
| SHORT_URL_BASE | false | `<your-short-base-url>` | Needed if you want to enable shorter links for Link Surveys |
|
||||||
| ASSET_PREFIX_URL | false | `<your-asset-hosted-base-url>` | Needed if you have a separate URL for hosted assets |
|
|
||||||
|
|
||||||
### Deprecated / Removed Environment Variables
|
### Deprecated / Removed Environment Variables
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ export const metadata = {
|
|||||||
|
|
||||||
#### Self-Hosting
|
#### Self-Hosting
|
||||||
|
|
||||||
# Deploying Formbricks to Production
|
# One-Click Setup
|
||||||
|
|
||||||
If you want to quickly set up a production instance of Formbricks on a server running Ubuntu, we've got you covered! This method utilizes a convenient shell script that takes care of everything, including Docker, Postgres DB, and SSL certificate configuration. The shell script will automatically install all the required dependencies and configure your server, making the process a breeze.
|
If you want to quickly set up a production instance of Formbricks on a server running Ubuntu, we've got you covered! This method utilizes a convenient shell script that takes care of everything, including Docker, Postgres DB, and SSL certificate configuration. The shell script will automatically install all the required dependencies and configure your server, making the process a breeze.
|
||||||
|
|
||||||
@@ -29,7 +29,7 @@ Copy and paste the following command into your terminal:
|
|||||||
<CodeGroup title="Single Command to deploy Formbricks">
|
<CodeGroup title="Single Command to deploy Formbricks">
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
/bin/sh -c "$(curl -fsSL https://raw.githubusercontent.com/formbricks/formbricks/main/docker/production.sh)"
|
curl -fsSL https://raw.githubusercontent.com/formbricks/formbricks/main/docker/production.sh -o formbricks.sh && chmod +x formbricks.sh && ./formbricks.sh install
|
||||||
```
|
```
|
||||||
|
|
||||||
</CodeGroup>
|
</CodeGroup>
|
||||||
@@ -154,6 +154,69 @@ my.hosted.url.com
|
|||||||
|
|
||||||
</CodeGroup>
|
</CodeGroup>
|
||||||
</Col>
|
</Col>
|
||||||
|
|
||||||
|
## Update Formbricks
|
||||||
|
|
||||||
|
To update Formbricks, simply run the following command:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Update Formbricks">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
./formbricks.sh update
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
The script will automatically pull the latest version of Formbricks from GitHub Container Registry and restart the containers.
|
||||||
|
|
||||||
|
## Stop Formbricks Instance
|
||||||
|
|
||||||
|
To stop Formbricks, simply run the following command:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Stop Formbricks">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
./formbricks.sh stop
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
The script will automatically stop all the Formbricks related containers and brings the entire stack down.
|
||||||
|
|
||||||
|
## Restart Formbricks Instance
|
||||||
|
|
||||||
|
To restart Formbricks, simply run the following command:
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Restart Formbricks">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
./formbricks.sh restart
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
|
||||||
|
The script will automatically restart all the Formbricks related containers and brings the entire stack up with the previous configuration.
|
||||||
|
|
||||||
|
|
||||||
|
## Uninstall Formbricks
|
||||||
|
|
||||||
|
To uninstall Formbricks, simply run the following command, but keep in mind that this will delete all your data!
|
||||||
|
|
||||||
|
<Col>
|
||||||
|
<CodeGroup title="Uninstall Formbricks">
|
||||||
|
|
||||||
|
```bash
|
||||||
|
./formbricks.sh uninstall
|
||||||
|
```
|
||||||
|
|
||||||
|
</CodeGroup>
|
||||||
|
</Col>
|
||||||
|
The script will automatically stop all the Formbricks related containers, remove the Formbricks directory, and delete the Docker network.
|
||||||
|
|
||||||
## Debugging
|
## Debugging
|
||||||
|
|
||||||
If you encounter any issues, you can check the logs of the container with:
|
If you encounter any issues, you can check the logs of the container with:
|
||||||
@@ -168,6 +231,7 @@ cd formbricks && docker compose logs -f
|
|||||||
</Col>
|
</Col>
|
||||||
You can close the logs again with `CTRL + C`.
|
You can close the logs again with `CTRL + C`.
|
||||||
|
|
||||||
|
|
||||||
## Troubleshooting
|
## Troubleshooting
|
||||||
|
|
||||||
If you encounter any issues, consider the following steps:
|
If you encounter any issues, consider the following steps:
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { useEffect } from "react";
|
|
||||||
import { ThemeProvider, useTheme } from "next-themes";
|
import { ThemeProvider, useTheme } from "next-themes";
|
||||||
|
import { useEffect } from "react";
|
||||||
|
|
||||||
function ThemeWatcher() {
|
function ThemeWatcher() {
|
||||||
let { resolvedTheme, setTheme } = useTheme();
|
let { resolvedTheme, setTheme } = useTheme();
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import Link from "next/link";
|
|
||||||
import { type MotionValue, motion, useMotionTemplate, useMotionValue } from "framer-motion";
|
|
||||||
|
|
||||||
import { GridPattern } from "./GridPattern";
|
|
||||||
import { Heading } from "./Heading";
|
|
||||||
import { ChatBubbleIcon } from "@/components/docs/icons/ChatBubbleIcon";
|
import { ChatBubbleIcon } from "@/components/docs/icons/ChatBubbleIcon";
|
||||||
import { EnvelopeIcon } from "@/components/docs/icons/EnvelopeIcon";
|
import { EnvelopeIcon } from "@/components/docs/icons/EnvelopeIcon";
|
||||||
import { UserIcon } from "@/components/docs/icons/UserIcon";
|
import { UserIcon } from "@/components/docs/icons/UserIcon";
|
||||||
import { UsersIcon } from "@/components/docs/icons/UsersIcon";
|
import { UsersIcon } from "@/components/docs/icons/UsersIcon";
|
||||||
|
import { type MotionValue, motion, useMotionTemplate, useMotionValue } from "framer-motion";
|
||||||
|
import Link from "next/link";
|
||||||
|
|
||||||
|
import { GridPattern } from "./GridPattern";
|
||||||
|
import { Heading } from "./Heading";
|
||||||
|
|
||||||
interface BestPractice {
|
interface BestPractice {
|
||||||
href: string;
|
href: string;
|
||||||
@@ -73,7 +73,7 @@ const bestPractices: Array<BestPractice> = [
|
|||||||
|
|
||||||
function BestPracticeIcon({ icon: Icon }: { icon: BestPractice["icon"] }) {
|
function BestPracticeIcon({ icon: Icon }: { icon: BestPractice["icon"] }) {
|
||||||
return (
|
return (
|
||||||
<div className="dark:bg-white/7.5 dark:ring-white/15 flex h-7 w-7 items-center justify-center rounded-full bg-slate-900/5 ring-1 ring-slate-900/25 backdrop-blur-[2px] transition duration-300 group-hover:bg-white/50 group-hover:ring-slate-900/25 dark:group-hover:bg-emerald-300/10 dark:group-hover:ring-emerald-400">
|
<div className="dark:bg-white/7.5 flex h-7 w-7 items-center justify-center rounded-full bg-slate-900/5 ring-1 ring-slate-900/25 backdrop-blur-[2px] transition duration-300 group-hover:bg-white/50 group-hover:ring-slate-900/25 dark:ring-white/15 dark:group-hover:bg-emerald-300/10 dark:group-hover:ring-emerald-400">
|
||||||
<Icon className="h-5 w-5 fill-slate-700/10 stroke-slate-700 transition-colors duration-300 group-hover:stroke-slate-900 dark:fill-white/10 dark:stroke-slate-400 dark:group-hover:fill-emerald-300/10 dark:group-hover:stroke-emerald-400" />
|
<Icon className="h-5 w-5 fill-slate-700/10 stroke-slate-700 transition-colors duration-300 group-hover:stroke-slate-900 dark:fill-white/10 dark:stroke-slate-400 dark:group-hover:fill-emerald-300/10 dark:group-hover:stroke-emerald-400" />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -157,7 +157,7 @@ export default function BestPractices() {
|
|||||||
<Heading level={2} id="resources">
|
<Heading level={2} id="resources">
|
||||||
Best Practices
|
Best Practices
|
||||||
</Heading>
|
</Heading>
|
||||||
<div className="not-prose mt-4 grid grid-cols-1 gap-8 border-t border-slate-900/5 pt-10 dark:border-white/5 sm:grid-cols-2 xl:grid-cols-4">
|
<div className="not-prose mt-4 grid grid-cols-1 gap-8 border-t border-slate-900/5 pt-10 sm:grid-cols-2 xl:grid-cols-4 dark:border-white/5">
|
||||||
{bestPractices.map((resource) => (
|
{bestPractices.map((resource) => (
|
||||||
<BestPractice key={resource.href} resource={resource} />
|
<BestPractice key={resource.href} resource={resource} />
|
||||||
))}
|
))}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import Link from "next/link";
|
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
import Link from "next/link";
|
||||||
|
|
||||||
function ArrowIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
function ArrowIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { Children, createContext, isValidElement, useContext, useEffect, useRef, useState } from "react";
|
import { Tag } from "@/components/docs/Tag";
|
||||||
import { Tab } from "@headlessui/react";
|
import { Tab } from "@headlessui/react";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
import { Children, createContext, isValidElement, useContext, useEffect, useRef, useState } from "react";
|
||||||
import { create } from "zustand";
|
import { create } from "zustand";
|
||||||
|
|
||||||
import { Tag } from "@/components/docs/Tag";
|
|
||||||
|
|
||||||
const languageNames: Record<string, string> = {
|
const languageNames: Record<string, string> = {
|
||||||
js: "JavaScript",
|
js: "JavaScript",
|
||||||
ts: "TypeScript",
|
ts: "TypeScript",
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { useState } from "react";
|
||||||
|
|
||||||
import { Button } from "@formbricks/ui/Button";
|
import { Button } from "@formbricks/ui/Button";
|
||||||
import { Popover, PopoverContent, PopoverTrigger } from "@formbricks/ui/Popover";
|
import { Popover, PopoverContent, PopoverTrigger } from "@formbricks/ui/Popover";
|
||||||
import { useState } from "react";
|
|
||||||
|
|
||||||
export const DocsFeedback: React.FC = () => {
|
export const DocsFeedback: React.FC = () => {
|
||||||
const [isOpen, setIsOpen] = useState(false);
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { forwardRef, Fragment, useState } from "react";
|
|
||||||
import { usePathname } from "next/navigation";
|
|
||||||
import { Transition } from "@headlessui/react";
|
import { Transition } from "@headlessui/react";
|
||||||
|
import { usePathname } from "next/navigation";
|
||||||
|
import { Fragment, forwardRef, useState } from "react";
|
||||||
|
|
||||||
import { handleFeedbackSubmit } from "../../lib/handleFeedbackSubmit";
|
import { handleFeedbackSubmit } from "../../lib/handleFeedbackSubmit";
|
||||||
|
|
||||||
function CheckIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
function CheckIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
||||||
|
|||||||
@@ -2,7 +2,8 @@
|
|||||||
|
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
import { FaGithub, FaXTwitter, FaDiscord } from "react-icons/fa6";
|
import { FaDiscord, FaGithub, FaXTwitter } from "react-icons/fa6";
|
||||||
|
|
||||||
import { Button } from "./Button";
|
import { Button } from "./Button";
|
||||||
import { navigation } from "./Navigation";
|
import { navigation } from "./Navigation";
|
||||||
|
|
||||||
@@ -88,7 +89,7 @@ function SmallPrint() {
|
|||||||
const currentYear = new Date().getFullYear();
|
const currentYear = new Date().getFullYear();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col items-center justify-between gap-5 border-t border-slate-900/5 pt-8 dark:border-white/5 sm:flex-row">
|
<div className="flex flex-col items-center justify-between gap-5 border-t border-slate-900/5 pt-8 sm:flex-row dark:border-white/5">
|
||||||
<p className="text-xs text-slate-600 dark:text-slate-400">
|
<p className="text-xs text-slate-600 dark:text-slate-400">
|
||||||
Formbricks GmbH © {currentYear}. All rights reserved.
|
Formbricks GmbH © {currentYear}. All rights reserved.
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ export function GettingStarted() {
|
|||||||
<Heading level={2} id="getting-started">
|
<Heading level={2} id="getting-started">
|
||||||
Quick Resources
|
Quick Resources
|
||||||
</Heading>
|
</Heading>
|
||||||
<div className="not-prose mt-4 grid grid-cols-1 gap-8 border-t border-slate-900/5 pt-10 dark:border-white/5 sm:grid-cols-2 xl:grid-cols-4">
|
<div className="not-prose mt-4 grid grid-cols-1 gap-8 border-t border-slate-900/5 pt-10 sm:grid-cols-2 xl:grid-cols-4 dark:border-white/5">
|
||||||
{gettingStarted.map((guide) => (
|
{gettingStarted.map((guide) => (
|
||||||
<div key={guide.href}>
|
<div key={guide.href}>
|
||||||
<h3 className="text-sm font-semibold text-slate-900 dark:text-white">{guide.name}</h3>
|
<h3 className="text-sm font-semibold text-slate-900 dark:text-white">{guide.name}</h3>
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { FooterLogo } from "@/components/shared/Logo";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { motion, useScroll, useTransform } from "framer-motion";
|
import { motion, useScroll, useTransform } from "framer-motion";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { forwardRef } from "react";
|
import { forwardRef } from "react";
|
||||||
|
|
||||||
import { FooterLogo } from "@/components/shared/Logo";
|
|
||||||
import { Button } from "./Button";
|
import { Button } from "./Button";
|
||||||
import { MobileNavigation, useIsInsideMobileNavigation, useMobileNavigationStore } from "./MobileNavigation";
|
import { MobileNavigation, useIsInsideMobileNavigation, useMobileNavigationStore } from "./MobileNavigation";
|
||||||
import { MobileSearch, Search } from "./Search";
|
import { MobileSearch, Search } from "./Search";
|
||||||
@@ -40,7 +40,7 @@ export const Header = forwardRef<React.ElementRef<"div">, { className?: string }
|
|||||||
className={clsx(
|
className={clsx(
|
||||||
className,
|
className,
|
||||||
"fixed inset-x-0 top-0 z-50 flex h-20 items-center justify-between gap-12 px-4 transition sm:px-6 lg:left-72 lg:z-30 lg:px-8 xl:left-80",
|
"fixed inset-x-0 top-0 z-50 flex h-20 items-center justify-between gap-12 px-4 transition sm:px-6 lg:left-72 lg:z-30 lg:px-8 xl:left-80",
|
||||||
!isInsideMobileNavigation && "backdrop-blur-sm dark:backdrop-blur lg:left-72 xl:left-80",
|
!isInsideMobileNavigation && "backdrop-blur-sm lg:left-72 xl:left-80 dark:backdrop-blur",
|
||||||
isInsideMobileNavigation
|
isInsideMobileNavigation
|
||||||
? "bg-white dark:bg-slate-900"
|
? "bg-white dark:bg-slate-900"
|
||||||
: "bg-white/[var(--bg-opacity-light)] dark:bg-slate-900/[var(--bg-opacity-dark)]"
|
: "bg-white/[var(--bg-opacity-light)] dark:bg-slate-900/[var(--bg-opacity-dark)]"
|
||||||
@@ -73,7 +73,7 @@ export const Header = forwardRef<React.ElementRef<"div">, { className?: string }
|
|||||||
<TopLevelNavItem href="https://formbricks.com/discord">Join our Discord</TopLevelNavItem>
|
<TopLevelNavItem href="https://formbricks.com/discord">Join our Discord</TopLevelNavItem>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
<div className="md:dark:bg-white/15 hidden md:block md:h-5 md:w-px md:bg-slate-900/10" />
|
<div className="hidden md:block md:h-5 md:w-px md:bg-slate-900/10 md:dark:bg-white/15" />
|
||||||
<div className="flex gap-4">
|
<div className="flex gap-4">
|
||||||
<MobileSearch />
|
<MobileSearch />
|
||||||
<ThemeToggle />
|
<ThemeToggle />
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { remToPx } from "@/lib/remToPx";
|
||||||
import { useInView } from "framer-motion";
|
import { useInView } from "framer-motion";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
import { usePathname } from "next/navigation";
|
||||||
import { useEffect, useRef } from "react";
|
import { useEffect, useRef } from "react";
|
||||||
|
|
||||||
import { remToPx } from "@/lib/remToPx";
|
|
||||||
import { useSectionStore } from "./SectionProvider";
|
import { useSectionStore } from "./SectionProvider";
|
||||||
import { Tag } from "./Tag";
|
import { Tag } from "./Tag";
|
||||||
import { usePathname } from "next/navigation";
|
|
||||||
|
|
||||||
function AnchorIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
function AnchorIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { Navigation } from "@/components/docs/Navigation";
|
||||||
|
import { FooterLogo } from "@/components/shared/Logo";
|
||||||
|
import { motion } from "framer-motion";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
import { motion } from "framer-motion";
|
|
||||||
|
|
||||||
import { Footer } from "./Footer";
|
import { Footer } from "./Footer";
|
||||||
import { Header } from "./Header";
|
import { Header } from "./Header";
|
||||||
import { type Section, SectionProvider } from "./SectionProvider";
|
import { type Section, SectionProvider } from "./SectionProvider";
|
||||||
import { FooterLogo } from "@/components/shared/Logo";
|
|
||||||
import { Navigation } from "@/components/docs/Navigation";
|
|
||||||
|
|
||||||
export function Layout({
|
export function Layout({
|
||||||
children,
|
children,
|
||||||
@@ -25,7 +25,7 @@ export function Layout({
|
|||||||
<motion.header
|
<motion.header
|
||||||
layoutScroll
|
layoutScroll
|
||||||
className="contents lg:pointer-events-none lg:fixed lg:inset-0 lg:z-40 lg:flex">
|
className="contents lg:pointer-events-none lg:fixed lg:inset-0 lg:z-40 lg:flex">
|
||||||
<div className="contents lg:pointer-events-auto lg:block lg:w-72 lg:overflow-y-auto lg:border-r lg:border-slate-900/10 lg:px-6 lg:pb-8 lg:pt-4 lg:dark:border-white/10 xl:w-80">
|
<div className="contents lg:pointer-events-auto lg:block lg:w-72 lg:overflow-y-auto lg:border-r lg:border-slate-900/10 lg:px-6 lg:pb-8 lg:pt-4 xl:w-80 lg:dark:border-white/10">
|
||||||
<div className="hidden lg:flex">
|
<div className="hidden lg:flex">
|
||||||
<Link href="/" aria-label="Home">
|
<Link href="/" aria-label="Home">
|
||||||
<FooterLogo className="h-8" />
|
<FooterLogo className="h-8" />
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
import Image from "next/image";
|
|
||||||
|
|
||||||
import logoHtml from "@/images/logos/html5.svg";
|
import logoHtml from "@/images/logos/html5.svg";
|
||||||
import logoNextjs from "@/images/logos/nextjs.svg";
|
import logoNextjs from "@/images/logos/nextjs.svg";
|
||||||
import logoReactJs from "@/images/logos/reactjs.svg";
|
import logoReactJs from "@/images/logos/reactjs.svg";
|
||||||
import logoVueJs from "@/images/logos/vuejs.svg";
|
import logoVueJs from "@/images/logos/vuejs.svg";
|
||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
import { Button } from "./Button";
|
import { Button } from "./Button";
|
||||||
|
|
||||||
const libraries = [
|
const libraries = [
|
||||||
@@ -37,7 +37,7 @@ const libraries = [
|
|||||||
export function Libraries() {
|
export function Libraries() {
|
||||||
return (
|
return (
|
||||||
<div className="my-16 xl:max-w-none">
|
<div className="my-16 xl:max-w-none">
|
||||||
<div className="not-prose mt-4 grid grid-cols-1 gap-x-6 gap-y-10 border-slate-900/5 dark:border-white/5 sm:grid-cols-2 xl:max-w-none xl:grid-cols-3">
|
<div className="not-prose mt-4 grid grid-cols-1 gap-x-6 gap-y-10 border-slate-900/5 sm:grid-cols-2 xl:max-w-none xl:grid-cols-3 dark:border-white/5">
|
||||||
{libraries.map((library) => (
|
{libraries.map((library) => (
|
||||||
<a
|
<a
|
||||||
key={library.name}
|
key={library.name}
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { createContext, Fragment, Suspense, useContext, useEffect, useRef } from "react";
|
import { Header } from "@/components/docs/Header";
|
||||||
import { usePathname, useSearchParams } from "next/navigation";
|
|
||||||
import { Dialog, Transition } from "@headlessui/react";
|
import { Dialog, Transition } from "@headlessui/react";
|
||||||
import { motion } from "framer-motion";
|
import { motion } from "framer-motion";
|
||||||
|
import { usePathname, useSearchParams } from "next/navigation";
|
||||||
|
import { Fragment, Suspense, createContext, useContext, useEffect, useRef } from "react";
|
||||||
import { create } from "zustand";
|
import { create } from "zustand";
|
||||||
|
|
||||||
import { Header } from "@/components/docs/Header";
|
|
||||||
import { Navigation } from "./Navigation";
|
import { Navigation } from "./Navigation";
|
||||||
|
|
||||||
function MenuIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
function MenuIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
||||||
@@ -90,7 +90,7 @@ function MobileNavigationDialog({ isOpen, close }: { isOpen: boolean; close: ()
|
|||||||
leaveTo="-translate-x-full">
|
leaveTo="-translate-x-full">
|
||||||
<motion.div
|
<motion.div
|
||||||
layoutScroll
|
layoutScroll
|
||||||
className="ring-slate-900/7.5 fixed bottom-0 left-0 top-14 w-full overflow-y-auto bg-white px-4 pb-4 pt-6 shadow-lg shadow-slate-900/10 ring-1 dark:bg-slate-900 dark:ring-slate-800 min-[416px]:max-w-sm sm:px-6 sm:pb-10">
|
className="ring-slate-900/7.5 fixed bottom-0 left-0 top-14 w-full overflow-y-auto bg-white px-4 pb-4 pt-6 shadow-lg shadow-slate-900/10 ring-1 min-[416px]:max-w-sm sm:px-6 sm:pb-10 dark:bg-slate-900 dark:ring-slate-800">
|
||||||
<Navigation />
|
<Navigation />
|
||||||
</motion.div>
|
</motion.div>
|
||||||
</Transition.Child>
|
</Transition.Child>
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { remToPx } from "@/lib/remToPx";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { AnimatePresence, motion, useIsPresent } from "framer-motion";
|
import { AnimatePresence, motion, useIsPresent } from "framer-motion";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { usePathname } from "next/navigation";
|
import { usePathname } from "next/navigation";
|
||||||
import { useRef } from "react";
|
import { useRef } from "react";
|
||||||
|
|
||||||
import { remToPx } from "@/lib/remToPx";
|
|
||||||
import { Button } from "./Button";
|
import { Button } from "./Button";
|
||||||
import { useIsInsideMobileNavigation } from "./MobileNavigation";
|
import { useIsInsideMobileNavigation } from "./MobileNavigation";
|
||||||
import { useSectionStore } from "./SectionProvider";
|
import { useSectionStore } from "./SectionProvider";
|
||||||
@@ -212,6 +212,14 @@ export const navigation: Array<NavGroup> = [
|
|||||||
{ title: "Code Actions", href: "/docs/actions/code" },
|
{ title: "Code Actions", href: "/docs/actions/code" },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
title: "Link Surveys",
|
||||||
|
links: [
|
||||||
|
{ title: "Data Prefilling", href: "/docs/link-surveys/data-prefilling" },
|
||||||
|
{ title: "Identify Users", href: "/docs/link-surveys/user-identification" },
|
||||||
|
{ title: "Single Use Links", href: "/docs/link-surveys/single-use-links" },
|
||||||
|
],
|
||||||
|
},
|
||||||
{
|
{
|
||||||
title: "Best Practices",
|
title: "Best Practices",
|
||||||
links: [
|
links: [
|
||||||
@@ -229,27 +237,20 @@ export const navigation: Array<NavGroup> = [
|
|||||||
links: [
|
links: [
|
||||||
{ title: "Airtable", href: "/docs/integrations/airtable" },
|
{ title: "Airtable", href: "/docs/integrations/airtable" },
|
||||||
{ title: "Google Sheets", href: "/docs/integrations/google-sheets" },
|
{ title: "Google Sheets", href: "/docs/integrations/google-sheets" },
|
||||||
|
{ title: "Notion", href: "/docs/integrations/notion" },
|
||||||
{ title: "Make.com", href: "/docs/integrations/make" },
|
{ title: "Make.com", href: "/docs/integrations/make" },
|
||||||
{ title: "n8n", href: "/docs/integrations/n8n" },
|
{ title: "n8n", href: "/docs/integrations/n8n" },
|
||||||
{ title: "Zapier", href: "/docs/integrations/zapier" },
|
{ title: "Zapier", href: "/docs/integrations/zapier" },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
|
||||||
title: "Link Surveys",
|
|
||||||
links: [
|
|
||||||
{ title: "Data Prefilling", href: "/docs/link-surveys/data-prefilling" },
|
|
||||||
{ title: "User Identification", href: "/docs/link-surveys/user-identification" },
|
|
||||||
{ title: "Single Use Links", href: "/docs/link-surveys/single-use-links" },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
title: "Self-hosting",
|
title: "Self-hosting",
|
||||||
links: [
|
links: [
|
||||||
{ title: "Deployment", href: "/docs/self-hosting/deployment" },
|
{ title: "Introduction", href: "/docs/self-hosting/deployment" },
|
||||||
{ title: "Production", href: "/docs/self-hosting/production" },
|
{ title: "One-Click Setup", href: "/docs/self-hosting/production" },
|
||||||
{ title: "Docker", href: "/docs/self-hosting/docker" },
|
{ title: "Advanced Setup", href: "/docs/self-hosting/docker" },
|
||||||
|
{ title: "Configure", href: "/docs/self-hosting/external-auth-providers" },
|
||||||
{ title: "Migration Guide", href: "/docs/self-hosting/migration-guide" },
|
{ title: "Migration Guide", href: "/docs/self-hosting/migration-guide" },
|
||||||
{ title: "External auth providers", href: "/docs/self-hosting/external-auth-providers" },
|
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,19 +1,18 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { forwardRef, Fragment, Suspense, useCallback, useEffect, useId, useRef, useState } from "react";
|
import { navigation } from "@/components/docs/Navigation";
|
||||||
import Highlighter from "react-highlight-words";
|
import { type Result } from "@/mdx/search.mjs";
|
||||||
import { usePathname, useRouter, useSearchParams } from "next/navigation";
|
|
||||||
import {
|
import {
|
||||||
type AutocompleteApi,
|
type AutocompleteApi,
|
||||||
createAutocomplete,
|
|
||||||
type AutocompleteState,
|
|
||||||
type AutocompleteCollection,
|
type AutocompleteCollection,
|
||||||
|
type AutocompleteState,
|
||||||
|
createAutocomplete,
|
||||||
} from "@algolia/autocomplete-core";
|
} from "@algolia/autocomplete-core";
|
||||||
import { Dialog, Transition } from "@headlessui/react";
|
import { Dialog, Transition } from "@headlessui/react";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
import { usePathname, useRouter, useSearchParams } from "next/navigation";
|
||||||
import { type Result } from "@/mdx/search.mjs";
|
import { Fragment, Suspense, forwardRef, useCallback, useEffect, useId, useRef, useState } from "react";
|
||||||
import { navigation } from "@/components/docs/Navigation";
|
import Highlighter from "react-highlight-words";
|
||||||
|
|
||||||
type EmptyObject = Record<string, never>;
|
type EmptyObject = Record<string, never>;
|
||||||
|
|
||||||
@@ -244,7 +243,7 @@ const SearchInput = forwardRef<
|
|||||||
<input
|
<input
|
||||||
ref={inputRef}
|
ref={inputRef}
|
||||||
className={clsx(
|
className={clsx(
|
||||||
"flex-auto appearance-none bg-transparent pl-10 text-slate-900 outline-none placeholder:text-slate-500 focus:w-full focus:flex-none dark:text-white sm:text-sm [&::-webkit-search-cancel-button]:hidden [&::-webkit-search-decoration]:hidden [&::-webkit-search-results-button]:hidden [&::-webkit-search-results-decoration]:hidden",
|
"flex-auto appearance-none bg-transparent pl-10 text-slate-900 outline-none placeholder:text-slate-500 focus:w-full focus:flex-none sm:text-sm dark:text-white [&::-webkit-search-cancel-button]:hidden [&::-webkit-search-decoration]:hidden [&::-webkit-search-results-button]:hidden [&::-webkit-search-results-decoration]:hidden",
|
||||||
autocompleteState.status === "stalled" ? "pr-11" : "pr-4"
|
autocompleteState.status === "stalled" ? "pr-11" : "pr-4"
|
||||||
)}
|
)}
|
||||||
{...inputProps}
|
{...inputProps}
|
||||||
@@ -337,7 +336,7 @@ function SearchDialog({
|
|||||||
leave="ease-in duration-200"
|
leave="ease-in duration-200"
|
||||||
leaveFrom="opacity-100 scale-100"
|
leaveFrom="opacity-100 scale-100"
|
||||||
leaveTo="opacity-0 scale-95">
|
leaveTo="opacity-0 scale-95">
|
||||||
<Dialog.Panel className="ring-slate-900/7.5 mx-auto transform-gpu overflow-hidden rounded-lg bg-slate-50 shadow-xl ring-1 dark:bg-slate-900 dark:ring-slate-800 sm:max-w-xl">
|
<Dialog.Panel className="ring-slate-900/7.5 mx-auto transform-gpu overflow-hidden rounded-lg bg-slate-50 shadow-xl ring-1 sm:max-w-xl dark:bg-slate-900 dark:ring-slate-800">
|
||||||
<div {...autocomplete.getRootProps({})}>
|
<div {...autocomplete.getRootProps({})}>
|
||||||
<form
|
<form
|
||||||
ref={formRef}
|
ref={formRef}
|
||||||
@@ -410,7 +409,7 @@ export function Search() {
|
|||||||
<div className="hidden lg:block lg:max-w-md lg:flex-auto">
|
<div className="hidden lg:block lg:max-w-md lg:flex-auto">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="ui-not-focus-visible:outline-none hidden h-8 w-full items-center gap-2 rounded-full bg-white pl-2 pr-3 text-sm text-slate-500 ring-1 ring-slate-900/10 transition hover:ring-slate-900/20 dark:bg-white/5 dark:text-slate-400 dark:ring-inset dark:ring-white/10 dark:hover:ring-white/20 lg:flex"
|
className="ui-not-focus-visible:outline-none hidden h-8 w-full items-center gap-2 rounded-full bg-white pl-2 pr-3 text-sm text-slate-500 ring-1 ring-slate-900/10 transition hover:ring-slate-900/20 lg:flex dark:bg-white/5 dark:text-slate-400 dark:ring-inset dark:ring-white/10 dark:hover:ring-white/20"
|
||||||
{...buttonProps}>
|
{...buttonProps}>
|
||||||
<SearchIcon className="h-5 w-5 stroke-current" />
|
<SearchIcon className="h-5 w-5 stroke-current" />
|
||||||
Find something...
|
Find something...
|
||||||
@@ -433,7 +432,7 @@ export function MobileSearch() {
|
|||||||
<div className="contents lg:hidden">
|
<div className="contents lg:hidden">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="ui-not-focus-visible:outline-none flex h-6 w-6 items-center justify-center rounded-md transition hover:bg-slate-900/5 dark:hover:bg-white/5 lg:hidden"
|
className="ui-not-focus-visible:outline-none flex h-6 w-6 items-center justify-center rounded-md transition hover:bg-slate-900/5 lg:hidden dark:hover:bg-white/5"
|
||||||
aria-label="Find something..."
|
aria-label="Find something..."
|
||||||
{...buttonProps}>
|
{...buttonProps}>
|
||||||
<SearchIcon className="h-5 w-5 stroke-slate-900 dark:stroke-white" />
|
<SearchIcon className="h-5 w-5 stroke-slate-900 dark:stroke-white" />
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { remToPx } from "@/lib/remToPx";
|
||||||
import { createContext, useContext, useEffect, useLayoutEffect, useState } from "react";
|
import { createContext, useContext, useEffect, useLayoutEffect, useState } from "react";
|
||||||
import { type StoreApi, createStore, useStore } from "zustand";
|
import { type StoreApi, createStore, useStore } from "zustand";
|
||||||
|
|
||||||
import { remToPx } from "@/lib/remToPx";
|
|
||||||
|
|
||||||
export interface Section {
|
export interface Section {
|
||||||
id: string;
|
id: string;
|
||||||
title: string;
|
title: string;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { useEffect, useState } from "react";
|
|
||||||
import { useTheme } from "next-themes";
|
import { useTheme } from "next-themes";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
function SunIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
function SunIcon(props: React.ComponentPropsWithoutRef<"svg">) {
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "@formbricks/ui/Accordion";
|
import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "@formbricks/ui/Accordion";
|
||||||
|
|
||||||
import FaqJsonLdComponent from "./faQJsonLD";
|
import FaqJsonLdComponent from "./faQJsonLD";
|
||||||
|
|
||||||
const FAQ_DATA = [
|
const FAQ_DATA = [
|
||||||
@@ -45,6 +46,15 @@ const FAQ_DATA = [
|
|||||||
</>
|
</>
|
||||||
),
|
),
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
question: "How can I change Button texts in my survey?",
|
||||||
|
answer: () => (
|
||||||
|
<>
|
||||||
|
For the question that you want to change the button text, click on the <b>Show Advanced Settings</b>{" "}
|
||||||
|
toggle and change the button label in the <b>Button Text</b> field.
|
||||||
|
</>
|
||||||
|
),
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export const faqJsonLdData = FAQ_DATA.map((faq) => ({
|
export const faqJsonLdData = FAQ_DATA.map((faq) => ({
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import Link from "next/link";
|
|
||||||
import clsx from "clsx";
|
|
||||||
|
|
||||||
import { Feedback } from "@/components/docs/Feedback";
|
import { Feedback } from "@/components/docs/Feedback";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import Link from "next/link";
|
||||||
|
|
||||||
import { Heading } from "./Heading";
|
import { Heading } from "./Heading";
|
||||||
import { Prose } from "./Prose";
|
import { Prose } from "./Prose";
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { Button } from "@formbricks/ui/Button";
|
|
||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@formbricks/ui/Select";
|
|
||||||
import { PlusIcon, TrashIcon } from "@heroicons/react/24/solid";
|
import { PlusIcon, TrashIcon } from "@heroicons/react/24/solid";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
|
|
||||||
|
import { Button } from "@formbricks/ui/Button";
|
||||||
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@formbricks/ui/Select";
|
||||||
|
|
||||||
const DummyUI: React.FC = () => {
|
const DummyUI: React.FC = () => {
|
||||||
const actionClasses = [
|
const actionClasses = [
|
||||||
{ id: "1", name: "View Dashboard" },
|
{ id: "1", name: "View Dashboard" },
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@formbricks/ui/Select";
|
|
||||||
import { Button } from "@formbricks/ui/Button";
|
|
||||||
import { Label } from "@formbricks/ui/Label";
|
|
||||||
import { Input } from "@formbricks/ui/Input";
|
|
||||||
import { RadioGroup, RadioGroupItem } from "@formbricks/ui/RadioGroup";
|
|
||||||
import { CursorArrowRaysIcon } from "@heroicons/react/24/solid";
|
import { CursorArrowRaysIcon } from "@heroicons/react/24/solid";
|
||||||
|
|
||||||
|
import { Button } from "@formbricks/ui/Button";
|
||||||
|
import { Input } from "@formbricks/ui/Input";
|
||||||
|
import { Label } from "@formbricks/ui/Label";
|
||||||
import { Modal } from "@formbricks/ui/Modal";
|
import { Modal } from "@formbricks/ui/Modal";
|
||||||
|
import { RadioGroup, RadioGroupItem } from "@formbricks/ui/RadioGroup";
|
||||||
|
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@formbricks/ui/Select";
|
||||||
|
|
||||||
interface EventDetailModalProps {
|
interface EventDetailModalProps {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
@@ -54,7 +55,7 @@ export const AddNoCodeEventModalDummy: React.FC<EventDetailModalProps> = ({ open
|
|||||||
Inner Text
|
Inner Text
|
||||||
</Label>
|
</Label>
|
||||||
</div>
|
</div>
|
||||||
<div className="hidden items-center space-x-2 rounded-lg bg-slate-50 p-3 dark:bg-slate-600 md:flex">
|
<div className="hidden items-center space-x-2 rounded-lg bg-slate-50 p-3 md:flex dark:bg-slate-600">
|
||||||
<RadioGroupItem disabled value="cssSelector" id="cssSelector" className="bg-slate-50" />
|
<RadioGroupItem disabled value="cssSelector" id="cssSelector" className="bg-slate-50" />
|
||||||
<Label
|
<Label
|
||||||
htmlFor="cssSelector"
|
htmlFor="cssSelector"
|
||||||
@@ -79,7 +80,7 @@ export const AddNoCodeEventModalDummy: React.FC<EventDetailModalProps> = ({ open
|
|||||||
<Label>URL</Label>
|
<Label>URL</Label>
|
||||||
<Select defaultValue="endsWith">
|
<Select defaultValue="endsWith">
|
||||||
<SelectTrigger
|
<SelectTrigger
|
||||||
className="w-[110px] dark:text-slate-200 md:w-[180px]"
|
className="w-[110px] md:w-[180px] dark:text-slate-200"
|
||||||
onClick={(e) => e.preventDefault()}
|
onClick={(e) => e.preventDefault()}
|
||||||
disabled>
|
disabled>
|
||||||
<SelectValue placeholder="Select match type" />
|
<SelectValue placeholder="Select match type" />
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { TSurveyCTAQuestion } from "@formbricks/types/surveys";
|
import { TSurveyCTAQuestion } from "@formbricks/types/surveys";
|
||||||
|
|
||||||
import Headline from "./Headline";
|
import Headline from "./Headline";
|
||||||
import HtmlBody from "./HtmlBody";
|
import HtmlBody from "./HtmlBody";
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
|
|
||||||
|
import { TTemplate } from "@formbricks/types/templates";
|
||||||
|
|
||||||
import PreviewSurvey from "./PreviewSurvey";
|
import PreviewSurvey from "./PreviewSurvey";
|
||||||
import { findTemplateByName } from "./templates";
|
import { findTemplateByName } from "./templates";
|
||||||
import { TTemplate } from "@formbricks/types/templates";
|
|
||||||
|
|
||||||
interface DemoPreviewProps {
|
interface DemoPreviewProps {
|
||||||
template: string;
|
template: string;
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
import { TTemplate } from "@formbricks/types/templates";
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
|
import { TTemplate } from "@formbricks/types/templates";
|
||||||
|
|
||||||
import PreviewSurvey from "./PreviewSurvey";
|
import PreviewSurvey from "./PreviewSurvey";
|
||||||
import TemplateList from "./TemplateList";
|
import TemplateList from "./TemplateList";
|
||||||
import { templates } from "./templates";
|
import { templates } from "./templates";
|
||||||
@@ -25,7 +27,7 @@ export default function SurveyTemplatesPage({}) {
|
|||||||
setActiveTemplate(template);
|
setActiveTemplate(template);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<aside className="group relative h-full flex-1 flex-shrink-0 overflow-hidden rounded-r-lg bg-slate-200 shadow-inner dark:bg-slate-700 md:flex md:flex-col">
|
<aside className="group relative h-full flex-1 flex-shrink-0 overflow-hidden rounded-r-lg bg-slate-200 shadow-inner md:flex md:flex-col dark:bg-slate-700">
|
||||||
{activeTemplate && (
|
{activeTemplate && (
|
||||||
<PreviewSurvey
|
<PreviewSurvey
|
||||||
activeQuestionId={activeQuestionId}
|
activeQuestionId={activeQuestionId}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { ReactNode, useEffect, useState } from "react";
|
import { ReactNode, useEffect, useState } from "react";
|
||||||
|
|
||||||
import { cn } from "@formbricks/lib/cn";
|
import { cn } from "@formbricks/lib/cn";
|
||||||
|
|
||||||
export default function Modal({
|
export default function Modal({
|
||||||
@@ -21,7 +22,7 @@ export default function Modal({
|
|||||||
<div
|
<div
|
||||||
className={cn(
|
className={cn(
|
||||||
show ? "translate-x-0 opacity-100" : "translate-x-28 opacity-0",
|
show ? "translate-x-0 opacity-100" : "translate-x-28 opacity-0",
|
||||||
"pointer-events-auto w-full max-w-sm overflow-hidden rounded-lg bg-white px-4 py-6 shadow-lg ring-1 ring-black ring-opacity-5 transition-all duration-500 ease-in-out dark:bg-slate-900 sm:p-6"
|
"pointer-events-auto w-full max-w-sm overflow-hidden rounded-lg bg-white px-4 py-6 shadow-lg ring-1 ring-black ring-opacity-5 transition-all duration-500 ease-in-out sm:p-6 dark:bg-slate-900"
|
||||||
)}>
|
)}>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
import { useState, useEffect } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
import { cn } from "@formbricks/lib/cn";
|
import { cn } from "@formbricks/lib/cn";
|
||||||
import { TSurveyMultipleChoiceMultiQuestion } from "@formbricks/types/surveys";
|
import { TSurveyMultipleChoiceMultiQuestion } from "@formbricks/types/surveys";
|
||||||
|
|
||||||
import Headline from "./Headline";
|
import Headline from "./Headline";
|
||||||
import Subheader from "./Subheader";
|
import Subheader from "./Subheader";
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
|
||||||
import { cn } from "@formbricks/lib/cn";
|
import { cn } from "@formbricks/lib/cn";
|
||||||
import { TSurveyMultipleChoiceSingleQuestion } from "@formbricks/types/surveys";
|
import { TSurveyMultipleChoiceSingleQuestion } from "@formbricks/types/surveys";
|
||||||
import { useState } from "react";
|
|
||||||
import Headline from "./Headline";
|
import Headline from "./Headline";
|
||||||
import Subheader from "./Subheader";
|
import Subheader from "./Subheader";
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
|
|
||||||
import { cn } from "@formbricks/lib/cn";
|
import { cn } from "@formbricks/lib/cn";
|
||||||
import { TSurveyNPSQuestion } from "@formbricks/types/surveys";
|
import { TSurveyNPSQuestion } from "@formbricks/types/surveys";
|
||||||
|
|
||||||
import Headline from "./Headline";
|
import Headline from "./Headline";
|
||||||
import Subheader from "./Subheader";
|
import Subheader from "./Subheader";
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
import { TSurveyOpenTextQuestion } from "@formbricks/types/surveys";
|
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
|
|
||||||
|
import { TSurveyOpenTextQuestion } from "@formbricks/types/surveys";
|
||||||
|
|
||||||
import Headline from "./Headline";
|
import Headline from "./Headline";
|
||||||
import Subheader from "./Subheader";
|
import Subheader from "./Subheader";
|
||||||
|
|
||||||
@@ -40,7 +42,7 @@ export default function OpenTextQuestion({
|
|||||||
onChange={(e) => setValue(e.target.value)}
|
onChange={(e) => setValue(e.target.value)}
|
||||||
placeholder={question.placeholder}
|
placeholder={question.placeholder}
|
||||||
required={question.required}
|
required={question.required}
|
||||||
className="block w-full rounded-md border border-slate-100 bg-slate-50 p-2 shadow-sm focus:border-slate-500 focus:ring-0 dark:border-slate-500 dark:bg-slate-700 dark:text-white sm:text-sm"></textarea>
|
className="block w-full rounded-md border border-slate-100 bg-slate-50 p-2 shadow-sm focus:border-slate-500 focus:ring-0 sm:text-sm dark:border-slate-500 dark:bg-slate-700 dark:text-white"></textarea>
|
||||||
</div>
|
</div>
|
||||||
<div className="mt-4 flex w-full justify-between">
|
<div className="mt-4 flex w-full justify-between">
|
||||||
<div></div>
|
<div></div>
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
|
|
||||||
|
import { TSurvey, TSurveyQuestion } from "@formbricks/types/surveys";
|
||||||
|
|
||||||
import Modal from "./Modal";
|
import Modal from "./Modal";
|
||||||
import QuestionConditional from "./QuestionConditional";
|
import QuestionConditional from "./QuestionConditional";
|
||||||
import { TSurveyQuestion, TSurvey } from "@formbricks/types/surveys";
|
|
||||||
import ThankYouCard from "./ThankYouCard";
|
import ThankYouCard from "./ThankYouCard";
|
||||||
|
|
||||||
interface PreviewSurveyProps {
|
interface PreviewSurveyProps {
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import { TSurveyQuestion, TSurveyQuestionType } from "@formbricks/types/surveys";
|
import { TSurveyQuestion, TSurveyQuestionType } from "@formbricks/types/surveys";
|
||||||
import OpenTextQuestion from "./OpenTextQuestion";
|
|
||||||
import MultipleChoiceSingleQuestion from "./MultipleChoiceSingleQuestion";
|
|
||||||
import MultipleChoiceMultiQuestion from "./MultipleChoiceMultiQuestion";
|
|
||||||
import NPSQuestion from "./NPSQuestion";
|
|
||||||
import CTAQuestion from "./CTAQuestion";
|
import CTAQuestion from "./CTAQuestion";
|
||||||
|
import MultipleChoiceMultiQuestion from "./MultipleChoiceMultiQuestion";
|
||||||
|
import MultipleChoiceSingleQuestion from "./MultipleChoiceSingleQuestion";
|
||||||
|
import NPSQuestion from "./NPSQuestion";
|
||||||
|
import OpenTextQuestion from "./OpenTextQuestion";
|
||||||
import RatingQuestion from "./RatingQuestion";
|
import RatingQuestion from "./RatingQuestion";
|
||||||
|
|
||||||
interface QuestionConditionalProps {
|
interface QuestionConditionalProps {
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
import { TSurveyRatingQuestion } from "@formbricks/types/surveys";
|
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
|
|
||||||
import { cn } from "@formbricks/lib/cn";
|
import { cn } from "@formbricks/lib/cn";
|
||||||
|
import { TSurveyRatingQuestion } from "@formbricks/types/surveys";
|
||||||
|
|
||||||
import Headline from "./Headline";
|
import Headline from "./Headline";
|
||||||
import Subheader from "./Subheader";
|
import Subheader from "./Subheader";
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
import { TTemplate } from "@formbricks/types/templates";
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
import { cn } from "@formbricks/lib/cn";
|
import { cn } from "@formbricks/lib/cn";
|
||||||
|
import { TTemplate } from "@formbricks/types/templates";
|
||||||
|
|
||||||
import { templates } from "./templates";
|
import { templates } from "./templates";
|
||||||
|
|
||||||
type TemplateList = {
|
type TemplateList = {
|
||||||
|
|||||||
@@ -1,3 +1,7 @@
|
|||||||
|
import { createId } from "@paralleldrive/cuid2";
|
||||||
|
|
||||||
|
import { TSurveyQuestionType } from "@formbricks/types/surveys";
|
||||||
|
import { TTemplate } from "@formbricks/types/templates";
|
||||||
import {
|
import {
|
||||||
AppPieChartIcon,
|
AppPieChartIcon,
|
||||||
ArrowRightCircleIcon,
|
ArrowRightCircleIcon,
|
||||||
@@ -22,16 +26,18 @@ import {
|
|||||||
VideoTabletAdjustIcon,
|
VideoTabletAdjustIcon,
|
||||||
} from "@formbricks/ui/icons";
|
} from "@formbricks/ui/icons";
|
||||||
|
|
||||||
import { createId } from "@paralleldrive/cuid2";
|
|
||||||
import { TTemplate } from "@formbricks/types/templates";
|
|
||||||
import { TSurveyQuestionType } from "@formbricks/types/surveys";
|
|
||||||
|
|
||||||
const thankYouCardDefault = {
|
const thankYouCardDefault = {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
headline: "Thank you!",
|
headline: "Thank you!",
|
||||||
subheader: "We appreciate your feedback.",
|
subheader: "We appreciate your feedback.",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const welcomeCardDefault = {
|
||||||
|
enabled: true,
|
||||||
|
timeToFinish: false,
|
||||||
|
showResponseCount: false,
|
||||||
|
};
|
||||||
|
|
||||||
export const customSurvey: TTemplate = {
|
export const customSurvey: TTemplate = {
|
||||||
name: "Start from scratch",
|
name: "Start from scratch",
|
||||||
description: "Create a survey without template.",
|
description: "Create a survey without template.",
|
||||||
@@ -51,10 +57,7 @@ export const customSurvey: TTemplate = {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -150,10 +153,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -260,10 +260,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -340,10 +337,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -389,10 +383,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -447,10 +438,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -513,10 +501,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -582,10 +567,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -640,10 +622,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -685,10 +664,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -723,10 +699,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -752,10 +725,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -803,10 +773,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -848,10 +815,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -905,10 +869,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -961,10 +922,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1013,10 +971,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1043,10 +998,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1071,10 +1023,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1098,10 +1047,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1142,10 +1088,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1179,10 +1122,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1216,10 +1156,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -1281,10 +1218,7 @@ export const templates: TTemplate[] = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
thankYouCard: thankYouCardDefault,
|
thankYouCard: thankYouCardDefault,
|
||||||
welcomeCard: {
|
welcomeCard: welcomeCardDefault,
|
||||||
enabled: false,
|
|
||||||
timeToFinish: false,
|
|
||||||
},
|
|
||||||
hiddenFields: {
|
hiddenFields: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
|
import HeadingCentered from "@/components/shared/HeadingCentered";
|
||||||
import { FAQPageJsonLd } from "next-seo";
|
import { FAQPageJsonLd } from "next-seo";
|
||||||
|
|
||||||
import HeadingCentered from "@/components/shared/HeadingCentered";
|
|
||||||
import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "@formbricks/ui/Accordion";
|
import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "@formbricks/ui/Accordion";
|
||||||
|
|
||||||
const FAQ_DATA = [
|
const FAQ_DATA = [
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { CodeFileIcon, EyeIcon, HandPuzzleIcon } from "@formbricks/ui/icons";
|
import { CodeFileIcon, EyeIcon, HandPuzzleIcon } from "@formbricks/ui/icons";
|
||||||
|
|
||||||
import HeadingCentered from "../shared/HeadingCentered";
|
import HeadingCentered from "../shared/HeadingCentered";
|
||||||
|
|
||||||
const features = [
|
const features = [
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import PHIcon from "@/images/formtribe/ph-logo.png";
|
|
||||||
import Image from "next/image";
|
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
|
||||||
export const GitHubSponsorship: React.FC = () => {
|
export const GitHubSponsorship: React.FC = () => {
|
||||||
@@ -22,7 +20,7 @@ export const GitHubSponsorship: React.FC = () => {
|
|||||||
/> */}
|
/> */}
|
||||||
|
|
||||||
<div className="col-span-2">
|
<div className="col-span-2">
|
||||||
<h2 className="text-2xl font-bold tracking-tight text-slate-800 dark:text-slate-200 lg:text-2xl">
|
<h2 className="text-2xl font-bold tracking-tight text-slate-800 lg:text-2xl dark:text-slate-200">
|
||||||
We are live on ProductHunt today 🚀
|
We are live on ProductHunt today 🚀
|
||||||
</h2>
|
</h2>
|
||||||
<p className="lg:text-md mt-2 max-w-3xl text-slate-500 dark:text-slate-400">
|
<p className="lg:text-md mt-2 max-w-3xl text-slate-500 dark:text-slate-400">
|
||||||
@@ -38,7 +36,7 @@ export const GitHubSponsorship: React.FC = () => {
|
|||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center justify-end">
|
<div className="flex items-center justify-end">
|
||||||
<Image src={PHIcon} alt="Product Hunt Logo" width={80} className="" />
|
{/* <Image src={PHIcon} alt="Product Hunt Logo" width={80} className="" /> */}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Link>
|
</Link>
|
||||||
|
|||||||
@@ -1,16 +1,18 @@
|
|||||||
import CalLogoDark from "@/images/clients/cal-logo-dark.svg";
|
import CalLogoDark from "@/images/clients/cal-logo-dark.svg";
|
||||||
import CalLogoLight from "@/images/clients/cal-logo-light.svg";
|
import CalLogoLight from "@/images/clients/cal-logo-light.svg";
|
||||||
import ClovyrLogo from "@/images/clients/clovyr-logo.svg";
|
|
||||||
import CrowdLogoDark from "@/images/clients/crowd-logo-dark.svg";
|
import CrowdLogoDark from "@/images/clients/crowd-logo-dark.svg";
|
||||||
import CrowdLogoLight from "@/images/clients/crowd-logo-light.svg";
|
import CrowdLogoLight from "@/images/clients/crowd-logo-light.svg";
|
||||||
|
import FlixbusLogo from "@/images/clients/flixbus-white.svg";
|
||||||
import NILogoDark from "@/images/clients/niLogoDark.svg";
|
import NILogoDark from "@/images/clients/niLogoDark.svg";
|
||||||
import NILogoLight from "@/images/clients/niLogoWhite.svg";
|
import NILogoLight from "@/images/clients/niLogoWhite.svg";
|
||||||
import AnimationFallback from "@/public/animations/opensource-xm-platform-formbricks-fallback.png";
|
import AnimationFallback from "@/public/animations/opensource-xm-platform-formbricks-fallback.png";
|
||||||
import { Button } from "@formbricks/ui/Button";
|
import { ShieldCheckIcon, StarIcon } from "@heroicons/react/24/outline";
|
||||||
import { ChevronRightIcon } from "@heroicons/react/24/outline";
|
|
||||||
import { usePlausible } from "next-plausible";
|
import { usePlausible } from "next-plausible";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
|
import { Button } from "@formbricks/ui/Button";
|
||||||
|
|
||||||
import HeroAnimation from "./HeroAnimation";
|
import HeroAnimation from "./HeroAnimation";
|
||||||
|
|
||||||
export const Hero: React.FC = ({}) => {
|
export const Hero: React.FC = ({}) => {
|
||||||
@@ -19,87 +21,92 @@ export const Hero: React.FC = ({}) => {
|
|||||||
return (
|
return (
|
||||||
<div className="relative">
|
<div className="relative">
|
||||||
<div className="px-4 pb-20 pt-16 text-center sm:px-6 lg:px-8 lg:pb-32 lg:pt-20">
|
<div className="px-4 pb-20 pt-16 text-center sm:px-6 lg:px-8 lg:pb-32 lg:pt-20">
|
||||||
<a
|
<div className="xs:text-sm flex items-center justify-center space-x-4 divide-x-2 text-xs text-slate-600">
|
||||||
href="https://formbricks.com/github"
|
<p>
|
||||||
target="_blank"
|
<ShieldCheckIcon className="mb-1 inline h-4 w-4" /> Privacy-first
|
||||||
className="border-brand-dark xs:text-sm animate-bounce rounded-full border px-4 py-1.5 text-xs text-slate-500 hover:bg-slate-100 dark:text-slate-300 dark:hover:bg-slate-800">
|
</p>
|
||||||
We're Open Source - Star us on GitHub
|
<a href="https://formbricks.com/github" target="_blank" className="hover:text-slate-800">
|
||||||
<ChevronRightIcon className="mb-1 ml-1 inline h-4 w-4 text-slate-300" />
|
<StarIcon className="mb-1 ml-3 mr-1 inline h-4 w-4" />
|
||||||
</a>
|
Star us on GitHub
|
||||||
<h1 className="mt-10 text-3xl font-bold tracking-tight text-slate-800 dark:text-slate-200 sm:text-4xl md:text-5xl">
|
</a>
|
||||||
<span className="xl:inline">The Open Source Survey Suite</span>
|
</div>
|
||||||
|
<h1 className="mt-10 text-3xl font-bold tracking-tight text-slate-800 sm:text-4xl md:text-5xl dark:text-slate-200">
|
||||||
|
<span className="xl:inline">
|
||||||
|
Turn customer insights
|
||||||
|
<br />
|
||||||
|
into irresistible experiences
|
||||||
|
</span>
|
||||||
</h1>
|
</h1>
|
||||||
|
<p className="xs:max-w-none mx-auto mt-3 max-w-xs text-balance text-base text-slate-500 sm:text-lg md:mt-5 md:text-xl dark:text-slate-400">
|
||||||
<p className="xs:max-w-none mx-auto mt-3 max-w-xs text-base text-slate-500 dark:text-slate-400 sm:text-lg md:mt-5 md:text-xl">
|
Formbricks is an Experience Management Suite built on the largest open source survey stack
|
||||||
Run link surveys, in-app surveys and email surveys in one app —{" "}
|
worldwide. Gracefully gather feedback at every step of the customer journey to{" "}
|
||||||
<span className="decoration-brand-dark underline underline-offset-4">all privacy-first.</span>
|
<span className="decoration-brand-dark underline underline-offset-4">
|
||||||
|
know what your customers need.
|
||||||
|
</span>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<div className="mx-auto mt-5 max-w-2xl items-center px-4 sm:flex sm:justify-center md:mt-6 md:space-x-8 md:px-0">
|
<div className="mx-auto mt-5 max-w-2xl items-center px-4 sm:flex sm:justify-center md:mt-6 md:space-x-8 md:px-0">
|
||||||
<p className="hidden whitespace-nowrap pt-3 text-xs text-slate-400 dark:text-slate-500 md:block">
|
|
||||||
Trusted by
|
|
||||||
</p>
|
|
||||||
<div className="grid grid-cols-4 items-center gap-6 pt-2 md:gap-8">
|
<div className="grid grid-cols-4 items-center gap-6 pt-2 md:gap-8">
|
||||||
|
<Image
|
||||||
|
src={FlixbusLogo}
|
||||||
|
alt="Flixbus Flix Flixtrain Logo"
|
||||||
|
className="rounded-lg pb-1 hover:opacity-100 md:opacity-50"
|
||||||
|
width={200}
|
||||||
|
/>
|
||||||
<Image
|
<Image
|
||||||
src={CalLogoLight}
|
src={CalLogoLight}
|
||||||
alt="Cal Logo"
|
alt="Cal Logo"
|
||||||
className="block rounded-lg hover:opacity-100 dark:hidden md:opacity-50"
|
className="block rounded-lg hover:opacity-100 md:opacity-50 dark:hidden"
|
||||||
width={170}
|
width={170}
|
||||||
/>
|
/>
|
||||||
<Image
|
<Image
|
||||||
src={CalLogoDark}
|
src={CalLogoDark}
|
||||||
alt="Cal Logo"
|
alt="Cal Logo"
|
||||||
className="hidden rounded-lg hover:opacity-100 dark:block md:opacity-50"
|
className="hidden rounded-lg hover:opacity-100 md:opacity-50 dark:block"
|
||||||
width={170}
|
width={170}
|
||||||
/>
|
/>
|
||||||
<Image
|
<Image
|
||||||
src={CrowdLogoLight}
|
src={CrowdLogoLight}
|
||||||
alt="Crowd.dev Logo"
|
alt="Crowd.dev Logo"
|
||||||
className="block rounded-lg pb-1 hover:opacity-100 dark:hidden md:opacity-50"
|
className="block rounded-lg pb-1 hover:opacity-100 md:opacity-50 dark:hidden"
|
||||||
width={200}
|
width={200}
|
||||||
/>
|
/>
|
||||||
<Image
|
<Image
|
||||||
src={CrowdLogoDark}
|
src={CrowdLogoDark}
|
||||||
alt="Crowd.dev Logo"
|
alt="Crowd.dev Logo"
|
||||||
className="hidden rounded-lg pb-1 hover:opacity-100 dark:block md:opacity-50"
|
className="hidden rounded-lg pb-1 hover:opacity-100 md:opacity-50 dark:block"
|
||||||
width={200}
|
width={200}
|
||||||
/>
|
/>
|
||||||
<Image
|
<Image
|
||||||
src={NILogoDark}
|
src={NILogoDark}
|
||||||
alt="Neverinstall Logo"
|
alt="Neverinstall Logo"
|
||||||
className="block pb-1 hover:opacity-100 dark:hidden md:opacity-50"
|
className="block pb-1 hover:opacity-100 md:opacity-50 dark:hidden"
|
||||||
width={200}
|
width={200}
|
||||||
/>
|
/>
|
||||||
<Image
|
<Image
|
||||||
src={NILogoLight}
|
src={NILogoLight}
|
||||||
alt="Neverinstall Logo"
|
alt="Neverinstall Logo"
|
||||||
className="hidden pb-1 hover:opacity-100 dark:block md:opacity-50"
|
className="hidden pb-1 hover:opacity-100 md:opacity-50 dark:block"
|
||||||
width={200}
|
|
||||||
/>
|
|
||||||
<Image
|
|
||||||
src={ClovyrLogo}
|
|
||||||
alt="Clovyr Logo"
|
|
||||||
className="rounded-lg pb-1 hover:opacity-100 md:opacity-50"
|
|
||||||
width={200}
|
width={200}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="hidden pt-10 md:block">
|
<div className="hidden pt-14 md:block">
|
||||||
<Button
|
<Button
|
||||||
variant="highlight"
|
variant="highlight"
|
||||||
className="mr-3 px-6"
|
className="mr-3 px-6"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
router.push("https://app.formbricks.com/auth/signup");
|
router.push("https://app.formbricks.com/auth/signup");
|
||||||
plausible("Hero_CTA_CreateSurvey");
|
plausible("Hero_CTA_GetStartedItsFree");
|
||||||
}}>
|
}}>
|
||||||
Create survey
|
Get Started, it's Free
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
variant="secondary"
|
variant="secondary"
|
||||||
className="px-6"
|
className="px-6"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
router.push("https://formbricks.com/github");
|
router.push("https://formbricks.com/github");
|
||||||
/* plausible("Hero_CTA_LaunchDemo"); */
|
plausible("Hero_CTA_ViewGitHub");
|
||||||
}}>
|
}}>
|
||||||
View Code on GitHub
|
View Code on GitHub
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -21,7 +21,7 @@ export const Highlights: React.FC = ({}) => {
|
|||||||
significantly higher conversion rate.
|
significantly higher conversion rate.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="rounded-lg bg-slate-100 py-6 pr-4 dark:bg-slate-800 sm:py-16 sm:pr-8">
|
<div className="rounded-lg bg-slate-100 py-6 pr-4 sm:py-16 sm:pr-8 dark:bg-slate-800">
|
||||||
<Image
|
<Image
|
||||||
src={ImageEventTriggerLight}
|
src={ImageEventTriggerLight}
|
||||||
alt="react library"
|
alt="react library"
|
||||||
@@ -39,7 +39,7 @@ export const Highlights: React.FC = ({}) => {
|
|||||||
<div className="mx-auto mb-12 mt-8 max-w-lg md:mb-0 md:mt-32 md:max-w-none">
|
<div className="mx-auto mb-12 mt-8 max-w-lg md:mb-0 md:mt-32 md:max-w-none">
|
||||||
<div className="px-4 sm:max-w-4xl sm:px-6 lg:max-w-7xl lg:px-8">
|
<div className="px-4 sm:max-w-4xl sm:px-6 lg:max-w-7xl lg:px-8">
|
||||||
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
||||||
<div className="order-last rounded-lg bg-slate-100 p-4 dark:bg-slate-800 sm:p-8 md:order-first">
|
<div className="order-last rounded-lg bg-slate-100 p-4 sm:p-8 md:order-first dark:bg-slate-800">
|
||||||
<Image
|
<Image
|
||||||
src={ImageAttributesLight}
|
src={ImageAttributesLight}
|
||||||
alt="react library"
|
alt="react library"
|
||||||
@@ -48,7 +48,7 @@ export const Highlights: React.FC = ({}) => {
|
|||||||
<Image src={ImageAttributesDark} alt="react library" className="hidden rounded-lg dark:block" />
|
<Image src={ImageAttributesDark} alt="react library" className="hidden rounded-lg dark:block" />
|
||||||
</div>
|
</div>
|
||||||
<div className="pb-8 md:pb-0">
|
<div className="pb-8 md:pb-0">
|
||||||
<h2 className="xs:text-3xl text-2xl font-bold leading-7 tracking-tight text-slate-800 dark:text-slate-100 sm:text-3xl">
|
<h2 className="xs:text-3xl text-2xl font-bold leading-7 tracking-tight text-slate-800 sm:text-3xl dark:text-slate-100">
|
||||||
Don't ‘Spray and pray’.
|
Don't ‘Spray and pray’.
|
||||||
<br />
|
<br />
|
||||||
<span className="font-light">Pre-segment granularly.</span>
|
<span className="font-light">Pre-segment granularly.</span>
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { Button } from "@formbricks/ui/Button";
|
|
||||||
import { ArrowUpIcon } from "@heroicons/react/24/solid";
|
import { ArrowUpIcon } from "@heroicons/react/24/solid";
|
||||||
import throttle from "lodash/throttle";
|
import throttle from "lodash/throttle";
|
||||||
import { useCallback, useEffect, useState } from "react";
|
import { useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
import { Button } from "@formbricks/ui/Button";
|
||||||
|
|
||||||
const ScrollToTopButton = () => {
|
const ScrollToTopButton = () => {
|
||||||
const [visible, setVisible] = useState(false);
|
const [visible, setVisible] = useState(false);
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { IoLogoHtml5, IoLogoNpm } from "react-icons/io5";
|
import { IoLogoHtml5, IoLogoNpm } from "react-icons/io5";
|
||||||
|
|
||||||
import CodeBlock from "../shared/CodeBlock";
|
import CodeBlock from "../shared/CodeBlock";
|
||||||
|
|
||||||
interface SecondNavbarProps {
|
interface SecondNavbarProps {
|
||||||
@@ -60,7 +61,7 @@ if (typeof window !== "undefined") {
|
|||||||
</>
|
</>
|
||||||
) : activeTab === "html" ? (
|
) : activeTab === "html" ? (
|
||||||
<CodeBlock>{`<script type="text/javascript">
|
<CodeBlock>{`<script type="text/javascript">
|
||||||
!function(){var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src="https://unpkg.com/@formbricks/js@^1.2.0/dist/index.umd.js";var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e),setTimeout(function(){window.formbricks.init("claDadXk29dak92dK9","https://app.formbricks.com")},500)}();
|
!function(){var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src="https://unpkg.com/@formbricks/js@^1.4.0/dist/index.umd.js";var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e),setTimeout(function(){window.formbricks.init("claDadXk29dak92dK9","https://app.formbricks.com")},500)}();
|
||||||
</script>`}</CodeBlock>
|
</script>`}</CodeBlock>
|
||||||
) : null}
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
import DemoPreview from "@/components/dummyUI/DemoPreview";
|
import DemoPreview from "@/components/dummyUI/DemoPreview";
|
||||||
import DashboardMockupDark from "@/images/dashboard-mockup-dark.png";
|
import DashboardMockupDark from "@/images/dashboard-mockup-dark.png";
|
||||||
import DashboardMockup from "@/images/dashboard-mockup.png";
|
import DashboardMockup from "@/images/dashboard-mockup.png";
|
||||||
import { Button } from "@formbricks/ui/Button";
|
|
||||||
import { CursorArrowRaysIcon } from "@heroicons/react/24/solid";
|
import { CursorArrowRaysIcon } from "@heroicons/react/24/solid";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
|
|
||||||
|
import { Button } from "@formbricks/ui/Button";
|
||||||
|
|
||||||
import AddEventDummy from "../dummyUI/AddEventDummy";
|
import AddEventDummy from "../dummyUI/AddEventDummy";
|
||||||
import AddNoCodeEventModalDummy from "../dummyUI/AddNoCodeEventModalDummy";
|
import AddNoCodeEventModalDummy from "../dummyUI/AddNoCodeEventModalDummy";
|
||||||
import HeadingCentered from "../shared/HeadingCentered";
|
import HeadingCentered from "../shared/HeadingCentered";
|
||||||
@@ -43,7 +45,7 @@ export const Steps: React.FC = () => {
|
|||||||
<div className="mx-auto mb-12 mt-8 max-w-lg md:mb-0 md:mt-32 md:max-w-none">
|
<div className="mx-auto mb-12 mt-8 max-w-lg md:mb-0 md:mt-32 md:max-w-none">
|
||||||
<div className="px-4 sm:max-w-4xl sm:px-6 lg:max-w-7xl lg:px-8">
|
<div className="px-4 sm:max-w-4xl sm:px-6 lg:max-w-7xl lg:px-8">
|
||||||
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
||||||
<div className="order-last w-full rounded-lg bg-slate-100 p-4 dark:bg-slate-800 sm:py-8 md:order-first">
|
<div className="order-last w-full rounded-lg bg-slate-100 p-4 sm:py-8 md:order-first dark:bg-slate-800">
|
||||||
<div className="flex h-40 items-center justify-center">
|
<div className="flex h-40 items-center justify-center">
|
||||||
<Button
|
<Button
|
||||||
variant="primary"
|
variant="primary"
|
||||||
@@ -58,7 +60,7 @@ export const Steps: React.FC = () => {
|
|||||||
</div>
|
</div>
|
||||||
<div className="pb-8 md:pb-0">
|
<div className="pb-8 md:pb-0">
|
||||||
<h4 className="text-brand-dark font-bold">Step 2</h4>
|
<h4 className="text-brand-dark font-bold">Step 2</h4>
|
||||||
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 dark:text-slate-100 sm:text-3xl">
|
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 sm:text-3xl dark:text-slate-100">
|
||||||
No-Code: Track User Actions
|
No-Code: Track User Actions
|
||||||
</h2>
|
</h2>
|
||||||
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
||||||
@@ -74,7 +76,7 @@ export const Steps: React.FC = () => {
|
|||||||
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
||||||
<div className="pb-8 sm:pl-10 md:pb-0">
|
<div className="pb-8 sm:pl-10 md:pb-0">
|
||||||
<h4 className="text-brand-dark font-bold">Step 3</h4>
|
<h4 className="text-brand-dark font-bold">Step 3</h4>
|
||||||
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 dark:text-slate-200 sm:text-3xl">
|
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 sm:text-3xl dark:text-slate-200">
|
||||||
Create your survey
|
Create your survey
|
||||||
</h2>
|
</h2>
|
||||||
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
||||||
@@ -82,7 +84,7 @@ export const Steps: React.FC = () => {
|
|||||||
adjust the look and feel of your survey.
|
adjust the look and feel of your survey.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="relative w-full rounded-lg p-1 dark:bg-slate-800 sm:p-8">
|
<div className="relative w-full rounded-lg p-1 sm:p-8 dark:bg-slate-800">
|
||||||
<DemoPreview template="Product Market Fit Survey (short)" />
|
<DemoPreview template="Product Market Fit Survey (short)" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -91,14 +93,14 @@ export const Steps: React.FC = () => {
|
|||||||
<div className="mx-auto mb-12 mt-8 max-w-lg md:mb-0 md:mt-32 md:max-w-none">
|
<div className="mx-auto mb-12 mt-8 max-w-lg md:mb-0 md:mt-32 md:max-w-none">
|
||||||
<div className="px-4 sm:max-w-4xl sm:px-6 lg:max-w-7xl lg:px-8">
|
<div className="px-4 sm:max-w-4xl sm:px-6 lg:max-w-7xl lg:px-8">
|
||||||
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
||||||
<div className="order-last w-full rounded-lg bg-slate-100 p-4 dark:bg-slate-800 sm:py-8 md:order-first">
|
<div className="order-last w-full rounded-lg bg-slate-100 p-4 sm:py-8 md:order-first dark:bg-slate-800">
|
||||||
<div className="mx-auto flex flex-col items-center justify-center md:w-3/4">
|
<div className="mx-auto flex flex-col items-center justify-center md:w-3/4">
|
||||||
<AddEventDummy />
|
<AddEventDummy />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="pb-8 md:pb-0">
|
<div className="pb-8 md:pb-0">
|
||||||
<h4 className="text-brand-dark font-bold">Step 4</h4>
|
<h4 className="text-brand-dark font-bold">Step 4</h4>
|
||||||
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 dark:text-slate-100 sm:text-3xl">
|
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 sm:text-3xl dark:text-slate-100">
|
||||||
Set segment and trigger
|
Set segment and trigger
|
||||||
</h2>
|
</h2>
|
||||||
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
||||||
@@ -114,7 +116,7 @@ export const Steps: React.FC = () => {
|
|||||||
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
<div className="grid md:grid-cols-2 md:items-center md:gap-16">
|
||||||
<div className="pb-8 sm:pl-10 md:pb-0">
|
<div className="pb-8 sm:pl-10 md:pb-0">
|
||||||
<h4 className="text-brand-dark font-bold">Step 5</h4>
|
<h4 className="text-brand-dark font-bold">Step 5</h4>
|
||||||
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 dark:text-slate-200 sm:text-3xl">
|
<h2 className="xs:text-3xl text-2xl font-bold tracking-tight text-slate-800 sm:text-3xl dark:text-slate-200">
|
||||||
Make better decisions
|
Make better decisions
|
||||||
</h2>
|
</h2>
|
||||||
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
<p className="text-md mt-6 max-w-lg leading-7 text-slate-500 dark:text-slate-400">
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { ResponsiveVideo } from "@formbricks/ui/ResponsiveVideo";
|
|
||||||
import { Modal } from "@formbricks/ui/Modal";
|
import { Modal } from "@formbricks/ui/Modal";
|
||||||
|
import { ResponsiveVideo } from "@formbricks/ui/ResponsiveVideo";
|
||||||
|
|
||||||
interface VideoWalkThroughProps {
|
interface VideoWalkThroughProps {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import AuthorJohannes from "@/images/blog/johannes-co-founder-formbricks-small.jpg";
|
import AuthorJohannes from "@/images/blog/johannes-co-founder-formbricks-small.jpg";
|
||||||
|
import AuthorOla from "@/images/blog/ola-content-writer.jpg";
|
||||||
import Image from "next/image";
|
import Image from "next/image";
|
||||||
|
|
||||||
interface AuthorBoxProps {
|
interface AuthorBoxProps {
|
||||||
@@ -6,14 +7,15 @@ interface AuthorBoxProps {
|
|||||||
title: string;
|
title: string;
|
||||||
date: string;
|
date: string;
|
||||||
duration: string;
|
duration: string;
|
||||||
|
author: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function AuthorBox({ name, title, date, duration }: AuthorBoxProps) {
|
export default function AuthorBox({ name, title, date, duration, author }: AuthorBoxProps) {
|
||||||
return (
|
return (
|
||||||
<div className="mb-8 flex items-center space-x-4 rounded-lg border border-slate-200 bg-slate-100 px-6 py-3 dark:border-slate-700 dark:bg-slate-800">
|
<div className="mb-8 flex items-center space-x-4 rounded-lg border border-slate-200 bg-slate-100 px-6 py-3 dark:border-slate-700 dark:bg-slate-800">
|
||||||
<Image
|
<Image
|
||||||
className="m-0 rounded-full"
|
className="m-0 rounded-full"
|
||||||
src={AuthorJohannes}
|
src={author === "Johannes" ? AuthorJohannes : AuthorOla}
|
||||||
alt={name}
|
alt={name}
|
||||||
width={45}
|
width={45}
|
||||||
height={45}
|
height={45}
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
|
import clsx from "clsx";
|
||||||
|
import Link from "next/link";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
BaseballIcon,
|
BaseballIcon,
|
||||||
CancelSubscriptionIcon,
|
CancelSubscriptionIcon,
|
||||||
@@ -8,8 +11,6 @@ import {
|
|||||||
OnboardingIcon,
|
OnboardingIcon,
|
||||||
PMFIcon,
|
PMFIcon,
|
||||||
} from "@formbricks/ui/icons";
|
} from "@formbricks/ui/icons";
|
||||||
import clsx from "clsx";
|
|
||||||
import Link from "next/link";
|
|
||||||
|
|
||||||
export default function BestPracticeNavigation() {
|
export default function BestPracticeNavigation() {
|
||||||
const BestPractices = [
|
const BestPractices = [
|
||||||
@@ -83,8 +84,8 @@ export default function BestPracticeNavigation() {
|
|||||||
return (
|
return (
|
||||||
<div className="mx-auto grid grid-cols-1 gap-6 px-2 md:grid-cols-3">
|
<div className="mx-auto grid grid-cols-1 gap-6 px-2 md:grid-cols-3">
|
||||||
{BestPractices.map((bestPractice) => (
|
{BestPractices.map((bestPractice) => (
|
||||||
<Link href={bestPractice.href} key={bestPractice.name}>
|
<Link className="relative block" href={bestPractice.href} key={bestPractice.name}>
|
||||||
<div className="drop-shadow-card duration-120 hover:border-brand-dark relative rounded-lg border border-slate-100 bg-slate-100 p-6 transition-all ease-in-out hover:scale-105 hover:cursor-pointer dark:border-slate-600 dark:bg-slate-800">
|
<div className="drop-shadow-card duration-120 hover:border-brand-dark relative h-full rounded-lg border border-slate-100 bg-slate-100 p-6 transition-all ease-in-out hover:scale-105 hover:cursor-pointer dark:border-slate-600 dark:bg-slate-800">
|
||||||
<div
|
<div
|
||||||
className={clsx(
|
className={clsx(
|
||||||
// base styles independent what type of button it is
|
// base styles independent what type of button it is
|
||||||
@@ -105,7 +106,9 @@ export default function BestPracticeNavigation() {
|
|||||||
<h3 className="mb-1 mt-3 text-xl font-bold text-slate-700 dark:text-slate-200">
|
<h3 className="mb-1 mt-3 text-xl font-bold text-slate-700 dark:text-slate-200">
|
||||||
{bestPractice.name}
|
{bestPractice.name}
|
||||||
</h3>
|
</h3>
|
||||||
<p className="text-sm text-slate-600 dark:text-slate-400">{bestPractice.description}</p>
|
<p className="flex self-end text-sm text-slate-600 dark:text-slate-400">
|
||||||
|
{bestPractice.description}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</Link>
|
</Link>
|
||||||
))}
|
))}
|
||||||
|
|||||||