Files
formipay-public/vendor/wpcfto/metaboxes/general_components/js/repeater.js

393 lines
27 KiB
JavaScript

(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
"use strict";
const Repeater = (window.Swal && typeof window.Swal.mixin === 'function')
? window.Swal.mixin({
customClass: {
container: 'wpcfto-settings',
confirmButton: 'button'
}
})
: {
// Fallback that mimics Swal.fire returning a Promise
fire: (opts) => Promise.resolve({
isConfirmed: window.confirm((opts && (opts.text || opts.html || opts.title)) || 'Are you sure?')
})
};
Vue.component('wpcfto_repeater', {
mixins: (window.validationMixin ? [window.validationMixin] : []),
props: ['fields', 'field_label', 'field_name', 'field_id', 'field_value'],
data: function data() {
return {
repeater: [],
repeater_values: {}
};
},
template: `
<div class="wpcfto_generic_field wpcfto_generic_field_repeater wpcfto-repeater unflex_fields">
<wpcfto_fields_aside_before
:fields="fields"
:field_label="field_label"
:required="fields && fields.required === true"
/>
<div class="wpcfto-field-content">
<!-- Required proxy: forces validation when repeater is empty -->
<input
class="wpcfto-required-proxy"
type="text"
:name="field_name + '__required__'"
:value="(repeater && repeater.length ? 'ok' : '')"
:required="fields && fields.required === true && (!repeater || repeater.length === 0)"
:disabled="!(fields && fields.required === true && (!repeater || repeater.length === 0))"
tabindex="-1" readonly
style="position:absolute; left:-9999px; top:auto; width:1px; height:1px; opacity:0; pointer-events:none;"
/>
<div v-for="(area, area_key) in repeater" class="wpcfto-repeater-single" :class="'wpcfto-repeater_' + field_name + '_' + area_key">
<div class="wpcfto_group_title" @click="toggleArea(area)" v-html="getGroupTitle(area, area_key)"></div>
<div class="repeater_inner" :class="{ closed: area.closed_tab }">
<div class="wpcfto-repeater-field" v-for="(field, field_name_inner) in fields.fields">
<component
:is="'wpcfto_' + field.type"
:fields="field"
:field_id="field.field_id + '_' + area_key + '_' + field_name_inner"
:field_name="field_name + '_' + area_key + '_' + field_name_inner"
:field_label="field.label"
:field_value="getFieldValue(area_key, field, field_name_inner)"
:field_data="field"
:field_native_name="field_name"
:field_native_name_inner="field_name_inner"
@wpcfto-get-value="$set(repeater[area_key], field_name_inner, $event)"
:ref="'field_' + area_key + '_' + field_name_inner"
></component>
</div>
</div>
<span class="wpcfto-repeater-single-delete" @click="removeArea(area_key)">
<i class="fa fa-trash-alt"></i>Delete
</span>
</div>
<div v-if="repeater && repeater.length > 0" class="separator"></div>
<div class="addArea" @click="validateAndAddArea">
<i class="fa fa-plus-circle"></i>
<span v-html="'Add ' + field_label"></span>
</div>
</div>
<wpcfto_fields_aside_after :fields="fields"></wpcfto_fields_aside_after>
</div>
`,
mounted: function mounted() {
var _this = this;
if (typeof _this.field_value === 'string' && WpcftoIsJsonString(_this.field_value)) {
_this.field_value = JSON.parse(_this.field_value);
}
if (typeof _this.field_value !== 'undefined' && typeof _this.field_value !== 'string') {
_this.$set(_this, 'repeater_values', _this.field_value);
_this.repeater_values.forEach(function(item, index) {
var repeaterItem = { closed_tab: true };
for (var field_name in item) {
repeaterItem[field_name] = item[field_name];
}
_this.repeater.push(repeaterItem);
});
// *** Ensure ALL rows are closed after initialization ***
_this.repeater.forEach(function(item) {
item.closed_tab = true;
});
}
// Initial validation state
if (typeof this.validateField === 'function') this.validateField();
// Block Save when repeater is required but empty (length check only)
const handler = (e) => {
// Only enforce when this component is in DOM
if (!this.$el || !document.body.contains(this.$el)) return;
const emptyRequired = !!(this.fields && this.fields.required === true && (!this.repeater || this.repeater.length === 0));
if (emptyRequired) {
e.preventDefault();
e.stopPropagation();
this.focusSelf();
}
};
try {
const btns = document.querySelectorAll('.wpcfto_save_settings, .wpcfto_save_metabox');
btns.forEach(btn => btn.addEventListener('click', handler, true));
this.__wpcftoSaveHandler = handler;
} catch(e) {}
// Also prevent <form> submission if invalid (works for both settings + metabox)
try {
const form = this.$el.closest('form');
if (form) {
const onSubmit = (e) => {
const emptyRequired = !!(this.fields && this.fields.required === true && (!this.repeater || this.repeater.length === 0));
if (emptyRequired) {
e.preventDefault();
e.stopPropagation();
this.focusSelf();
}
};
form.addEventListener('submit', onSubmit, true);
this.__wpcftoFormSubmitHandler = onSubmit;
}
} catch(e) {}
if (typeof this.validateField === 'function') this.validateField();
},
beforeDestroy: function () {
if (this.__teardown) this.__teardown();
},
methods: {
focusSelf() {
try {
const rect = this.$el.getBoundingClientRect();
const top = window.pageYOffset + rect.top - 60;
window.scrollTo({ top, behavior: 'smooth' });
} catch(e) {}
try { this.$el.classList.add('wpcfto-invalid'); } catch(e) {}
},
// scrollIntoViewIfNeeded() {
// try {
// const top = window.pageYOffset + this.$el.getBoundingClientRect().top - 60;
// window.scrollTo({ top, behavior: 'smooth' });
// } catch(e) {}
// },
formatLabelPath(parts) {
return parts.filter(Boolean).map(function (s) {
return String(s).trim();
}).filter(Boolean).join(' → ');
},
validateField() {
let isValid = true;
if ('required' in this.fields && this.fields.required === true) {
isValid = !!(this.fields.value.length > 0);
}
if (!isValid) {
try { this.$el.classList.add('wpcfto-invalid'); } catch(e){}
try { this.$el.setAttribute('aria-invalid','true'); } catch(e){}
// Tell the global collector (validationMixin) about our state
try {
const fid = this.field_id || (this.fields && this.fields.field_id);
if (fid) this.$root.$emit('field-validation', { fieldId: fid, isValid: false });
} catch(e) {}
return false;
}
try { this.$el.classList.remove('wpcfto-invalid'); } catch(e){}
try { this.$el.removeAttribute('aria-invalid'); } catch(e){}
try {
const fid = this.field_id || (this.fields && this.fields.field_id);
if (fid) this.$root.$emit('field-validation', { fieldId: fid, isValid: true });
} catch(e) {}
return true;
},
// This method validates all visible required fields in all rows before adding a new row
validateAndAddArea: function() {
const fieldsInRepeater = this._props.fields.fields;
let group_title = null;
Object.values(fieldsInRepeater).forEach(field => {
if ('is_group_title' in field) {
if (field.is_group_title !== false) {
group_title = field.label;
}
}
});
var invalidFields = [];
// Loop through all repeater rows and all fields
this.repeater.forEach((area, area_key) => {
for (const field_name_inner in this.fields.fields) {
const field = this.fields.fields[field_name_inner];
// Find the child component via $refs
const refName = 'field_' + area_key + '_' + field_name_inner;
const child = this.$refs[refName];
// $refs[refName] can be an array if used inside v-for, so get the first element
const childComponent = Array.isArray(child) ? child[0] : child;
// If the child component exists and has validateField method, call it
if (childComponent && typeof childComponent.validateField === 'function') {
// Only validate visible required fields
if (childComponent.isVisible && childComponent.fields && childComponent.fields.required) {
if (!childComponent.validateField()) {
invalidFields.push({
repeater_label: this._props.field_label,
group_title: group_title,
label: childComponent.fields.label || field_name_inner,
area: area_key + 1
});
// Optionally, you can add a visual error class here
// childComponent.$el.classList.add('invalid-field');
}
}
}
}
});
if (invalidFields.length > 0) {
// Compose error message
var msg = 'Please fill all required fields before adding a new item:\n\n';
const firstInvalidItem = invalidFields[0];
msg += 'Check item #'+firstInvalidItem.area+': <b>'+this.formatLabelPath([firstInvalidItem.group_title, firstInvalidItem.label])+'</b>';
// Use SweetAlert2 if available, otherwise alert()
if (typeof window.Swal === 'function') {
Repeater.fire({
icon: 'warning',
title: 'Required fields missing',
html: msg.replace(/\n/g, '<br>'),
width: 'fit-content',
customClass: {
confirmButton: 'btn text-bg-primary'
},
});
} else {
alert(msg);
}
return;
}
// If all required fields are valid, add a new row
this.addArea();
},
addArea: function addArea() {
// Close all rows before adding new
this.repeater.forEach(item => { item.closed_tab = true; });
this.repeater.push({ closed_tab: false }); // new row is open
var el = 'wpcfto-repeater_' + this.field_name + '_' + (this.repeater.length - 1);
Vue.nextTick(function () {
if (typeof jQuery !== 'undefined') {
var $ = jQuery;
$([document.documentElement, document.body]).animate({
scrollTop: $("." + el).offset().top - 40
}, 400);
}
});
this.validateField();
if (typeof jQuery !== 'undefined') {
jQuery(document).trigger('repeater-item-added', [this._props]);
}
},
toggleArea: function toggleArea(area) {
// Close all other rows
this.repeater.forEach(item => {
if (item !== area) {
this.$set(item, 'closed_tab', true);
}
});
// Toggle clicked row's closed_tab state
const isCurrentlyClosed = area.closed_tab === true || typeof area.closed_tab === 'undefined';
this.$set(area, 'closed_tab', !isCurrentlyClosed);
},
removeArea: function removeArea(areaIndex) {
// if (confirm('Do your really want to delete this field?')) {
// this.repeater.splice(areaIndex, 1);
// }
Repeater.fire({
icon: 'question',
html: 'Do your really want to delete this item?',
showCancelButton: true,
reverseButtons: true
}).then((result) => {
if (result.isConfirmed) {
this.repeater.splice(areaIndex, 1);
// jQuery action hook trigger example:
if (typeof jQuery !== 'undefined') {
jQuery(document).trigger('repeater-item-removed', [this._props, areaIndex]);
}
}
});
},
getFieldValue: function getFieldValue(key, field, field_name) {
if (this.repeater[key] && typeof this.repeater[key][field_name] !== 'undefined') {
return this.repeater[key][field_name];
}
if (typeof this.repeater_values !== 'undefined' &&
typeof this.repeater_values[key] !== 'undefined' &&
typeof this.repeater_values[key][field_name] !== 'undefined') {
return this.repeater_values[key][field_name];
}
return field.value;
},
getGroupTitle(area, area_key) {
let groupTitleFieldName = null;
let groupTitleField = null;
for (const [fname, field] of Object.entries(this.fields.fields)) {
if (field.is_group_title) {
groupTitleFieldName = fname;
groupTitleField = field;
break;
}
}
if (groupTitleFieldName) {
let value = area[groupTitleFieldName];
if (typeof value === 'undefined' && this.repeater_values && this.repeater_values[area_key]) {
value = this.repeater_values[area_key][groupTitleFieldName];
}
if (typeof value === 'undefined' || value === '') {
value = groupTitleField.label || this.field_label;
} else {
if (groupTitleField.options && groupTitleField.options[value]) {
value = groupTitleField.options[value];
}
}
return value;
}
return this.field_label + ' #' + (area_key + 1);
},
validateAllRows() {
let invalidFields = [];
this.repeater.forEach((area, area_key) => {
for (const field_name_inner in this.fields.fields) {
const field = this.fields.fields[field_name_inner];
// Find child component by ref
const refName = 'field_' + area_key + '_' + field_name_inner;
const child = this.$refs[refName];
const childComponent = Array.isArray(child) ? child[0] : child;
if (childComponent && typeof childComponent.validateField === 'function') {
if (childComponent.isVisible && field.required) {
if (!childComponent.validateField()) {
invalidFields.push({
label: field.label || field_name_inner,
row: area_key + 1
});
// Optionally add visual error here
}
}
}
}
});
if (invalidFields.length > 0) {
let msg = 'Please fill all required fields before saving:\n\n';
msg += invalidFields.map(f => `Row #${f.row}: ${f.label}`).join('\n');
if (typeof window.Swal === 'function') {
Swal.fire({
icon: 'warning',
title: 'Required fields missing',
html: msg.replace(/\n/g, '<br>')
});
} else {
alert(msg);
}
return false;
}
return true;
},
saveChanges() {
// Block if repeater required but empty OR if any child invalid
const selfValid = this.validateField();
// const rowsValid = this.validateAllRows();
if (!selfValid) { return; }
// Proceed with save logic here...
}
},
watch: {
repeater: {
deep: true,
handler: function handler(repeater) {
this.$emit('wpcfto-get-value', repeater);
this.validateField();
}
}
}
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["fake_3a22d5a4.js"],"names":["Vue","component","props","data","repeater","repeater_values","template","mounted","_this","field_value","WpcftoIsJsonString","JSON","parse","$set","forEach","push","methods","addArea","closed_tab","el","field_name","length","nextTick","jQuery","$","document","documentElement","body","animate","scrollTop","offset","top","toggleArea","area","currentState","removeArea","areaIndex","confirm","splice","getFieldValue","key","field","value","watch","deep","handler","$emit"],"mappings":"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","sourcesContent":["\"use strict\";\n\nVue.component('wpcfto_repeater', {\n  props: ['fields', 'field_label', 'field_name', 'field_id', 'field_value'],\n  data: function data() {\n    return {\n      repeater: [],\n      repeater_values: {}\n    };\n  },\n  template: \"\\n    <div class=\\\"wpcfto_generic_field wpcfto_generic_field_repeater wpcfto-repeater unflex_fields\\\">\\n\\n        <wpcfto_fields_aside_before :fields=\\\"fields\\\" :field_label=\\\"field_label\\\"></wpcfto_fields_aside_before>\\n        \\n        <div class=\\\"wpcfto-field-content\\\">\\n\\n            <div v-for=\\\"(area, area_key) in repeater\\\" class=\\\"wpcfto-repeater-single\\\" :class=\\\"'wpcfto-repeater_' + field_name + '_' + area_key \\\">\\n    \\n                <div class=\\\"wpcfto_group_title\\\" v-html=\\\"field_label + ' #' + (area_key + 1)\\\"></div>\\n    \\n                <div class=\\\"repeater_inner\\\">\\n    \\n                    <div class=\\\"wpcfto-repeater-field\\\" v-for=\\\"(field, field_name_inner) in fields.fields\\\">\\n                    \\n                        <component :is=\\\"'wpcfto_' + field.type\\\"\\n                                   :fields=\\\"field\\\"\\n                                   :field_name=\\\"field_name + '_' + area_key + '_' + field_name_inner\\\"\\n                                   :field_label=\\\"field.label\\\"\\n                                   :field_value=\\\"getFieldValue(area_key, field, field_name_inner)\\\"\\n                                   :field_data=\\\"field\\\"\\n                                   :field_native_name=\\\"field_name\\\"\\n                                   :field_native_name_inner=\\\"field_name_inner\\\"\\n                                   @wpcfto-get-value=\\\"$set(repeater[area_key], field_name_inner, $event)\\\">\\n                        </component>\\n    \\n                    </div>\\n    \\n                </div>\\n    \\n                <span class=\\\"wpcfto-repeater-single-delete\\\" @click=\\\"removeArea(area_key)\\\">\\n                    <i class=\\\"fa fa-trash-alt\\\"></i>Delete\\n                </span>\\n    \\n            </div>\\n    \\n            <div v-if=\\\"repeater && repeater.length > 0\\\" class=\\\"separator\\\"></div>\\n    \\n            <div class=\\\"addArea\\\" @click=\\\"addArea\\\">\\n                <i class=\\\"fa fa-plus-circle\\\"></i>\\n                <span v-html=\\\"'Add ' + field_label\\\"></span>\\n            </div>\\n        \\n        </div>\\n        \\n        <wpcfto_fields_aside_after :fields=\\\"fields\\\"></wpcfto_fields_aside_after>\\n\\n    </div>\\n    \",\n  mounted: function mounted() {\n    var _this = this;\n\n    if (typeof _this.field_value === 'string' && WpcftoIsJsonString(_this.field_value)) {\n      _this.field_value = JSON.parse(_this.field_value);\n    }\n\n    if (typeof _this.field_value !== 'undefined' && typeof _this.field_value !== 'string') {\n      _this.$set(_this, 'repeater_values', _this.field_value);\n\n      _this.repeater_values.forEach(function () {\n        _this.repeater.push({});\n      });\n    }\n  },\n  methods: {\n    addArea: function addArea() {\n      this.repeater.push({\n        closed_tab: true\n      });\n      var el = 'wpcfto-repeater_' + this.field_name + '_' + (this.repeater.length - 1);\n      Vue.nextTick(function () {\n        if (typeof jQuery !== 'undefined') {\n          var $ = jQuery;\n          $([document.documentElement, document.body]).animate({\n            scrollTop: $(\".\" + el).offset().top - 40\n          }, 400);\n        }\n      });\n    },\n    toggleArea: function toggleArea(area) {\n      var currentState = typeof area['closed_tab'] !== 'undefined' ? area['closed_tab'] : false;\n      this.$set(area, 'closed_tab', !currentState);\n    },\n    removeArea: function removeArea(areaIndex) {\n      if (confirm('Do your really want to delete this field?')) {\n        this.repeater.splice(areaIndex, 1);\n      }\n    },\n    getFieldValue: function getFieldValue(key, field, field_name) {\n      if (typeof this.repeater_values === 'undefined') return field.value;\n      if (typeof this.repeater_values[key] === 'undefined') return field.value;\n      if (typeof this.repeater_values[key][field_name] === 'undefined') return field.value;\n      return this.repeater_values[key][field_name];\n    }\n  },\n  watch: {\n    repeater: {\n      deep: true,\n      handler: function handler(repeater) {\n        this.$emit('wpcfto-get-value', repeater);\n      }\n    }\n  }\n});"]}
},{}]},{},[1])