384 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			384 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| @once
 | |
|     @push('css')
 | |
|         <style>
 | |
|             :root {
 | |
|                 --l2fd-background-color: rgb(246, 250, 255);
 | |
|                 --l2fd-border-color:     #d2d6de;
 | |
|                 --l2fd-font-color:       #555555;
 | |
|                 
 | |
|                 --list-padding:          2px;
 | |
| 
 | |
|                 --listitem-font-color:       #555555;
 | |
|                 --listitem-padding:          8px;
 | |
|                 --listitem-spacing:          2px;
 | |
|                 --listitem-background-color: white;
 | |
|                 --listitem-border-color:     #ccc;
 | |
|                 --listitem-border-radius:    2px;
 | |
|                 
 | |
|                 --listitem-hover-font-color:       var(--listitem-font-color);
 | |
|                 --listitem-hover-background-color: var(--listitem-background-color);
 | |
|                 --listitem-hover-border-color:     rgb(102, 175, 233);
 | |
|                 
 | |
|                 --listitem-selected-font-color:       var(--listitem-font-color);
 | |
|                 --listitem-selected-background-color: rgb(236, 240, 245);
 | |
|                 --listitem-selected-border-color:     var(--listitem-hover-border-color);
 | |
| 
 | |
|                 --buttonbar-button-font-color:       #555555;
 | |
|                 --buttonbar-button-background-color: white;
 | |
|                 --buttonbar-button-border-color:     #ccc;
 | |
|                 --buttonbar-button-border-radius:    2px;
 | |
|                 
 | |
|                 --buttonbar-button-hover-font-color:       var(--buttonbar-button-font-color);
 | |
|                 --buttonbar-button-hover-background-color: var(--buttonbar-button-background-color);
 | |
|                 --buttonbar-button-hover-border-color:     var(--listitem-hover-border-color);
 | |
|                 
 | |
|                 --buttonbar-button-disabled-font-color:       var(--buttonbar-button-font-color);
 | |
|                 --buttonbar-button-disabled-background-color: #eee;
 | |
|                 --buttonbar-button-disabled-border-color:     var(--buttonbar-button-border-color);
 | |
|             }
 | |
| 
 | |
|             .l2fd-root,
 | |
|             .l2fd-root * {
 | |
|                 box-sizing: border-box;
 | |
|             }
 | |
| 
 | |
|             .l2fd-root {
 | |
|                 height: 400px;
 | |
|                 display: flex;
 | |
|                 flex-direction: column;
 | |
|             }
 | |
| 
 | |
|             .l2fd-title {
 | |
|                 font-size: 1.4em;
 | |
|                 padding: 6px;
 | |
|                 margin: 0;
 | |
|             }
 | |
|             
 | |
|             .l2fd-list {
 | |
|                 overflow-y: scroll;
 | |
|                 padding:    var(--list-padding);
 | |
|             }
 | |
| 
 | |
|             .l2fd-main {
 | |
|                 flex: 1;
 | |
|                 display: grid;
 | |
|                 grid-template-areas:
 | |
|                     'fields-title   options-title'
 | |
|                     'fields-list    options-list'
 | |
|                     'fields-buttons options-buttons';
 | |
|                 grid-template-columns: 50% 50%;
 | |
|                 grid-template-rows: max-content auto max-content;
 | |
| 
 | |
|                 background-color: var(--l2fd-background-color);
 | |
|                 border:           1px solid var(--l2fd-border-color);
 | |
|                 color:            var(--l2fd-font-color);
 | |
|             }
 | |
| 
 | |
|             .l2fd-listitem {
 | |
|                 color: var(--listitem-font-color);
 | |
|                 cursor: pointer;
 | |
|                 padding: var(--listitem-padding);
 | |
|                 margin-bottom: var(--listitem-spacing);
 | |
|                 background-color: var(--listitem-background-color);
 | |
|                 border: 1px solid var(--listitem-border-color);
 | |
|                 border-radius: var(--listitem-border-radius);
 | |
|             }
 | |
|             .l2fd-listitem:hover {
 | |
|                 color: var(--listitem-hover-font-color);
 | |
|                 background-color: var(--listitem-hover-background-color);
 | |
|                 border: 1px solid var(--listitem-hover-border-color);
 | |
|             }
 | |
|             .l2fd-listitem.selected {
 | |
|                 color: var(--listitem-selected-font-color);
 | |
|                 background-color: var(--listitem-selected-background-color);
 | |
|                 border: 1px solid var(--listitem-selected-border-color);
 | |
|             }
 | |
| 
 | |
|             .l2fd-itemgrid {
 | |
|                 display: grid;
 | |
|                 grid-template-areas:
 | |
|                     'label-title source-title'
 | |
|                     'label-field source-field';
 | |
|                 grid-template-columns: 50% 50%;
 | |
|                 grid-template-rows: auto auto;
 | |
|             }
 | |
| 
 | |
|             .l2fd-listitem label {
 | |
|                 cursor: pointer;
 | |
|                 font-size: 0.9em;
 | |
|                 padding: 0;
 | |
|                 margin: 0;
 | |
|             }
 | |
| 
 | |
|             .l2fd-buttonbar {
 | |
|                 display: flex;
 | |
|                 flex-direction: row;
 | |
|                 height: 35px;
 | |
|             }
 | |
|             .l2fd-buttonbar > button {
 | |
|                 flex: 1 1 100%;
 | |
|                 background-color: var(--buttonbar-button-background-color);
 | |
|                 border:           1px solid var(--buttonbar-button-border-color);
 | |
|                 border-radius:    var(--buttonbar-button-border-radius);
 | |
|                 color:            var(--buttonbar-button-font-color);
 | |
|             }
 | |
|             .l2fd-buttonbar > button:hover {
 | |
|                 background-color: var(--buttonbar-button-hover-background-color);
 | |
|                 border:           1px solid var(--buttonbar-button-hover-border-color);
 | |
|                 color:            var(--buttonbar-button-hover-font-color);
 | |
|             }
 | |
|             .l2fd-buttonbar > button.disabled {
 | |
|                 background-color: var(--buttonbar-button-disabled-background-color);
 | |
|                 border:           1px solid var(--buttonbar-button-disabled-border-color);
 | |
|                 color:            var(--buttonbar-button-disabled-font-color);
 | |
|                 cursor:           not-allowed;
 | |
|             }
 | |
|             
 | |
|         </style>
 | |
|     @endpush
 | |
| @endonce
 | |
| 
 | |
| @push('js')
 | |
|     <script>
 | |
|         document.addEventListener('alpine:init', () => {
 | |
| 
 | |
|             Alpine.data('{{ $name }}', () => ({
 | |
| 
 | |
|                 _name: '{{ $name }}',
 | |
|                 _defaultValue: '{{ $value }}',
 | |
|                   _init: function() {
 | |
|                     this.fields = this.fromString(this._defaultValue);
 | |
|                     this.$watch('valueString', () => {
 | |
|                         this.$refs.input.form.dispatchEvent(new Event('change'));
 | |
|                     });
 | |
|                 },
 | |
| 
 | |
|                 /* Fields */
 | |
|                 fields: [],
 | |
|                 get _templateField() { return ({ options: [ this._templateOption ] }); },
 | |
|                 _selectedField: null,
 | |
|                 get selectedField() { return this._selectedField; },
 | |
|                 set selectedField(field) {
 | |
|                     this._selectedField = field;
 | |
|                     this.selectedOption = null;
 | |
|                 },
 | |
|                 get selectedFieldIndex() {
 | |
|                     return this.selectedField ? this.fields.indexOf(this.selectedField) : -1;
 | |
|                 },
 | |
|                 shiftSelectedField: function(offset) {
 | |
|                     this.shiftArrayValue(this.fields, this.selectedField, offset);
 | |
|                 },
 | |
|                 trashSelectedField: function() {
 | |
|                     this.fields.splice(this.fields.indexOf(this.selectedField), 1);
 | |
|                     this.selectedField = null;
 | |
|                 },
 | |
|                 addField: function() {
 | |
|                     let newField = JSON.parse(JSON.stringify(this._templateField));
 | |
|                     this.fields.push(newField);
 | |
|                     this.selectedField = newField;
 | |
|                 },
 | |
| 
 | |
|                 /* Options */
 | |
|                 get _templateOption() { return ({ label: '', datasource: '' }); },
 | |
|                 selectedOption: null,
 | |
|                 get selectedOptionIndex() {
 | |
|                     return this.selectedOption ? this.selectedField.options.indexOf(this.selectedOption) : -1;
 | |
|                 },
 | |
|                 shiftSelectedOption: function(offset) {
 | |
|                     this.shiftArrayValue(this.selectedField.options, this.selectedOption, offset);
 | |
|                 },
 | |
|                 trashSelectedOption: function() {
 | |
|                     this.selectedField.options.splice(this.selectedField.options.indexOf(this.selectedOption), 1);
 | |
|                     this.selectedOption = null;
 | |
|                 },
 | |
|                 addOption: function() {
 | |
|                     let newOption = JSON.parse(JSON.stringify(this._templateOption));
 | |
|                     this.selectedField.options.push(newOption);
 | |
|                     this.selectedOption = newOption;
 | |
|                 },
 | |
| 
 | |
| 
 | |
|                 /* Helpers */
 | |
| 
 | |
|                 shiftArrayValue: function(array, value, offset) {
 | |
|                     let oldIndex = array.indexOf(value);
 | |
|                     let newIndex = oldIndex + offset;
 | |
|                     newIndex = Math.max(newIndex, 0);
 | |
|                     newIndex = Math.min(newIndex, array.length);
 | |
| 
 | |
|                     array.splice(newIndex, 0, array.splice(oldIndex, 1)[0]);
 | |
|                 },
 | |
| 
 | |
|                 get valueString() { return this.toString(this.fields); },
 | |
|                 onTest: function(a) {
 | |
|                     console.log('test', a);
 | |
|                 },
 | |
| 
 | |
|                 getFieldLabel: function(field) {
 | |
|                     return field.options.map(option => option.label).join(' | ');
 | |
|                 },
 | |
|                 fromString: function(string) {
 | |
|                     return string
 | |
|                         .split(';').filter(fieldString => fieldString !== '')
 | |
|                         .map(fieldString => ({
 | |
|                             options: fieldString
 | |
|                                 .split('|').filter(optionString => optionString !== '')
 | |
|                                 .map(optionString => {
 | |
|                                     let [l,d] = optionString.split('=');
 | |
|                                     return { label: l, datasource: d };
 | |
|                                 })
 | |
|                         }));
 | |
|                 },
 | |
|                 toString: function(fields) {
 | |
|                     return fields
 | |
|                         .map(field => field.options
 | |
|                             .map(option => option.label + '=' + option.datasource)
 | |
|                             .join('|')
 | |
|                         )
 | |
|                         .join(';');
 | |
|                 },
 | |
| 
 | |
|             }));
 | |
| 
 | |
|         });
 | |
|     </script>
 | |
| @endpush
 | |
| @php
 | |
|     $selector = '[x-data="'.$name.'"]';
 | |
| @endphp
 | |
| @push('css')
 | |
|     <style>
 | |
| 
 | |
| 
 | |
|     </style>
 | |
| @endpush
 | |
| 
 | |
| <div x-data="{{ $name }}" x-init="_init" class="l2fd-root">
 | |
|     <input type="hidden" name="{{ $name }}" x-model="valueString" x-ref="input" />
 | |
|     <div class="l2fd-main">
 | |
|         <h1 class="l2fd-title" style="grid-area: fields-title">Fields</h1>
 | |
|         <div class="l2fd-list" style="grid-area: fields-list">
 | |
|             <template x-for="(field, index) in fields">
 | |
|                 <div 
 | |
|                     x-bind:key="'field-' + index" 
 | |
|                     x-bind:class="{
 | |
|                         'l2fd-listitem': true,
 | |
|                         'selected': selectedField === field
 | |
|                     }"
 | |
|                     x-on:click="selectedField = field" >
 | |
|                     <label><span x-text="index+1"></span>: <span x-text="getFieldLabel(field)"></span></label>
 | |
|                 </div>
 | |
|             </template>
 | |
|         </div>
 | |
|         <div class="l2fd-buttonbar" style="grid-area: fields-buttons">
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) shiftSelectedField(-1)"
 | |
|                 x-bind:class="{ 'disabled': !selectedField || selectedFieldIndex == 0 }"
 | |
|                 ><i class="fa-solid fa-caret-up"></i></button>
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) shiftSelectedField(+1)"
 | |
|                 x-bind:class="{ 'disabled': !selectedField || selectedFieldIndex == fields.length - 1 }"
 | |
|                 ><i class="fa-solid fa-caret-down"></i></button>
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) addField()"
 | |
|                 x-bind:class="{}"
 | |
|                 ><i class="fa-solid fa-plus"></i></button>
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) trashSelectedField()"
 | |
|                 x-bind:class="{ 'disabled': !selectedField }"
 | |
|                 ><i class="fa-solid fa-trash"></i></button>
 | |
|         </div>
 | |
| 
 | |
|         <h1 class="l2fd-title" style="grid-area: options-title">Options</h1>
 | |
|         <div class="l2fd-list" style="grid-area: options-list">
 | |
|             <template x-if="selectedField">
 | |
|                 <template x-for="(option, index) in selectedField.options">
 | |
|                     <div 
 | |
|                         x-bind:key="'option-' + index"
 | |
|                         x-bind:class="{
 | |
|                             'l2fd-listitem': true,
 | |
|                             'l2fd-itemgrid': true,
 | |
|                             'selected': selectedOption == option 
 | |
|                         }"
 | |
|                         x-on:click="selectedOption = option" >
 | |
|                         <label style="grid-area: label-title">Label</label>
 | |
|                         <input style="grid-area: label-field" x-model="option.label" />
 | |
|                         <label style="grid-area: source-title">DataSource</label>
 | |
|                         <select style="grid-area: source-field" x-model="option.datasource">
 | |
|                             <optgroup label="Asset">
 | |
|                                 <option value="asset_tag">{{trans('admin/hardware/table.asset_tag')}}</option>
 | |
|                                 <option value="name">{{trans('admin/hardware/form.name')}}</option>
 | |
|                                 <option value="serial">{{trans('admin/hardware/table.serial')}}</option>
 | |
|                                 <option value="asset_eol_date">{{trans('admin/hardware/form.eol_date')}}</option>
 | |
|                                 <option value="order_number">{{trans('admin/hardware/form.order')}}</option>
 | |
|                                 <option value="purchase_date">{{trans('admin/hardware/table.purchase_date')}}</option>
 | |
|                                 <option value="assignedTo">{{trans('admin/hardware/table.assigned_to')}}</option>
 | |
|                                 <option value="last_audit_date">{{trans('general.last_audit')}}</option>
 | |
|                                 <option value="next_audit_date">{{trans('general.next_audit_date')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Asset Model">
 | |
|                                 <option value="model.name">{{trans('admin/models/table.name')}}</option>
 | |
|                                 <option value="model.model_number">{{trans('admin/models/table.modelnumber')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Manufacturer">
 | |
|                                 <option value="model.manufacturer.name">{{trans('admin/hardware/form.manufacturer')}}</option>
 | |
|                                 <option value="model.manufacturer.support_email">{{trans('admin/manufacturers/table.support_email')}}</option>
 | |
|                                 <option value="model.manufacturer.support_phone">{{trans('admin/manufacturers/table.support_phone')}}</option>
 | |
|                                 <option value="model.manufacturer.support_url">{{trans('general.url')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Category">
 | |
|                                 <option value="model.category.name">{{trans('admin/categories/general.category_name')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Status">
 | |
|                                 <option value="assetstatus.name">{{trans('admin/statuslabels/table.name')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Supplier">
 | |
|                                 <option value="supplier.name">{{trans('admin/suppliers/table.name')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Default Location">
 | |
|                                 <option value="defaultLoc.name">{{trans('admin/hardware/form.default_location')}}</option>
 | |
|                                 <option value="defaultLoc.phone">{{trans('admin/hardware/form.default_location_phone')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Location">
 | |
|                                 <option value="location.name">{{trans('admin/locations/table.name')}}</option>
 | |
|                                 <option value="location.phone">{{trans('admin/locations/table.phone')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Company">
 | |
|                                 <option value="company.email">{{trans('admin/companies/table.email')}}</option>
 | |
|                                 <option value="company.name">{{trans('admin/companies/table.name')}}</option>
 | |
|                                 <option value="company.phone">{{trans('admin/companies/table.phone')}}</option>
 | |
|                             </optgroup>
 | |
|                             <optgroup label="Custom Fields">
 | |
|                                 @foreach($customFields as $customField)
 | |
|                                     <option value="{{ $customField->db_column }}">{{ $customField->name }}</option>
 | |
|                                 @endforeach
 | |
|                             </optgroup>
 | |
|                         </select>
 | |
|                     </div>
 | |
|                 </template>
 | |
|             </template>
 | |
|             <template x-if="!selectedField">
 | |
|                 <div>Please select a field</div>
 | |
|             </template>
 | |
|         </div>
 | |
|         <div class="l2fd-buttonbar" style="grid-area: options-buttons">
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) shiftSelectedOption(-1)"
 | |
|                 x-bind:class="{ 'disabled': !selectedField || !selectedOption || selectedOptionIndex == 0 }"
 | |
|                 ><i class="fa-solid fa-caret-up"></i></button>
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) shiftSelectedOption(+1)"
 | |
|                 x-bind:class="{ 'disabled': !selectedField || !selectedOption || selectedOptionIndex == selectedField.options.length - 1 }"
 | |
|                 ><i class="fa-solid fa-caret-down"></i></button>
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) addOption()"
 | |
|                 x-bind:class="{ 'disabled': !selectedField }"
 | |
|                 ><i class="fa-solid fa-plus"></i></button>
 | |
|             <button 
 | |
|                 x-on:click.prevent="if(!$event.target.classList.contains('disabled')) trashSelectedOption()"
 | |
|                 x-bind:class="{ 'disabled': !selectedField || !selectedOption }"
 | |
|                 ><i class="fa-solid fa-trash"></i></button>
 | |
|         </div>
 | |
|     </div>
 | |
| </div>
 |