Aknaproff/resources/js/components/FormTab.vue
2024-03-18 19:33:31 +05:00

1104 lines
46 KiB
Vue

<template>
<div class="row mt-3">
<div class="col-md-2" id="tour_step_1">
<div class="element-sidebar">
<div class="card">
<div class="card-header">
<p class="card-title">
{{ trans("messages.elements") }}
</p>
</div>
<div class="collapse show" id="basicElements">
<div
class="card-body pr-1 pl-1 elements_sidebar_height"
>
<draggable
class="dragArea list-group"
:list="basic_elements"
:group="{
name: 'element',
pull: 'clone',
put: false,
}"
:clone="cloneElement"
:sort="false"
@change="change"
>
<div
class="list-group-item"
v-for="element in basic_elements"
:key="element.type"
>
<button
type="button"
class="btn btn-primary btn-block"
:class="[
_.includes(
['card_form'],
settings.layout
) &&
_.includes(
['page_break'],
element.type
)
? ''
: 'hvr-grow',
]"
:title="element.tooltip"
:disabled="
_.includes(
['card_form'],
settings.layout
) &&
_.includes(
['page_break'],
element.type
)
"
>
<i
class="mt-1 mb-1"
:class="
'float-left fas fa-' +
element.display_icon
"
></i>
{{ element.label }}
<i
v-if="
_.includes(
['card_form'],
settings.layout
) &&
_.includes(
['page_break'],
element.type
)
"
class="fas fa-info-circle float-right mt-1 mb-1"
data-toggle="tooltip"
:title="
trans(
'messages.page_break_disabled'
)
"
></i>
</button>
</div>
</draggable>
</div>
</div>
</div>
</div>
</div>
<div
class="col-md-7"
:style="{
'background-color': settings.color.page_color,
padding: `10px`,
}"
>
<div
class="card scrollableCard"
:style="{
background: getBackgroundStyle(),
'background-image': `${backgroundImg}`,
}"
style="height: 100%"
id="tour_step_2"
>
<div
class="text-center mt-2"
v-if="selected_elements.length == 0"
>
<h5>{{ trans("messages.pls_add_element") }}</h5>
</div>
<draggable
class="dragArea card-body"
:list="selected_elements"
group="element"
handle=".handle"
@change="change"
:animation="200"
>
<transition-group
name="custom-classes-transition"
enter-active-class="animated zoomIn"
leave-active-class="animated fadeOut"
tag="div"
class="row"
:class="[
selected_elements.length < 3
? 'transition-card-body'
: '',
]"
>
<div
v-for="(element, index) in selected_elements"
:key="element.id"
:class="element.col"
@click="toggleConfigurator(index, true)"
>
<fieldGenerator
:element="element"
:settings="settings"
:applyValidations="applyValidations"
action=""
></fieldGenerator>
<div
class="element-config-action-btn float-right"
v-show="element.extras.showConfigurator"
>
<p>
<button
type="button"
v-if="
!_.includes(
['page_break'],
element.type
)
"
class="btn btn-xs btn-secondary"
:title="trans('messages.properties')"
@click="openElementConfigurator()"
>
<i class="fas fa-cog"></i>
</button>
</p>
<p>
<button
type="button"
class="btn btn-xs btn-danger"
:title="trans('messages.remove')"
@click="deleteElement(index)"
>
<i class="far fa-trash-alt"></i>
</button>
</p>
</div>
</div>
</transition-group>
</draggable>
</div>
</div>
<div class="col-md-3" id="tour_step_3">
<div class="element-sidebar">
<ul class="nav nav-tabs" role="tablist">
<li class="nav-item">
<a
class="nav-link"
id="formDetails-tab"
data-toggle="tab"
href="#formDetails"
role="tab"
:class="isConfiguratorOpen ? '' : 'active'"
aria-controls="formDetails"
aria-selected="true"
>
{{ trans("messages.form") }}
</a>
</li>
<li class="nav-item">
<a
class="nav-link"
id="configuration-tab"
:class="isConfiguratorOpen ? 'active' : ''"
data-toggle="tab"
href="#configuration"
role="tab"
aria-controls="configuration"
aria-selected="false"
>
{{ trans("messages.element_configuration") }}
</a>
</li>
</ul>
<div class="tab-content" id="tab_content" role="tabpanel">
<div
class="tab-pane fade"
:class="isConfiguratorOpen ? '' : 'active show'"
id="formDetails"
role="tabpanel"
aria-labelledby="formDetails-tab"
>
<div class="card mt-1 configurator_form_height">
<div class="card-body">
<div class="row">
<div class="col-md-12 mb-2">
<label>
{{ trans("messages.form_name") }}
<span class="error">*</span>
</label>
<input
type="text"
class="form-control"
v-model="form.name"
required
@change="
generateFormSlug(form.name)
"
/>
</div>
</div>
<div class="row">
<div class="col-md-12 mb-2">
<label>
{{ trans("messages.form_name_ru") }}
<span class="error">*</span>
</label>
<input
type="text"
class="form-control"
v-model="form.name_ru"
required
/>
</div>
</div>
<div class="row">
<div class="col-md-12 mb-2">
<label>
{{
trans("messages.form_name_est")
}}
<span class="error">*</span>
</label>
<input
type="text"
class="form-control"
v-model="form.name_est"
required
/>
</div>
</div>
<div class="row">
<div class="col-md-12 mb-2">
<label for="form_slug">
{{ trans("messages.form_slug") }}
<span class="error">*</span>
</label>
<input
type="text"
class="form-control"
v-model="form_slug"
required
readonly
/>
</div>
</div>
<div class="row mb-1">
<div class="col-md-12">
<label>
{{
trans(
"messages.form_description"
)
}}
</label>
<input
type="text"
class="form-control"
v-model="form.description"
/>
</div>
</div>
<div class="row mb-1">
<div class="col-md-12">
<label>
{{
trans(
"messages.form_description_ru"
)
}}
</label>
<input
type="text"
class="form-control"
v-model="form.description_ru"
/>
</div>
</div>
<div class="row mb-1">
<div class="col-md-12">
<label>
{{
trans(
"messages.form_description_est"
)
}}
</label>
<input
type="text"
class="form-control"
v-model="form.description_est"
/>
</div>
</div>
<form-layout
:selected_elements="selected_elements"
:settings="settings"
>
</form-layout>
<div class="row">
<!-- form custom attribute -->
<div
class="col-md-12"
v-if="form_custom_attributes.length"
>
<hr />
<label>{{
trans("messages.custom_attributes")
}}</label>
</div>
<div
class="row form_custom_attr_field"
v-for="(
form_custom_attribute, index
) in form_custom_attributes"
>
<div class="col-md-10">
<div class="input-group">
<input
type="text"
class="form-control"
v-model="
form_custom_attribute.key
"
:placeholder="
trans('messages.key')
"
/>
<input
type="text"
class="form-control"
v-model="
form_custom_attribute.value
"
:placeholder="
trans('messages.value')
"
/>
</div>
</div>
<div class="col-md-2">
<button
type="button"
class="btn btn-danger btn-sm"
@click="removeAttribute(index)"
>
<i class="fas fa-trash-alt"></i>
</button>
</div>
</div>
<button
type="button"
class="btn btn-sm btn-info"
@click="addMoreAttribute"
style="margin: 28px"
>
<i class="fas fa-plus-circle"></i>
{{
trans(
"messages.add_form_custom_attribute"
)
}}
</button>
</div>
</div>
</div>
</div>
<div
class="tab-pane fade mb-2"
:class="isConfiguratorOpen ? 'active show' : ''"
id="configuration"
role="tabpanel"
aria-labelledby="configuration-tab"
>
<template
v-for="(element, index) in selected_elements"
v-if="
element.extras.showConfigurator &&
isConfiguratorOpen
"
>
<fieldConfigurator
v-if="!_.includes(['page_break'], element.type)"
:key="element.id"
:element="element"
:index="index"
:selected_elements="selected_elements"
v-on:toggleConfigurator="
toggleConfiguratorEvent
"
v-on:deleteElement="deleteElement"
></fieldConfigurator>
<div
v-if="_.includes(['page_break'], element.type)"
class="mt-1 ml-3"
>
{{ trans("messages.no_config_option") }}
</div>
</template>
<div v-if="!isConfiguratorOpen">
<div class="mt-1 ml-3">
{{
trans(
"messages.pls_add_element_to_configure"
)
}}
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</template>
<script>
import draggable from "../../../node_modules/vuedraggable";
import fieldGenerator from "./FieldGenerator";
import fieldConfigurator from "./FieldConfigurator";
import ElementDetails from "./ElementDetails";
import FormLayout from "./Shared/Layout";
let idGlobal = 0;
export default {
props: {
selected_elements: Array,
settings: Object,
form: Array,
placeholder_img: String,
form_custom_attributes: Array,
},
components: {
draggable,
fieldGenerator,
fieldConfigurator,
ElementDetails,
FormLayout,
},
data() {
const self = this;
return {
applyValidations: false,
basic_elements: [
{
type: "text",
subtype: "text",
label: self.trans("messages.input"),
label_est: self.trans("messages.input"),
label_ru: self.trans("messages.input"),
help_text: "",
display_icon: "grip-lines",
prefix_icon: "none",
suffix_icon: "none",
placeholder: "",
placeholder_est: "",
placeholder_ru: "",
size: "",
custom_class: "",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.text_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
allowed_input: {
values: "",
error_msg: "This value is not allowed.",
},
},
{
type: "textarea",
label: self.trans("messages.textarea"),
label_est: self.trans("messages.textarea"),
label_ru: self.trans("messages.textarea"),
help_text: "",
display_icon: "list",
prefix_icon: "none",
suffix_icon: "none",
rows: 3,
columns: "",
placeholder: "",
placeholder_est: "",
placeholder_ru: "",
custom_class: "",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.textarea_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "dropdown",
label: self.trans("messages.dropdown"),
label_est: self.trans("messages.dropdown"),
label_ru: self.trans("messages.dropdown"),
help_text: "",
display_icon: "caret-square-down",
prefix_icon: "none",
suffix_icon: "none",
options: "Option-1\nOption-2",
size: "",
custom_class: "",
conditional_class: "",
multiselect: false,
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.dropdown_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "radio",
label: self.trans("messages.radio"),
label_est: self.trans("messages.radio"),
label_ru: self.trans("messages.radio"),
help_text: "",
spread_to_col: {
enable: false,
column: 2,
},
display_icon: "dot-circle",
options: "Option-1\nOption-2",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.radio_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "checkbox",
label: self.trans("messages.checkbox"),
label_est: self.trans("messages.checkbox"),
label_ru: self.trans("messages.checkbox"),
help_text: "",
spread_to_col: {
enable: false,
column: 2,
},
display_icon: "check-circle",
options: "Option-1\nOption-2",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.checkbox_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "heading",
label: self.trans("messages.heading_paragrahp"),
label_est: self.trans("messages.heading_paragrahp"),
label_ru: self.trans("messages.heading_paragrahp"),
tag: "h1",
text_color: "#212529",
content: "Click to change it",
display_icon: "heading",
col: "col-md-12",
tooltip: self.trans("messages.heading_paragrahp_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "range",
label: self.trans("messages.range"),
label_est: self.trans("messages.range"),
label_ru: self.trans("messages.range"),
help_text: "",
display_icon: "arrows-alt-h",
min: 1,
max: 10,
step: 1,
data_orientation: "horizontal",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.range_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "calendar",
label: self.trans("messages.datetime"),
label_est: self.trans("messages.datetime"),
label_ru: self.trans("messages.datetime"),
help_text: "",
size: "",
display_icon: "calendar-alt",
prefix_icon: "none",
suffix_icon: "none",
custom_class: "",
start_date: "none",
end_date: "none",
format: "MM-DD-YYYY",
disabled_days: [],
enable_time_picker: false,
time_format: "12",
time_picker_inline: false,
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.datetime_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "file_upload",
label: self.trans("messages.file_upload"),
label_est: self.trans("messages.file_upload"),
label_ru: self.trans("messages.file_upload"),
help_text: "",
display_icon: "upload",
upload_text: self.trans("messages.drop_a_file_here"),
no_of_files: "1",
send_as_email_attachment: false,
file_size_limit: 5,
file_type: "all",
allowed_file_type: "",
custom_class: "",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.file_upload_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "text_editor",
label: self.trans("messages.text_editor"),
label_est: self.trans("messages.text_editor"),
label_ru: self.trans("messages.text_editor"),
placeholder: self.trans("messages.jot_down_here"),
placeholder_est: self.trans("messages.jot_down_here"),
placeholder_ru: self.trans("messages.jot_down_here"),
help_text: "",
display_icon: "keyboard",
editor_height: 150,
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.text_editor_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "terms_and_condition",
label: self.trans("messages.terms_condition"),
label_est: self.trans("messages.terms_condition"),
label_ru: self.trans("messages.terms_condition"),
link: "",
display_icon: "file-signature",
custom_class: "",
conditional_class: "",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.terms_condition_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "rating",
label: self.trans("messages.rating"),
label_est: self.trans("messages.rating"),
label_ru: self.trans("messages.rating"),
display_icon: "star",
conditional_class: "",
stars_to_display: 5,
min_rating: 0,
max_rating: 5,
increment: 0.5,
direction: "ltr",
size: "md",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.rating_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "switch",
label: self.trans("messages.switch"),
label_est: self.trans("messages.switch"),
label_ru: self.trans("messages.switch"),
display_icon: "toggle-off",
conditional_class: "",
help_text: "",
size: "switch",
col: "col-md-12",
custom_attributes: [],
tooltip: self.trans("messages.switch_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "hr",
label: self.trans("messages.horizontal_line"),
label_est: self.trans("messages.horizontal_line"),
label_ru: self.trans("messages.horizontal_line"),
padding_top: 1,
padding_bottom: 1,
border_size: 1,
horizontal_space: 0,
border_type: "solid",
border_color: "#0a0a0a",
display_icon: "window-minimize",
col: "col-md-12",
tooltip: self.trans("messages.hr_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "html_text",
label: self.trans("messages.html_text"),
label_est: self.trans("messages.html_text"),
label_ru: self.trans("messages.html_text"),
html_text: "Write Something..",
display_icon: "code",
custom_class: "",
col: "col-md-12",
tooltip: self.trans("messages.html_text_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "signature",
label: self.trans("messages.signature"),
label_est: self.trans("messages.signature"),
label_ru: self.trans("messages.signature"),
display_icon: "signature",
col: "col-md-12",
html_text: "",
custom_class: "",
custom_attributes: [],
tooltip: self.trans("messages.signature_tooltip"),
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "page_break",
label: self.trans("messages.page_break"),
label_est: self.trans("messages.page_break"),
label_ru: self.trans("messages.page_break"),
label_align: "top",
display_icon: "cut",
col: "col-md-12",
tooltip: "",
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "youtube",
label: self.trans("messages.youtube"),
label_est: self.trans("messages.youtube"),
label_ru: self.trans("messages.youtube"),
display_icon: "video",
custom_class: "",
conditional_class: "",
col: "col-md-12",
tooltip: "",
iframe_url: "",
width: 100,
height: 350,
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "iframe",
label: self.trans("messages.iframe_embed"),
label_est: self.trans("messages.iframe_embed"),
label_ru: self.trans("messages.iframe_embed"),
display_icon: "crop-alt",
custom_class: "",
conditional_class: "",
col: "col-md-12",
iframe_url: "https://www.lipsum.com",
iframe_align: "left",
css: "",
width: 100,
height: 400,
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "pdf",
label: self.trans("messages.pdf_embedder"),
label_est: self.trans("messages.pdf_embedder"),
label_ru: self.trans("messages.pdf_embedder"),
display_icon: "file-pdf",
custom_class: "",
conditional_class: "",
col: "col-md-12",
tooltip: "",
pdf: "",
width: 100,
height: 350,
popover_help_text: {
enable: false,
content: "",
},
},
{
type: "countdown",
label: self.trans("messages.countdown"),
label_est: self.trans("messages.countdown"),
label_ru: self.trans("messages.countdown"),
display_icon: "hourglass-start",
custom_class: "",
conditional_class: "",
col: "col-md-12",
tooltip: "",
seconds: 0,
minutes: 0,
hours: 1,
labels_format: true,
size: "lg",
border_color: "#F0068E",
font_color: "#FFFFFF",
bg_color: "#000000",
time_separator: ":",
display_format: "HMS",
popover_help_text: {
enable: false,
content: "",
},
},
],
isConfiguratorOpen: false,
form_slug: "",
};
},
computed: {
backgroundImg() {
if (this.selected_elements.length == 0) {
return "url(" + this.placeholder_img + ")";
} else {
return "";
}
},
},
created() {
if (this.selected_elements.length > 0) {
var ids = [];
_.forEach(this.selected_elements, function (element) {
ids.push(element.id);
});
var id = _.max(ids);
idGlobal = ++id;
}
this.form_slug = this.form.slug;
},
methods: {
addMoreAttribute() {
var custom_attribute = { key: "", value: "" };
this.form_custom_attributes.push(custom_attribute);
},
removeAttribute(index) {
this.form_custom_attributes.splice(index, 1);
},
getBackgroundStyle() {
if (this.settings.background.bg_type !== "bg_image") {
return this.settings.color.background;
} else {
return "";
}
},
change: function (evt) {
window.console.log(evt);
if (evt.added != undefined) {
//find index of an added element
var index = this.selected_elements.indexOf(evt.added.element);
this.toggleConfigurator(index, true);
//add field for col visibility in datatable
if (
!_.includes(
[
"terms_and_condition",
"heading",
"hr",
"html_text",
"page_break",
"youtube",
"iframe",
"pdf",
"countdown",
],
evt.added.element.type
)
) {
this.settings.form_data.col_visible.push(
evt.added.element.name
);
}
}
},
cloneElement(element_details) {
var details_cloned = _.clone(element_details);
var data = {
id: idGlobal++,
name: "field_" + idGlobal,
extras: { showConfigurator: false },
};
_.unset(details_cloned, "id");
_.unset(details_cloned, "name");
_.unset(details_cloned, "display_icon");
_.unset(details_cloned, "tooltip");
var output = _.merge(data, details_cloned);
output["required"] = false;
output["required_error_msg"] = "This field is required";
output["validations"] = [];
return output;
},
toggleConfigurator(index, show) {
//Check if any previously open, then close it
const self = this;
if (show == true) {
_.forEach(this.selected_elements, function (value, key) {
self.selected_elements[key].extras.showConfigurator = false;
});
}
this.selected_elements[index].extras.showConfigurator = show;
},
toggleConfiguratorEvent(values) {
this.toggleConfigurator(values.index, values.show);
this.openElementConfigurator();
},
openElementConfigurator() {
this.isConfiguratorOpen = !this.isConfiguratorOpen;
},
deleteElement(index) {
const self = this;
Swal.fire({
title: self.trans("messages.are_you_sure"),
text: self.trans("messages.you_wont_be_able_to_revert"),
icon: "warning",
showCancelButton: true,
confirmButtonColor: "#3085d6",
cancelButtonColor: "#d33",
confirmButtonText: self.trans("messages.yes_remove_it"),
}).then((result) => {
if (result.isConfirmed) {
let element = self.selected_elements[index];
let deletedElementIndex =
self.settings.form_data.col_visible.indexOf(
element.name
);
self.selected_elements.splice(index, 1);
self.isConfiguratorOpen = false;
if (deletedElementIndex != "-1") {
self.settings.form_data.col_visible.splice(index, 1);
}
}
});
},
generateFormSlug(name) {
const self = this;
axios
.get("/generate-form-slug", {
params: {
name: name,
},
})
.then(function (response) {
self.form.slug = response.data;
self.form_slug = self.form.slug;
})
.catch(function (error) {
console.log(error);
});
},
},
};
</script>
<style scoped>
.scrollableCard {
display: flex;
flex-direction: column;
height: 200vh;
min-height: 392px;
padding-right: 25px;
padding-left: 15px;
}
#tab_content {
border-right: 1px solid #d2e3f5;
border-left: 1px solid #d2e3f5;
border-bottom: 1px solid #d2e3f5;
}
.card-title {
font-size: 1rem !important;
}
.list-group-item {
border: 0px !important;
padding-left: 0.25rem !important;
padding-right: 0.25rem !important;
}
.element-sidebar {
position: -webkit-sticky;
position: sticky;
top: 0rem;
height: 58vh;
}
.elements_sidebar_height {
overflow-y: auto;
max-height: 58vh;
}
.configurator_form_height {
overflow-y: auto;
max-height: 58vh;
}
#formDetails {
height: 58vh;
}
.transition-card-body {
min-height: 392px;
}
</style>