mirror of https://github.com/docusealco/docuseal
				
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							332 lines
						
					
					
						
							9.5 KiB
						
					
					
				
			
		
		
	
	
							332 lines
						
					
					
						
							9.5 KiB
						
					
					
				| <template>
 | |
|   <div
 | |
|     class="flex absolute lg:text-base"
 | |
|     :style="computedStyle"
 | |
|     :class="{ 'text-[1.5vw] lg:text-base': !textOverflowChars, 'text-[1.0vw] lg:text-xs': textOverflowChars, 'cursor-default': !submittable, 'bg-red-100 border cursor-pointer ': submittable, 'border-red-100': !isActive && submittable, 'bg-opacity-70': !isActive && !isValueSet && submittable, 'border-red-500 border-dashed z-10': isActive && submittable, 'bg-opacity-30': (isActive || isValueSet) && submittable }"
 | |
|   >
 | |
|     <div
 | |
|       v-if="!isActive && !isValueSet && field.type !== 'checkbox' && submittable && !area.option_uuid"
 | |
|       class="absolute top-0 bottom-0 right-0 left-0 items-center justify-center h-full w-full"
 | |
|     >
 | |
|       <span
 | |
|         v-if="field"
 | |
|         class="flex justify-center items-center h-full opacity-50"
 | |
|       >
 | |
|         <component
 | |
|           :is="fieldIcons[field.type]"
 | |
|           width="100%"
 | |
|           height="100%"
 | |
|           class="max-h-10 text-base-content"
 | |
|         />
 | |
|       </span>
 | |
|     </div>
 | |
|     <div
 | |
|       v-if="isActive && withLabel && !area.option_uuid"
 | |
|       class="absolute -top-7 rounded bg-base-content text-base-100 px-2 text-sm whitespace-nowrap"
 | |
|     >
 | |
|       {{ field.name || fieldNames[field.type] }}
 | |
|       <template v-if="field.type === 'checkbox' && !field.name">
 | |
|         {{ fieldIndex + 1 }}
 | |
|       </template>
 | |
|       <template v-else-if="!field.required && field.type !== 'checkbox'">
 | |
|         (optional)
 | |
|       </template>
 | |
|     </div>
 | |
|     <div
 | |
|       v-if="isActive"
 | |
|       ref="scrollToElem"
 | |
|       class="absolute -top-20"
 | |
|     />
 | |
|     <img
 | |
|       v-if="field.type === 'image' && image"
 | |
|       class="object-contain mx-auto"
 | |
|       :src="image.url"
 | |
|     >
 | |
|     <img
 | |
|       v-else-if="field.type === 'signature' && signature"
 | |
|       class="object-contain mx-auto"
 | |
|       :src="signature.url"
 | |
|     >
 | |
|     <img
 | |
|       v-else-if="field.type === 'initials' && initials"
 | |
|       class="object-contain mx-auto"
 | |
|       :src="initials.url"
 | |
|     >
 | |
|     <div
 | |
|       v-else-if="field.type === 'file'"
 | |
|       class="px-0.5 flex flex-col justify-center"
 | |
|     >
 | |
|       <a
 | |
|         v-for="(attachment, index) in attachments"
 | |
|         :key="index"
 | |
|         target="_blank"
 | |
|         :href="attachment.url"
 | |
|       >
 | |
|         <IconPaperclip
 | |
|           class="inline w-[1.5vw] h-[1.5vw] lg:w-4 lg:h-4"
 | |
|         />
 | |
|         {{ attachment.filename }}
 | |
|       </a>
 | |
|     </div>
 | |
|     <div
 | |
|       v-else-if="field.type === 'checkbox'"
 | |
|       class="w-full p-[0.2vw] flex items-center justify-center"
 | |
|     >
 | |
|       <input
 | |
|         v-if="submittable"
 | |
|         type="checkbox"
 | |
|         :value="false"
 | |
|         class="aspect-square base-checkbox"
 | |
|         :class="{ '!w-auto !h-full': area.w > area.h, '!w-full !h-auto': area.w <= area.h }"
 | |
|         :checked="!!modelValue"
 | |
|         @click="$emit('update:model-value', !modelValue)"
 | |
|       >
 | |
|       <IconCheck
 | |
|         v-else-if="modelValue"
 | |
|         class="aspect-square"
 | |
|         :class="{ '!w-auto !h-full': area.w > area.h, '!w-full !h-auto': area.w <= area.h }"
 | |
|       />
 | |
|     </div>
 | |
|     <div
 | |
|       v-else-if="field.type === 'radio' && area.option_uuid"
 | |
|       class="w-full p-[0.2vw] flex items-center justify-center"
 | |
|     >
 | |
|       <input
 | |
|         v-if="submittable"
 | |
|         type="radio"
 | |
|         :value="false"
 | |
|         class="aspect-square base-radio"
 | |
|         :class="{ '!w-auto !h-full': area.w > area.h, '!w-full !h-auto': area.w <= area.h }"
 | |
|         :checked="!!modelValue && modelValue === field.options.find((o) => o.uuid === area.option_uuid)?.value"
 | |
|         @click="$emit('update:model-value', field.options.find((o) => o.uuid === area.option_uuid)?.value)"
 | |
|       >
 | |
|       <IconCheck
 | |
|         v-else-if="!!modelValue && modelValue === field.options.find((o) => o.uuid === area.option_uuid)?.value"
 | |
|         class="aspect-square"
 | |
|         :class="{ '!w-auto !h-full': area.w > area.h, '!w-full !h-auto': area.w <= area.h }"
 | |
|       />
 | |
|     </div>
 | |
|     <div
 | |
|       v-else-if="field.type === 'multiple' && area.option_uuid"
 | |
|       class="w-full p-[0.2vw] flex items-center justify-center"
 | |
|     >
 | |
|       <input
 | |
|         v-if="submittable"
 | |
|         type="checkbox"
 | |
|         :value="false"
 | |
|         class="aspect-square base-checkbox"
 | |
|         :class="{ '!w-auto !h-full': area.w > area.h, '!w-full !h-auto': area.w <= area.h }"
 | |
|         :checked="!!modelValue && modelValue.includes(field.options.find((o) => o.uuid === area.option_uuid)?.value)"
 | |
|         @change="updateMultipleSelectValue(field.options.find((o) => o.uuid === area.option_uuid)?.value)"
 | |
|       >
 | |
|       <IconCheck
 | |
|         v-else-if="!!modelValue && modelValue.includes(field.options.find((o) => o.uuid === area.option_uuid)?.value)"
 | |
|         class="aspect-square"
 | |
|         :class="{ '!w-auto !h-full': area.w > area.h, '!w-full !h-auto': area.w <= area.h }"
 | |
|       />
 | |
|     </div>
 | |
|     <div
 | |
|       v-else-if="field.type === 'cells'"
 | |
|       class="w-full flex items-center"
 | |
|     >
 | |
|       <div
 | |
|         v-for="(char, index) in modelValue"
 | |
|         :key="index"
 | |
|         class="text-center flex-none"
 | |
|         :style="{ width: (area.cell_w / area.w * 100) + '%' }"
 | |
|       >
 | |
|         {{ char }}
 | |
|       </div>
 | |
|     </div>
 | |
|     <div
 | |
|       v-else
 | |
|       ref="textContainer"
 | |
|       class="flex items-center px-0.5"
 | |
|     >
 | |
|       <span v-if="Array.isArray(modelValue)">
 | |
|         {{ modelValue.join(', ') }}
 | |
|       </span>
 | |
|       <span v-else-if="field.type === 'date'">
 | |
|         {{ formattedDate }}
 | |
|       </span>
 | |
|       <span
 | |
|         v-else
 | |
|         class="whitespace-pre-wrap"
 | |
|       >{{ modelValue }}</span>
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| import { IconTextSize, IconWritingSign, IconCalendarEvent, IconPhoto, IconCheckbox, IconPaperclip, IconSelect, IconCircleDot, IconChecks, IconCheck, IconColumns3, IconPhoneCheck, IconLetterCaseUpper } from '@tabler/icons-vue'
 | |
| 
 | |
| export default {
 | |
|   name: 'FieldArea',
 | |
|   components: {
 | |
|     IconPaperclip,
 | |
|     IconCheck
 | |
|   },
 | |
|   props: {
 | |
|     field: {
 | |
|       type: Object,
 | |
|       required: true
 | |
|     },
 | |
|     isValueSet: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     submittable: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     modelValue: {
 | |
|       type: [Array, String, Number, Object, Boolean],
 | |
|       required: false,
 | |
|       default: ''
 | |
|     },
 | |
|     isActive: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: false
 | |
|     },
 | |
|     withLabel: {
 | |
|       type: Boolean,
 | |
|       required: false,
 | |
|       default: true
 | |
|     },
 | |
|     fieldIndex: {
 | |
|       type: Number,
 | |
|       required: false,
 | |
|       default: 0
 | |
|     },
 | |
|     attachmentsIndex: {
 | |
|       type: Object,
 | |
|       required: false,
 | |
|       default: () => ({})
 | |
|     },
 | |
|     area: {
 | |
|       type: Object,
 | |
|       required: true
 | |
|     }
 | |
|   },
 | |
|   emits: ['update:model-value'],
 | |
|   data () {
 | |
|     return {
 | |
|       textOverflowChars: 0
 | |
|     }
 | |
|   },
 | |
|   computed: {
 | |
|     fieldNames () {
 | |
|       return {
 | |
|         text: 'Text',
 | |
|         cells: 'Text',
 | |
|         signature: 'Signature',
 | |
|         date: 'Date',
 | |
|         image: 'Image',
 | |
|         initials: 'Initials',
 | |
|         file: 'File',
 | |
|         select: 'Select',
 | |
|         checkbox: 'Checkbox',
 | |
|         radio: 'Radio',
 | |
|         multiple: 'Multiple Select',
 | |
|         phone: 'Phone'
 | |
|       }
 | |
|     },
 | |
|     fieldIcons () {
 | |
|       return {
 | |
|         text: IconTextSize,
 | |
|         signature: IconWritingSign,
 | |
|         date: IconCalendarEvent,
 | |
|         image: IconPhoto,
 | |
|         initials: IconLetterCaseUpper,
 | |
|         file: IconPaperclip,
 | |
|         select: IconSelect,
 | |
|         checkbox: IconCheckbox,
 | |
|         radio: IconCircleDot,
 | |
|         cells: IconColumns3,
 | |
|         multiple: IconChecks,
 | |
|         phone: IconPhoneCheck
 | |
|       }
 | |
|     },
 | |
|     image () {
 | |
|       if (this.field.type === 'image') {
 | |
|         return this.attachmentsIndex[this.modelValue]
 | |
|       } else {
 | |
|         return null
 | |
|       }
 | |
|     },
 | |
|     signature () {
 | |
|       if (this.field.type === 'signature') {
 | |
|         return this.attachmentsIndex[this.modelValue]
 | |
|       } else {
 | |
|         return null
 | |
|       }
 | |
|     },
 | |
|     initials () {
 | |
|       if (this.field.type === 'initials') {
 | |
|         return this.attachmentsIndex[this.modelValue]
 | |
|       } else {
 | |
|         return null
 | |
|       }
 | |
|     },
 | |
|     formattedDate () {
 | |
|       if (this.field.type === 'date' && this.modelValue) {
 | |
|         return new Intl.DateTimeFormat([], { year: 'numeric', month: 'long', day: 'numeric', timeZone: 'UTC' }).format(new Date(this.modelValue))
 | |
|       } else {
 | |
|         return ''
 | |
|       }
 | |
|     },
 | |
|     attachments () {
 | |
|       if (this.field.type === 'file') {
 | |
|         return (this.modelValue || []).map((uuid) => this.attachmentsIndex[uuid])
 | |
|       } else {
 | |
|         return []
 | |
|       }
 | |
|     },
 | |
|     computedStyle () {
 | |
|       const { x, y, w, h } = this.area
 | |
| 
 | |
|       return {
 | |
|         top: y * 100 + '%',
 | |
|         left: x * 100 + '%',
 | |
|         width: w * 100 + '%',
 | |
|         height: h * 100 + '%'
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   watch: {
 | |
|     modelValue () {
 | |
|       if (this.field.type === 'text' && this.$refs.textContainer && (this.textOverflowChars === 0 || (this.textOverflowChars - 4) > this.modelValue.length)) {
 | |
|         this.textOverflowChars = this.$refs.textContainer.scrollHeight > this.$refs.textContainer.clientHeight ? this.modelValue.length : 0
 | |
|       }
 | |
|     }
 | |
|   },
 | |
|   mounted () {
 | |
|     if (this.field.type === 'text' && this.$refs.textContainer) {
 | |
|       this.$nextTick(() => {
 | |
|         this.textOverflowChars = this.$refs.textContainer.scrollHeight > this.$refs.textContainer.clientHeight ? this.modelValue.length : 0
 | |
|       })
 | |
|     }
 | |
|   },
 | |
|   methods: {
 | |
|     updateMultipleSelectValue (value) {
 | |
|       if (this.modelValue?.includes(value)) {
 | |
|         const newValue = [...this.modelValue]
 | |
| 
 | |
|         newValue.splice(newValue.indexOf(value), 1)
 | |
| 
 | |
|         this.$emit('update:model-value', newValue)
 | |
|       } else {
 | |
|         const newValue = this.modelValue ? [...this.modelValue] : []
 | |
| 
 | |
|         newValue.push(value)
 | |
| 
 | |
|         this.$emit('update:model-value', newValue)
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| </script>
 |