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

212 lines
21 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";
Vue.component('v-select', VueSelect.VueSelect);
Vue.component('wpcfto_autocomplete', {
props: ['fields', 'field_label', 'field_name', 'field_id', 'field_value', 'field_data', 'required'],
data() {
return {
ids: [],
items: [],
search: '',
options: [],
loading: true,
itemHovered: null,
value: '',
limit: 0
};
},
template: `
<div class="wpcfto_generic_field wpcfto_generic_field_autocomplete autocomplete">
<wpcfto_fields_aside_before
:fields="fields"
:field_label="field_label"
:required="fields.required === true"
></wpcfto_fields_aside_before>
<div class="wpcfto-field-content">
<div class="wpcfto-autocomplete-search" :class="{loading: loading}">
<div class="v-select-search" v-if="underLimit()">
<i class="fa fa-plus-circle"></i>
<v-select label="title"
v-model="search"
@input="setSelected($event)"
:options="options"
@search="onSearch($event)"
:placeholder="computedPlaceholder"
:clearable="false"
:searchable="true"
:closeOnSelect="true"
></v-select>
</div>
<ul class="wpcfto-autocomplete" :class="{limited: !underLimit()}">
<li v-for="(item, index) in items" :key="item.id || index" v-if="typeof item !== 'string'"
:class="{hovered: itemHovered === index}">
<div class="item-wrapper">
<img v-if="item.image" :src="item.image" class="item-image" />
<div class="item-data">
<span v-html="item.title" class="item-title"></span>
<span v-if="item.excerpt" v-html="item.excerpt" class="item-excerpt"></span>
</div>
</div>
<i class="fa fa-trash-alt"
@click="removeItem(index)"
@mouseover="itemHovered = index"
@mouseleave="itemHovered = null"></i>
</li>
</ul>
<input type="hidden"
:name="field_name"
:value="serializedValue"
:required="fields && fields.required === true"
:disabled="!(fields && fields.required === true) && (!value || (Array.isArray(value) && value.length === 0))"
/>
</div>
</div>
<wpcfto_fields_aside_after :fields="fields"></wpcfto_fields_aside_after>
</div>
`,
computed: {
serializedValue() {
const v = this.value;
if (Array.isArray(v)) return v.join(',');
return v || '';
},
computedPlaceholder() {
// Default placeholder template or fallback
const template = formipay_admin?.config?.autocomplete?.placeholder || 'Search {field_label}...';
return template.replace('{field_label}', this.field_label || '');
}
},
created() {
if (this.field_value) {
this.getPosts(
`${stm_wpcfto_ajaxurl}?action=wpcfto_search_posts&nonce=${stm_wpcfto_nonces['wpcfto_search_posts']}&posts_per_page=-1&orderby=post__in&ids=${this.field_value}&post_types=${this.fields.post_type.join(',')}`,
'items'
);
} else {
this.clearItems();
this.isLoading(false);
}
if (typeof this.field_data.limit !== 'undefined' && this.field_data.limit > 0) {
this.limit = this.field_data.limit;
} else {
this.limit = 5; // default limit
}
if (!this.field_value) {
this.value = [];
}
},
mounted() {
this.$nextTick(() => {
this.updatePlaceholders();
// Attach event listeners to restore placeholder on blur/focusout/mouseleave
const autocompleteSearch = this.$el.querySelector('.wpcfto-autocomplete-search');
if (autocompleteSearch) {
autocompleteSearch.addEventListener('mouseleave', this.restorePlaceholder);
autocompleteSearch.addEventListener('blur', this.restorePlaceholder, true);
autocompleteSearch.addEventListener('focusout', this.restorePlaceholder, true);
}
});
},
beforeDestroy() {
// Cleanup event listeners
const autocompleteSearch = this.$el.querySelector('.wpcfto-autocomplete-search');
if (autocompleteSearch) {
autocompleteSearch.removeEventListener('mouseleave', this.restorePlaceholder);
autocompleteSearch.removeEventListener('blur', this.restorePlaceholder, true);
autocompleteSearch.removeEventListener('focusout', this.restorePlaceholder, true);
}
},
methods: {
isLoading(isLoading) {
this.loading = isLoading;
},
setSelected(value) {
if (value) this.items.push(value);
this.options = [];
this.search = '';
},
clearItems() {
this.items = this.items.filter(el => el != null && el !== '');
},
underLimit() {
return this.items.length < this.limit;
},
onSearch(search) {
const exclude = this.ids.join(',');
const post_types = this.fields.post_type.join(',');
this.getPosts(
`${stm_wpcfto_ajaxurl}?action=wpcfto_search_posts&nonce=${stm_wpcfto_nonces['wpcfto_search_posts']}&exclude_ids=${exclude}&s=${encodeURIComponent(search)}&post_types=${post_types}`,
'options'
);
},
getPosts(url, variable) {
this.isLoading(true);
url += `&name=${this.field_name}`;
this.$http.get(url).then(response => {
this[variable] = response.body;
this.clearItems();
this.isLoading(false);
});
},
updateIds() {
this.ids = this.items.map(item => item.id);
this.value = this.ids;
this.$emit('wpcfto-get-value', this.ids);
},
removeItem(index) {
this.items.splice(index, 1);
},
updatePlaceholders() {
// Scoped to this component root element
const autocompleteFields = this.$el.closest('.wpcfto-box')?.querySelectorAll('.autocomplete') || [];
autocompleteFields.forEach(field => {
const labelEl = field.querySelector('.wpcfto-field-aside__label span');
if (!labelEl) return;
const label = labelEl.textContent || labelEl.innerText || '';
const placeholderTemplate = formipay_admin?.config?.autocomplete?.placeholder || 'Search {field_label}...';
const placeholder = placeholderTemplate.replace('{field_label}', label);
const searchInput = field.querySelector('input[type="text"]');
if (searchInput) {
searchInput.setAttribute('placeholder', placeholder);
if (searchInput.parentElement) {
searchInput.parentElement.setAttribute('data-input-placeholder', placeholder);
}
}
});
},
restorePlaceholder(event) {
const target = event.target;
if (!target) return;
const parent = target.parentElement;
if (!parent) return;
const placeholder = parent.getAttribute('data-input-placeholder');
if (placeholder) {
setTimeout(() => {
target.setAttribute('placeholder', placeholder);
}, 500);
}
}
},
watch: {
items() {
this.updateIds();
}
}
});
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["fake_7703cae8.js"],"names":["Vue","component","VueSelect","props","data","ids","items","search","options","loading","itemHovered","value","limit","template","created","field_value","getPosts","stm_wpcfto_ajaxurl","stm_wpcfto_nonces","fields","post_type","join","clearItems","isLoading","field_data","methods","_isLoading","setSelected","push","$set","vm","filtered","filter","el","underLimit","length","onSearch","_this","exclude","post_types","url","variable","field_name","$http","get","then","response","body","updateIds","forEach","key","id","$emit","callFunction","functionName","item","model","containsObject","obj","list","i","removeItem","index","splice","watch"],"mappings":"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","sourcesContent":["\"use strict\";\n\nVue.component('v-select', VueSelect.VueSelect);\nVue.component('wpcfto_autocomplete', {\n  props: ['fields', 'field_label', 'field_name', 'field_id', 'field_value', 'field_data'],\n  data: function data() {\n    return {\n      ids: [],\n      items: [],\n      search: '',\n      options: [],\n      loading: true,\n      itemHovered: null,\n      value: '',\n      limit: 0\n    };\n  },\n  template: \"\\n        <div class=\\\"wpcfto_generic_field wpcfto_generic_field_autocomplete\\\">\\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 class=\\\"wpcfto-autocomplete-search\\\" v-bind:class=\\\"{'loading': loading}\\\">\\n                  \\n                    <div class=\\\"v-select-search\\\" v-if=\\\"underLimit()\\\">\\n\\n                        <i class=\\\"fa fa-plus-circle\\\"></i>\\n\\n                        <v-select label=\\\"title\\\"\\n                                  v-model=\\\"search\\\"\\n                                  @input=\\\"setSelected($event)\\\"\\n                                  :options=\\\"options\\\"\\n                                  @search=\\\"onSearch($event)\\\">\\n                        </v-select>\\n\\n                    </div>\\n\\n                    <ul class=\\\"wpcfto-autocomplete\\\" v-bind:class=\\\"{'limited' : !underLimit()}\\\">\\n                        <li v-for=\\\"(item, index) in items\\\" v-if=\\\"typeof item !== 'string'\\\" :class=\\\"{ 'hovered' : itemHovered == index }\\\">\\n                            <div class=\\\"item-wrapper\\\">\\n                                <img v-bind:src=\\\"item.image\\\" v-if=\\\"item.image\\\" class=\\\"item-image\\\">\\n                                <div class=\\\"item-data\\\">\\n                                    <span v-html=\\\"item.title\\\" class=\\\"item-title\\\"></span>\\n                                    <span v-html=\\\"item.excerpt\\\" class=\\\"item-excerpt\\\" v-if=\\\"item.excerpt\\\"></span>\\n                                </div>\\n                            </div>\\n                            <i class=\\\"fa fa-trash-alt\\\" @click=\\\"removeItem(index)\\\" @mouseover=\\\"itemHovered = index\\\" @mouseleave=\\\"itemHovered = null\\\"></i>\\n                        </li>\\n                    </ul>\\n\\n                    <input type=\\\"hidden\\\"\\n                           v-bind:name=\\\"field_name\\\"\\n                           v-model=\\\"value\\\"/>\\n\\n                </div>\\n            \\n            </div>\\n\\n            <wpcfto_fields_aside_after :fields=\\\"fields\\\"></wpcfto_fields_aside_after>\\n\\n        </div>\\n    \",\n  created: function created() {\n    if (this.field_value) {\n      this.getPosts(stm_wpcfto_ajaxurl + '?action=wpcfto_search_posts&nonce=' + stm_wpcfto_nonces['wpcfto_search_posts'] + '&posts_per_page=-1&orderby=post__in&ids=' + this.field_value + '&post_types=' + this.fields.post_type.join(','), 'items');\n    } else {\n      this.clearItems();\n      this.isLoading(false);\n    }\n\n    if (typeof this.field_data.limit !== 'undefined') {\n      this.limit = this.field_data.limit;\n    }\n  },\n  methods: {\n    isLoading: function isLoading(_isLoading) {\n      this.loading = _isLoading;\n    },\n    setSelected: function setSelected(value) {\n      if (value) this.items.push(value);\n      /*Reset options*/\n\n      this.$set(this, 'options', []);\n      this.$set(this, 'search', '');\n    },\n    clearItems: function clearItems() {\n      var vm = this;\n      var filtered = vm['items'].filter(function (el) {\n        return el != null || el !== '';\n      });\n      vm.$set(vm, 'items', filtered);\n    },\n    underLimit: function underLimit() {\n      return this.items.length < this.limit;\n    },\n    onSearch: function onSearch(search) {\n      var _this = this;\n\n      var exclude = _this.ids.join(',');\n\n      var post_types = _this.fields['post_type'].join(',');\n\n      _this.getPosts(stm_wpcfto_ajaxurl + '?action=wpcfto_search_posts&nonce=' + stm_wpcfto_nonces['wpcfto_search_posts'] + '&exclude_ids=' + exclude + '&s=' + search + '&post_types=' + post_types, 'options');\n    },\n    getPosts: function getPosts(url, variable) {\n      var vm = this;\n      vm.isLoading(true);\n      /*Adding field ID to filters then*/\n\n      url += '&name=' + vm.field_name;\n      this.$http.get(url).then(function (response) {\n        vm[variable] = response.body;\n        vm.clearItems();\n        vm.isLoading(false);\n      });\n    },\n    updateIds: function updateIds() {\n      var vm = this;\n      vm.ids = [];\n      this.items.forEach(function (value, key) {\n        vm.ids.push(value.id);\n      });\n      vm.$set(this, 'value', vm.ids);\n      vm.$emit('wpcfto-get-value', vm.ids);\n    },\n    callFunction: function callFunction(functionName, item, model) {\n      functionName(item, model);\n    },\n    containsObject: function containsObject(obj, list) {\n      var i;\n\n      for (i = 0; i < list.length; i++) {\n        if (list[i]['id'] === obj['id']) {\n          return true;\n        }\n      }\n\n      return false;\n    },\n    removeItem: function removeItem(index) {\n      this.items.splice(index, 1);\n    }\n  },\n  watch: {\n    items: function items() {\n      this.updateIds();\n    }\n  }\n});"]}
},{}]},{},[1])