diff --git a/.jsbeautifyrc b/.jsbeautifyrc new file mode 100644 index 00000000..71c7efbf --- /dev/null +++ b/.jsbeautifyrc @@ -0,0 +1,6 @@ +{ + "html": { + "indent_char": "\t", + "indent_size": 1 + } +} \ No newline at end of file diff --git a/dev/index.html b/dev/index.html index 95feb7f0..5987ae71 100644 --- a/dev/index.html +++ b/dev/index.html @@ -20,6 +20,7 @@ + diff --git a/dev/schema.js b/dev/schema.js index 5a11cf7f..a5e3c1d2 100644 --- a/dev/schema.js +++ b/dev/schema.js @@ -7,424 +7,458 @@ import { validators } from "../src"; let fakerator = new Fakerator(); module.exports = { - fields: [ - { - type: "text", - label: "ID (disabled text field)", - model: "id", - readonly: true, - editableIfNew: true, // TODO - featured: false, - disabled: true - }, - { - type: "select", - label: "Type (select field)", - model: "type", - required: true, - values: [ - { id: "personal", name: "Personal" }, - { id: "business", name: "Business" } - ], - default: "personal" - }, - { - type: "text", - label: "First name", - model: "firstName", - featured: true, - required: true, - placeholder: "User's first name", - styleClasses: "half-width", - validator: validators.string, + fields: [{ + type: "text", + label: "ID (disabled text field)", + model: "id", + readonly: true, + editableIfNew: true, // TODO + featured: false, + disabled: true + }, { + type: "select", + label: "Type (select field)", + model: "type", + required: true, + values: [ + { id: "personal", name: "Personal" }, + { id: "business", name: "Business" } + ], + default: "personal" + }, { + type: "text", + label: "First name", + model: "firstName", + featured: true, + required: true, + placeholder: "User's first name", + styleClasses: "half-width", + validator: validators.string, - onChanged(model, newVal, oldVal, field) { - //console.log(`Model's name changed from ${oldVal} to ${newVal}. Model:`, model); - }, + onChanged(model, newVal, oldVal, field) { + //console.log(`Model's name changed from ${oldVal} to ${newVal}. Model:`, model); + }, - onValidated(model, errors, field) { - //if (errors.length > 0) - // console.warn("Validation error in Name field! Errors:", errors); - } - }, - { - type: "text", - label: "Last name", - model: "lastName", - featured: true, - required: true, - placeholder: "User's last name", - styleClasses: "half-width", - validator: validators.string - }, - { - type: "text", - label: "Username", - model: "userName", - featured: true, - required: true, - min: 5, - placeholder: "User's last name", - styleClasses: ["half-width", "first"], - validator: validators.string - }, - { - type: "password", - label: "Password (password field)", - model: "password", - min: 6, - required: true, - hint: "Minimum 6 characters", - styleClasses: "half-width", - validator: validators.string - }, - - { - type: "selectEx", - label: "Skills (selectEx field)", - model: "skills", - multi: true, - required: true, - multiSelect: true, - selectOptions: { - // https://silviomoreto.github.io/bootstrap-select/options/ - liveSearch: true, - //maxOptions: 3, - //size: 4, - //actionsBox: true, - selectedTextFormat: "count > 3" - }, - values: [ - "HTML5", - "Javascript", - "CSS3", - "CoffeeScript", - "AngularJS", - "ReactJS", - "VueJS" - ], - min: 2, - max: 4, - validator: validators.array - }, - { - type: "text", - label: "Company name", - model: "company.name", - styleClasses: ["company", "half-width"], + onValidated(model, errors, field) { + //if (errors.length > 0) + // console.warn("Validation error in Name field! Errors:", errors); + } + }, { + type: "text", + label: "Last name", + model: "lastName", + featured: true, + required: true, + placeholder: "User's last name", + styleClasses: "half-width", + validator: validators.string + }, { + type: "text", + label: "Username", + model: "userName", + featured: true, + required: true, + min: 5, + placeholder: "User's last name", + styleClasses: ["half-width", "first"], + validator: validators.string + }, { + type: "password", + label: "Password (password field)", + model: "password", + min: 6, + required: true, + hint: "Minimum 6 characters", + styleClasses: "half-width", + validator: validators.string + }, - visible(model) { return model && model.type == "business"; } - }, - { - type: "text", - label: "Company phone", - model: "company.phone", - styleClasses: "company", - pattern: "^\\+[0-9]{2}-[237]0-[0-9]{3}-[0-9]{4}$", - placeholder: "User's phone number", - hint: "Format: +36-(20|30|70)-000-0000", - styleClasses: "half-width", - visible(model) { return model && model.type == "business"; } - }, - { - type: "email", - label: "E-mail (email field)", - model: "email", - placeholder: "User's e-mail address" - }, - { - type: "text", - label: "Phone", - model: "phone", - pattern: "^\\+[0-9]{2}-[237]0-[0-9]{3}-[0-9]{4}$", - placeholder: "User's phone number", - hint: "Format: +36-(20|30|70)-000-0000", - help: "You can use any formatted texts. Or place a link to another site.", - styleClasses: "half-width" - //validator: validators.regexp - }, - { - type: "masked", - label: "Mobile (masked field)", - model: "mobile", - mask: "(99) 999-9999", - styleClasses: "half-width", - validator: validators.required - }, - { - type: "spectrum", - label: "Color (spectrum field)", - model: "favoriteColor", - required: true, - colorOptions: { - //preferredFormat: "rgb" - }, - validator: validators.required - }, - { - type: "image", - label: "Avatar (image field)", - model: "avatar", - required: true, - browse: true, - validator: validators.required + { + type: "vueMultiSelect", + label: "Skills (vue-multiSelect field)", + model: "skills", + required: true, + multiSelect: true, + selectOptions: { + // id:25, + // key:"name", + // label: "name", + searchable: true, + clearOnSelect: true, + hideSelected: true, + // maxHeight:300, + // allowEmpty:true, + // resetAfter:false, + // closeOnSelect: true, + // customLabel:function(){return ""}, + taggable: true, + tagPlaceholder: 'tagPlaceholder', + onNewTag(newTag, id, options, value) { + console.log("onNewTag", newTag, id, options, value); + options.push(newTag); + value.push(newTag); + }, + // showPointer: true, + onSearch(searchQuery, id, options){ + console.log("onSearch",searchQuery, id, options); + } + // selectLabel: "selectLabel", + // selectedLabel: "selectedLabel", + // deselectLabel: "deselectLabel", + // limit:2, + // limitText: count => `and ${count} more`, + // loading: false + }, + values: [ + "HTML5", + "Javascript", + "CSS3", + "CoffeeScript", + "AngularJS", + "ReactJS", + "VueJS" + ], + onChanged(model, newVal, oldVal, field) { + console.log(`Model's name changed from ${oldVal} to ${newVal}. Model:`, model); + }, + min: 2, + max: 4, + placeholder: "placeholder", + validator: validators.array + }, - }, - { - type: "number", - label: "Age (number field)", - model: "age", - multi: true, - disabled: true, - placeholder: "User's age", - hint: "Minimum 18 age.", - min: 18, - max: 100, - validator: [ - validators.integer, - validators.number - ] - }, - { - type: "dateTime", - label: "DOB (dateTime field)", - model: "dob", - required: true, - placeholder: "User's birth of date", - min: moment("1900-01-01").toDate(), - max: moment("2016-01-01").toDate(), - validator: [ - validators.date - ], - dateTimePickerOptions: { - format: "YYYY-MM-DD" - }, - onChanged(model, newVal, oldVal, field) { - model.age = moment().year() - moment(newVal).year(); - } + { + type: "selectEx", + label: "Skills (selectEx field)", + model: "skills", + multi: true, + required: true, + multiSelect: true, + selectOptions: { + // https://silviomoreto.github.io/bootstrap-select/options/ + liveSearch: true, + //maxOptions: 3, + //size: 4, + //actionsBox: true, + selectedTextFormat: "count > 3" + }, + values: [ + "HTML5", + "Javascript", + "CSS3", + "CoffeeScript", + "AngularJS", + "ReactJS", + "VueJS" + ], + min: 2, + max: 4, + validator: validators.array + }, { + type: "text", + label: "Company name", + model: "company.name", + styleClasses: ["company", "half-width"], - }, + visible(model) { + return model && model.type == "business"; + } + }, { + type: "text", + label: "Company phone", + model: "company.phone", + styleClasses: "company", + pattern: "^\\+[0-9]{2}-[237]0-[0-9]{3}-[0-9]{4}$", + placeholder: "User's phone number", + hint: "Format: +36-(20|30|70)-000-0000", + styleClasses: "half-width", + visible(model) { + return model && model.type == "business"; + } + }, { + type: "email", + label: "E-mail (email field)", + model: "email", + placeholder: "User's e-mail address" + }, { + type: "text", + label: "Phone", + model: "phone", + pattern: "^\\+[0-9]{2}-[237]0-[0-9]{3}-[0-9]{4}$", + placeholder: "User's phone number", + hint: "Format: +36-(20|30|70)-000-0000", + help: "You can use any formatted texts. Or place a link to another site.", + styleClasses: "half-width" + //validator: validators.regexp + }, { + type: "masked", + label: "Mobile (masked field)", + model: "mobile", + mask: "(99) 999-9999", + styleClasses: "half-width", + validator: validators.required + }, { + type: "spectrum", + label: "Color (spectrum field)", + model: "favoriteColor", + required: true, + colorOptions: { + //preferredFormat: "rgb" + }, + validator: validators.required + }, { + type: "image", + label: "Avatar (image field)", + model: "avatar", + required: true, + browse: true, + validator: validators.required - { - type: "dateTime", - label: "DT", - model: "dt", - multi: true, - validator: [ - validators.date - ], - dateTimePickerOptions: { - format: "YYYY-MM-DD HH:mm:ss" - } + }, { + type: "number", + label: "Age (number field)", + model: "age", + multi: true, + disabled: true, + placeholder: "User's age", + hint: "Minimum 18 age.", + min: 18, + max: 100, + validator: [ + validators.integer, + validators.number + ] + }, { + type: "dateTime", + label: "DOB (dateTime field)", + model: "dob", + required: true, + placeholder: "User's birth of date", + min: moment("1900-01-01").toDate(), + max: moment("2016-01-01").toDate(), + validator: [ + validators.date + ], + dateTimePickerOptions: { + format: "YYYY-MM-DD" + }, + onChanged(model, newVal, oldVal, field) { + model.age = moment().year() - moment(newVal).year(); + } - }, + }, - { - type: "dateTime", - label: "Time", - model: "time", - multi: true, - format: "HH:mm:ss", - /*validator: [ - validators.time - ],*/ - dateTimePickerOptions: { - format: "HH:mm:ss" - } + { + type: "dateTime", + label: "DT", + model: "dt", + multi: true, + validator: [ + validators.date + ], + dateTimePickerOptions: { + format: "YYYY-MM-DD HH:mm:ss" + } - }, + }, - { - type: "switch", - label: "Sex (switch field)", - model: "sex", - multi: true, - default: "male", - textOn: "Female", - textOff: "Male", - valueOn: "female", - valueOff: "male" - }, + { + type: "dateTime", + label: "Time", + model: "time", + multi: true, + format: "HH:mm:ss", + /*validator: [ + validators.time + ],*/ + dateTimePickerOptions: { + format: "HH:mm:ss" + } - { - type: "slider", - label: "Rank (slider field)", - model: "rank", - multi: true, - min: 1, - max: 10, - required: true, - sliderOptions: { - grid: true - }, - validator: validators.integer - }, + }, - { - type: "slider", - label: "Income", - model: "income", - multi: true, - min: 0, - max: 100000, - sliderOptions: { - type: "double", - prefix: "$", - step: 1000 - } - }, + { + type: "switch", + label: "Sex (switch field)", + model: "sex", + multi: true, + default: "male", + textOn: "Female", + textOff: "Male", + valueOn: "female", + valueOff: "male" + }, - { - type: "select", - label: "Role", - model: "role", - required: true, - values: [ - { id: "admin", name: "Administrator"}, - { id: "moderator", name: "Moderator"}, - { id: "user", name: "Registered User"}, - { id: "visitor", name: "Visitor"} - ], - styleClasses: "half-width", - validator: validators.required - }, + { + type: "slider", + label: "Rank (slider field)", + model: "rank", + multi: true, + min: 1, + max: 10, + required: true, + sliderOptions: { + grid: true + }, + validator: validators.integer + }, - { - type: "select", - label: "Language", - model: "language", - required: true, - values: [ - { id: "en-GB", name: "English (GB)" }, - { id: "en-US", name: "English (US)" }, - { id: "de", name: "German" }, - { id: "it", name: "Italic" }, - { id: "fr", name: "French" } - ], - hint: "Your native language", - styleClasses: "half-width", - validator: validators.required - }, - { - type: "selectEx", - label: "Country (selectEx field)", - model: "address.country", - multi: true, - required: true, - values: ["United Kingdom", "France", "Germany"], - //default: "United Kingdom", - multiSelect: false, - selectOptions: { - // https://silviomoreto.github.io/bootstrap-select/options/ - liveSearch: true, - size: 10 - }, - styleClasses: ["half-width", "first"], - validator: validators.required - }, - { - type: "text", - label: "City", - model: "address.city", - multi: true, - styleClasses: "half-width", - validator: validators.required - }, - { - type: "text", - label: "Street", - model: "address.street" - }, - { - type: "text", - label: "GPS", - model: "address.geo", - disabled: false, - get(model) { - if (model && model.address && model.address.geo) - return model.address.geo.latitude + ", " + model.address.geo.longitude; - }, - set(model, val) { - let values = val.split(","); - if (!model.address) - model.address = {}; + { + type: "slider", + label: "Income", + model: "income", + multi: true, + min: 0, + max: 100000, + sliderOptions: { + type: "double", + prefix: "$", + step: 1000 + } + }, - if (!model.address.geo) - model.address.geo = {}; + { + type: "select", + label: "Role", + model: "role", + required: true, + values: [ + { id: "admin", name: "Administrator" }, + { id: "moderator", name: "Moderator" }, + { id: "user", name: "Registered User" }, + { id: "visitor", name: "Visitor" } + ], + styleClasses: "half-width", + validator: validators.required + }, - if (values.length > 0 && values[0].trim() != "") - model.address.geo.latitude = parseFloat(values[0].trim()); - else - model.address.geo.latitude = 0 + { + type: "select", + label: "Language", + model: "language", + required: true, + values: [ + { id: "en-GB", name: "English (GB)" }, + { id: "en-US", name: "English (US)" }, + { id: "de", name: "German" }, + { id: "it", name: "Italic" }, + { id: "fr", name: "French" } + ], + hint: "Your native language", + styleClasses: "half-width", + validator: validators.required + }, { + type: "selectEx", + label: "Country (selectEx field)", + model: "address.country", + multi: true, + required: true, + values: ["United Kingdom", "France", "Germany"], + //default: "United Kingdom", + multiSelect: false, + selectOptions: { + // https://silviomoreto.github.io/bootstrap-select/options/ + liveSearch: true, + size: 10 + }, + styleClasses: ["half-width", "first"], + validator: validators.required + }, { + type: "text", + label: "City", + model: "address.city", + multi: true, + styleClasses: "half-width", + validator: validators.required + }, { + type: "text", + label: "Street", + model: "address.street" + }, { + type: "text", + label: "GPS", + model: "address.geo", + disabled: false, + get(model) { + if (model && model.address && model.address.geo) + return model.address.geo.latitude + ", " + model.address.geo.longitude; + }, + set(model, val) { + let values = val.split(","); + if (!model.address) + model.address = {}; - if (values.length > 1 && values[1].trim() != "") - model.address.geo.longitude = parseFloat(values[1].trim()); - else - model.address.geo.longitude = 0 - }, - buttons: [ - { - classes: "btn-location", - label: "Current location", - onclick: function(model) { - if (navigator.geolocation) { - navigator.geolocation.getCurrentPosition((pos) => { - if (!model.address) - model.address = {}; + if (!model.address.geo) + model.address.geo = {}; - if (!model.address.geo) - model.address.geo = {}; + if (values.length > 0 && values[0].trim() != "") + model.address.geo.latitude = parseFloat(values[0].trim()); + else + model.address.geo.latitude = 0 - model.address.geo.latitude = pos.coords.latitude.toFixed(5); - model.address.geo.longitude = pos.coords.longitude.toFixed(5); - }); - } else { - alert("Geolocation is not supported by this browser."); - } - } - }, - { - classes: "btn-clear", - label: "Clear", - onclick: function(model) { - model.address.geo = { - latitude: 0, - longitude: 0 - }; - } - } - ] - }, - { - type: "staticMap", - label: "Map", - model: "address.geo", - visible: false - }, - { - type: "label", - label: "Created (label field)", - model: "created", - get(model) { return model && model.created ? moment(model.created).format("LLL") : "-"; } - }, - { - type: "switch", - label: "Status (switch field)", - model: "status", - multi: true, - default: true, - textOn: "Active", - textOff: "Inactive" - }, - { - type: "textArea", - label: "Biography (textArea field)", - model: "bio", - hint: "Max 500 characters", - max: 500, - placeholder: "User's biography", - rows: 4, - validator: validators.string - } - ] -} \ No newline at end of file + if (values.length > 1 && values[1].trim() != "") + model.address.geo.longitude = parseFloat(values[1].trim()); + else + model.address.geo.longitude = 0 + }, + buttons: [{ + classes: "btn-location", + label: "Current location", + onclick: function(model) { + if (navigator.geolocation) { + navigator.geolocation.getCurrentPosition((pos) => { + if (!model.address) + model.address = {}; + + if (!model.address.geo) + model.address.geo = {}; + + model.address.geo.latitude = pos.coords.latitude.toFixed(5); + model.address.geo.longitude = pos.coords.longitude.toFixed(5); + }); + } else { + alert("Geolocation is not supported by this browser."); + } + } + }, { + classes: "btn-clear", + label: "Clear", + onclick: function(model) { + model.address.geo = { + latitude: 0, + longitude: 0 + }; + } + }] + }, { + type: "staticMap", + label: "Map", + model: "address.geo", + visible: false + }, { + type: "label", + label: "Created (label field)", + model: "created", + get(model) { + return model && model.created ? moment(model.created).format("LLL") : "-"; + } + }, { + type: "switch", + label: "Status (switch field)", + model: "status", + multi: true, + default: true, + textOn: "Active", + textOff: "Inactive" + }, { + type: "textArea", + label: "Biography (textArea field)", + model: "bio", + hint: "Max 500 characters", + max: 500, + placeholder: "User's biography", + rows: 4, + validator: validators.string + } + ] +} diff --git a/package.json b/package.json index ce1b4582..3f1e6e38 100644 --- a/package.json +++ b/package.json @@ -97,4 +97,4 @@ "babel-runtime": "6.11.6", "vue": "1.0.26" } -} +} \ No newline at end of file diff --git a/src/fields/fieldVueMultiSelect.vue b/src/fields/fieldVueMultiSelect.vue new file mode 100644 index 00000000..ac798c38 --- /dev/null +++ b/src/fields/fieldVueMultiSelect.vue @@ -0,0 +1,94 @@ + + + diff --git a/test/unit/specs/fields/fieldVueMultiSelect.spec.js b/test/unit/specs/fields/fieldVueMultiSelect.spec.js new file mode 100644 index 00000000..2d7b3fcf --- /dev/null +++ b/test/unit/specs/fields/fieldVueMultiSelect.spec.js @@ -0,0 +1,49 @@ +import { expect } from "chai"; +import { createVueField } from "../util"; + +import Vue from "vue"; +import fieldVueMultiSelect from "src/fields/fieldVueMultiSelect.vue"; + +Vue.component("fieldVueMultiSelect", fieldVueMultiSelect); + +// eslint-disable-next-line +let el, vm, field; + +function createField(schema = {}, model = null, disabled = false, options) { + [ el, vm, field ] = createVueField("fieldVueMultiSelect", schema, model, disabled, options); +} + +describe("fieldVueMultiSelect.vue", () => { + + describe("check template", () => { + let schema = { + type: "vueMultiSelect", + label: "Cities", + model: "city", + multiSelect: false, + required: false, + values: [ + "London", + "Paris", + "Rome", + "Berlin" + ] + }; + let model = { city: "Paris" }; + let input; + + before( () => { + createField(schema, model, false); + input = el.getElementsByTagName("select")[0]; + }); + + it("should contain a select element", () => { + expect(field).to.be.exist; + expect(field.$el).to.be.exist; + + expect(input).to.be.defined; + // expect(input.classList.contains("form-control")).to.be.false; + // expect(input.disabled).to.be.false; + }); + }); +}); \ No newline at end of file