mirror of
https://github.com/formbricks/formbricks.git
synced 2026-02-11 10:38:11 -06:00
Move repository into a monorepo with turborepo and pnpm. This is a big change in the way the code is organized, used and deployed.
129 lines
5.4 KiB
TypeScript
129 lines
5.4 KiB
TypeScript
import { API, BlockTool, BlockToolData, ToolConfig } from "@editorjs/editorjs";
|
|
import { GlobeAltIcon } from "@heroicons/react/24/solid";
|
|
import ReactDOM from "react-dom";
|
|
|
|
//styles imports in angular.json
|
|
interface WebsiteQuestionData extends BlockToolData {
|
|
label: string;
|
|
help: string;
|
|
placeholder: string;
|
|
required: boolean;
|
|
}
|
|
|
|
export default class WebsiteQuestion implements BlockTool {
|
|
settings: { name: string; icon: string }[];
|
|
api: API;
|
|
data: any;
|
|
wrapper: undefined | HTMLElement;
|
|
|
|
static get toolbox(): { icon: string; title?: string } {
|
|
return {
|
|
icon: `<svg xmlns="http://www.w3.org/2000/svg" class="h-5 w-5" viewBox="0 0 20 20" fill="currentColor">
|
|
<path fill-rule="evenodd" d="M4.083 9h1.946c.089-1.546.383-2.97.837-4.118A6.004 6.004 0 004.083 9zM10 2a8 8 0 100 16 8 8 0 000-16zm0 2c-.076 0-.232.032-.465.262-.238.234-.497.623-.737 1.182-.389.907-.673 2.142-.766 3.556h3.936c-.093-1.414-.377-2.649-.766-3.556-.24-.56-.5-.948-.737-1.182C10.232 4.032 10.076 4 10 4zm3.971 5c-.089-1.546-.383-2.97-.837-4.118A6.004 6.004 0 0115.917 9h-1.946zm-2.003 2H8.032c.093 1.414.377 2.649.766 3.556.24.56.5.948.737 1.182.233.23.389.262.465.262.076 0 .232-.032.465-.262.238-.234.498-.623.737-1.182.389-.907.673-2.142.766-3.556zm1.166 4.118c.454-1.147.748-2.572.837-4.118h1.946a6.004 6.004 0 01-2.783 4.118zm-6.268 0C6.412 13.97 6.118 12.546 6.03 11H4.083a6.004 6.004 0 002.783 4.118z" clip-rule="evenodd" />
|
|
</svg>`,
|
|
title: "Website Question",
|
|
};
|
|
}
|
|
|
|
constructor({ data }: { api: API; config?: ToolConfig; data?: WebsiteQuestionData }) {
|
|
this.wrapper = undefined;
|
|
this.settings = [
|
|
{
|
|
name: "required",
|
|
icon: `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 480 512" class="w-3 h-3"><path d="M471.99 334.43L336.06 256l135.93-78.43c7.66-4.42 10.28-14.2 5.86-21.86l-32.02-55.43c-4.42-7.65-14.21-10.28-21.87-5.86l-135.93 78.43V16c0-8.84-7.17-16-16.01-16h-64.04c-8.84 0-16.01 7.16-16.01 16v156.86L56.04 94.43c-7.66-4.42-17.45-1.79-21.87 5.86L2.15 155.71c-4.42 7.65-1.8 17.44 5.86 21.86L143.94 256 8.01 334.43c-7.66 4.42-10.28 14.21-5.86 21.86l32.02 55.43c4.42 7.65 14.21 10.27 21.87 5.86l135.93-78.43V496c0 8.84 7.17 16 16.01 16h64.04c8.84 0 16.01-7.16 16.01-16V339.14l135.93 78.43c7.66 4.42 17.45 1.8 21.87-5.86l32.02-55.43c4.42-7.65 1.8-17.43-5.86-21.85z"/></svg>`,
|
|
},
|
|
];
|
|
this.data = {
|
|
label: data.label || "",
|
|
help: data.help || "",
|
|
placeholder: data.placeholder || "https://",
|
|
required: data.required !== undefined ? data.required : true,
|
|
};
|
|
}
|
|
|
|
save(block: HTMLDivElement) {
|
|
return {
|
|
...this.data,
|
|
label: (block.firstElementChild.firstElementChild.firstElementChild as HTMLInputElement).value,
|
|
placeholder: (
|
|
(block.firstElementChild.childNodes[1] as HTMLInputElement).lastElementChild as HTMLInputElement
|
|
).value,
|
|
help: (block.firstElementChild.lastElementChild as HTMLInputElement).value,
|
|
};
|
|
}
|
|
|
|
renderSettings(): HTMLElement {
|
|
const wrapper = document.createElement("div");
|
|
|
|
this.settings.forEach((tune) => {
|
|
let button = document.createElement("div");
|
|
|
|
button.classList.add("cdx-settings-button");
|
|
button.classList.toggle("cdx-settings-button--active", this.data[tune.name]);
|
|
button.innerHTML = tune.icon;
|
|
wrapper.appendChild(button);
|
|
|
|
button.addEventListener("click", () => {
|
|
this._toggleTune(tune.name);
|
|
button.classList.toggle("cdx-settings-button--active");
|
|
});
|
|
});
|
|
|
|
return wrapper;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
* Click on the Settings Button
|
|
* @param {string} tune — tune name from this.settings
|
|
*/
|
|
_toggleTune(tune) {
|
|
this.wrapper.classList.toggle(tune.name, !!this.data[tune.name]);
|
|
|
|
if (tune === "required") {
|
|
this.data.required = !this.data.required;
|
|
this.wrapper.childNodes[0].childNodes[0].childNodes[1].textContent = this.data.required ? "*" : "";
|
|
}
|
|
}
|
|
|
|
render(): HTMLElement {
|
|
this.wrapper = document.createElement("div");
|
|
const toolView = (
|
|
<div className="pb-5">
|
|
<div className="text-md relative font-bold leading-7 text-gray-800 sm:truncate">
|
|
<input
|
|
type="text"
|
|
id="label"
|
|
defaultValue={this.data.label}
|
|
className="w-full border-0 border-transparent p-0 ring-0 placeholder:text-gray-300 focus:ring-0"
|
|
placeholder="Your Question"
|
|
/>
|
|
<div className="pointer-events-none absolute inset-y-0 right-0 flex items-center pr-3 text-red-500">
|
|
*
|
|
</div>
|
|
</div>
|
|
<div className="relative mt-1 max-w-sm rounded-md shadow-sm">
|
|
<div className="pointer-events-none absolute inset-y-0 left-0 flex items-center pl-3">
|
|
<GlobeAltIcon className="h-5 w-5 text-gray-400" aria-hidden="true" />
|
|
</div>
|
|
<input
|
|
type="text"
|
|
name="website"
|
|
className="block w-full rounded-md border-gray-300 pl-10 text-gray-300 sm:text-sm"
|
|
defaultValue={this.data.placeholder}
|
|
/>
|
|
</div>
|
|
<input
|
|
type="text"
|
|
id="help-text"
|
|
defaultValue={this.data.help}
|
|
className="mt-2 block w-full max-w-sm border-0 border-transparent p-0 text-sm font-light text-gray-500 ring-0 placeholder:text-gray-300 focus:ring-0"
|
|
placeholder="optional help text"
|
|
/>
|
|
</div>
|
|
);
|
|
ReactDOM.render(toolView, this.wrapper);
|
|
return this.wrapper;
|
|
}
|
|
}
|