From 9a149d320c634ab87d388e8591fe6409fc4e6a78 Mon Sep 17 00:00:00 2001 From: Icebob Date: Mon, 8 Aug 2016 15:03:50 +0200 Subject: [PATCH 1/2] add fieldSubmit.vue --- dev/schema.js | 871 +++++++++++++++++++------------------ src/fields/fieldSubmit.vue | 34 ++ 2 files changed, 474 insertions(+), 431 deletions(-) create mode 100644 src/fields/fieldSubmit.vue diff --git a/dev/schema.js b/dev/schema.js index a5e3c1d2..ea98db95 100644 --- a/dev/schema.js +++ b/dev/schema.js @@ -7,458 +7,467 @@ 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 - }, + 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: "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); + { + 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 - }, + }, + // 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: "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"], + { + 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 + 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: "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: "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: "DT", - model: "dt", - multi: true, - validator: [ - validators.date - ], - dateTimePickerOptions: { - format: "YYYY-MM-DD HH:mm:ss" - } + { + type: "dateTime", + label: "DT", + model: "dt", + multi: true, + validator: [ + validators.date + ], + dateTimePickerOptions: { + format: "YYYY-MM-DD HH:mm:ss" + } - }, + }, - { - type: "dateTime", - label: "Time", - model: "time", - multi: true, - format: "HH:mm:ss", - /*validator: [ - validators.time - ],*/ - dateTimePickerOptions: { - format: "HH:mm:ss" - } + { + type: "dateTime", + label: "Time", + model: "time", + multi: true, + format: "HH:mm:ss", + /*validator: [ + validators.time + ],*/ + dateTimePickerOptions: { + format: "HH:mm:ss" + } - }, + }, - { - type: "switch", - label: "Sex (switch field)", - model: "sex", - multi: true, - default: "male", - textOn: "Female", - textOff: "Male", - valueOn: "female", - valueOff: "male" - }, + { + type: "switch", + label: "Sex (switch field)", + model: "sex", + multi: true, + default: "male", + textOn: "Female", + textOff: "Male", + valueOn: "female", + valueOff: "male" + }, - { - 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: "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: "slider", + label: "Income", + model: "income", + multi: true, + min: 0, + max: 100000, + sliderOptions: { + type: "double", + prefix: "$", + step: 1000 + } + }, - { - 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: "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: "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: "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 (!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 + if (values.length > 0 && values[0].trim() != "") + model.address.geo.latitude = parseFloat(values[0].trim()); + else + model.address.geo.latitude = 0 - 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 (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 = {}; - 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 - } - ] + 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 + }, { + type: "submit", + label: "", + caption: "Submit form", + validateBeforeSubmit: true, + onSubmit(model, schema) { + console.log("Form submitted!", model); + alert("Form submitted!"); + } + } + ] } diff --git a/src/fields/fieldSubmit.vue b/src/fields/fieldSubmit.vue new file mode 100644 index 00000000..7fac5101 --- /dev/null +++ b/src/fields/fieldSubmit.vue @@ -0,0 +1,34 @@ + + + + + From 7b575c31f0b85bcd764ee4203b8458d91dfc0067 Mon Sep 17 00:00:00 2001 From: Icebob Date: Mon, 8 Aug 2016 15:27:24 +0200 Subject: [PATCH 2/2] create test for fieldSubmit --- test/unit/specs/fields/fieldSubmit.spec.js | 68 ++++++++++++++++++++++ 1 file changed, 68 insertions(+) create mode 100644 test/unit/specs/fields/fieldSubmit.spec.js diff --git a/test/unit/specs/fields/fieldSubmit.spec.js b/test/unit/specs/fields/fieldSubmit.spec.js new file mode 100644 index 00000000..098a194c --- /dev/null +++ b/test/unit/specs/fields/fieldSubmit.spec.js @@ -0,0 +1,68 @@ +/* global sinon */ +import { expect } from "chai"; +import { createVueField } from "../util"; + +import Vue from "vue"; +import FieldSubmit from "src/fields/fieldSubmit.vue"; + +Vue.component("FieldSubmit", FieldSubmit); + +// eslint-disable-next-line +let el, vm, field; + +function createField(schema = {}, model = null, disabled = false, options) { + [ el, vm, field ] = createVueField("fieldSubmit", schema, model, disabled, options); +} + +describe("fieldSubmit.vue", () => { + + describe("check template", () => { + let schema = { + type: "submit", + caption: "Submit form", + validateBeforeSubmit: false, + onSubmit() {} + }; + let model = { name: "John Doe" }; + let input; + + before( () => { + createField(schema, model, false); + input = el.getElementsByTagName("input")[0]; + }); + + it("should contain an input submit element", () => { + expect(field).to.be.exist; + expect(field.$el).to.be.exist; + + expect(input).to.be.defined; + expect(input.type).to.be.equal("submit"); + expect(input.value).to.be.equal("Submit form"); + }); + + it("should not call validate if validateBeforeSubmit is false", () => { + schema.onSubmit = sinon.spy(); + let cb = sinon.spy(); + field.$parent.validate = cb; + + input.click(); + expect(cb.called).to.be.false; + expect(schema.onSubmit.calledOnce).to.be.true; + expect(schema.onSubmit.calledWith(model, schema)).to.be.true; + }); + + + it("should call validate if validateBeforeSubmit is true", () => { + schema.validateBeforeSubmit = true; + schema.onSubmit = sinon.spy(); + let cb = sinon.spy(); + field.$parent.validate = cb; + + input.click(); + expect(cb.called).to.be.true; + expect(schema.onSubmit.called).to.be.false; + }); + + }); + +}); \ No newline at end of file