diff --git a/addon/components/validated-input.js b/addon/components/validated-input.js
index d0510fe8..da614c67 100644
--- a/addon/components/validated-input.js
+++ b/addon/components/validated-input.js
@@ -88,8 +88,8 @@ export class ValidatedInput extends Component {
@action
update(value) {
- if (this["on-update"]) {
- this["on-update"](value, this.args.model);
+ if (this.args["on-update"]) {
+ this.args["on-update"](value, this.args.model);
} else {
this.args.model.set
? this.args.model.set(this.args.name, value)
diff --git a/addon/components/validated-input/render.js b/addon/components/validated-input/render.js
index 9ddde1a9..e248992e 100644
--- a/addon/components/validated-input/render.js
+++ b/addon/components/validated-input/render.js
@@ -7,6 +7,9 @@ export default Component.extend({
layout,
selectComponent: themedComponent("validated-input/types/select"),
radioGroupComponent: themedComponent("validated-input/types/radio-group"),
+ checkboxGroupComponent: themedComponent(
+ "validated-input/types/checkbox-group"
+ ),
checkboxComponent: themedComponent("validated-input/types/checkbox"),
textareaComponent: themedComponent("validated-input/types/textarea"),
inputComponent: themedComponent("validated-input/types/input"),
diff --git a/addon/components/validated-input/types/-themes/bootstrap/checkbox-group.js b/addon/components/validated-input/types/-themes/bootstrap/checkbox-group.js
new file mode 100644
index 00000000..8c207449
--- /dev/null
+++ b/addon/components/validated-input/types/-themes/bootstrap/checkbox-group.js
@@ -0,0 +1,6 @@
+import layout from "../../../../../templates/components/validated-input/types/-themes/bootstrap/checkbox-group";
+import Component from "../../checkbox-group";
+
+export default Component.extend({
+ layout,
+});
diff --git a/addon/components/validated-input/types/-themes/uikit/checkbox-group.js b/addon/components/validated-input/types/-themes/uikit/checkbox-group.js
new file mode 100644
index 00000000..33828b5d
--- /dev/null
+++ b/addon/components/validated-input/types/-themes/uikit/checkbox-group.js
@@ -0,0 +1,6 @@
+import layout from "../../../../../templates/components/validated-input/types/-themes/uikit/checkbox-group";
+import Component from "../../checkbox-group";
+
+export default Component.extend({
+ layout,
+});
diff --git a/addon/components/validated-input/types/checkbox-group.js b/addon/components/validated-input/types/checkbox-group.js
new file mode 100644
index 00000000..631154ac
--- /dev/null
+++ b/addon/components/validated-input/types/checkbox-group.js
@@ -0,0 +1,23 @@
+import Component from "@ember/component";
+import { action } from "@ember/object";
+
+import layout from "../../../templates/components/validated-input/types/checkbox-group";
+
+export default Component.extend({
+ layout,
+
+ @action
+ onUpdate(key) {
+ const value = this.value || [];
+ const index = value.indexOf(key);
+
+ if (index > -1) {
+ value.splice(index, 1);
+ } else {
+ value.push(key);
+ }
+
+ this.update(value);
+ this.setDirty();
+ },
+});
diff --git a/addon/templates/components/validated-input/-themes/uikit/render.hbs b/addon/templates/components/validated-input/-themes/uikit/render.hbs
index 1d3cee35..503ef096 100644
--- a/addon/templates/components/validated-input/-themes/uikit/render.hbs
+++ b/addon/templates/components/validated-input/-themes/uikit/render.hbs
@@ -33,6 +33,20 @@
isValid=isValid
isInvalid=isInvalid
+ update=update
+ setDirty=setDirty
+ }}
+ {{else if (or (eq type "checkboxGroup") (eq type "checkbox-group"))}}
+ {{component checkboxGroupComponent
+ value=value
+ options=options
+ inputId=inputId
+ name=(or inputName name)
+ disabled=disabled
+
+ isValid=isValid
+ isInvalid=isInvalid
+
update=update
setDirty=setDirty
}}
diff --git a/addon/templates/components/validated-input/render.hbs b/addon/templates/components/validated-input/render.hbs
index 89a1e2be..c896321d 100644
--- a/addon/templates/components/validated-input/render.hbs
+++ b/addon/templates/components/validated-input/render.hbs
@@ -33,6 +33,20 @@
isValid=isValid
isInvalid=isInvalid
+ update=update
+ setDirty=setDirty
+ }}
+{{else if (or (eq type "checkboxGroup") (eq type "checkbox-group"))}}
+ {{component checkboxGroupComponent
+ value=value
+ options=options
+ inputId=inputId
+ name=(or inputName name)
+ disabled=disabled
+
+ isValid=isValid
+ isInvalid=isInvalid
+
update=update
setDirty=setDirty
}}
diff --git a/addon/templates/components/validated-input/types/-themes/bootstrap/checkbox-group.hbs b/addon/templates/components/validated-input/types/-themes/bootstrap/checkbox-group.hbs
new file mode 100644
index 00000000..3baf7109
--- /dev/null
+++ b/addon/templates/components/validated-input/types/-themes/bootstrap/checkbox-group.hbs
@@ -0,0 +1,14 @@
+{{#each options as |option i|}}
+
+
+
+
+{{/each}}
\ No newline at end of file
diff --git a/addon/templates/components/validated-input/types/-themes/uikit/checkbox-group.hbs b/addon/templates/components/validated-input/types/-themes/uikit/checkbox-group.hbs
new file mode 100644
index 00000000..11b8a498
--- /dev/null
+++ b/addon/templates/components/validated-input/types/-themes/uikit/checkbox-group.hbs
@@ -0,0 +1,15 @@
+{{#each options as |option i|}}
+ {{#unless (eq i 0)}}
{{/unless}}
+
+{{/each}}
\ No newline at end of file
diff --git a/addon/templates/components/validated-input/types/checkbox-group.hbs b/addon/templates/components/validated-input/types/checkbox-group.hbs
new file mode 100644
index 00000000..cf856134
--- /dev/null
+++ b/addon/templates/components/validated-input/types/checkbox-group.hbs
@@ -0,0 +1,14 @@
+{{#each options as |option i|}}
+
+{{/each}}
\ No newline at end of file
diff --git a/app/components/validated-input/types/-themes/bootstrap/checkbox-group.js b/app/components/validated-input/types/-themes/bootstrap/checkbox-group.js
new file mode 100644
index 00000000..58edb9d1
--- /dev/null
+++ b/app/components/validated-input/types/-themes/bootstrap/checkbox-group.js
@@ -0,0 +1 @@
+export { default } from "ember-validated-form/components/validated-input/types/-themes/bootstrap/checkbox-group";
diff --git a/app/components/validated-input/types/-themes/uikit/checkbox-group.js b/app/components/validated-input/types/-themes/uikit/checkbox-group.js
new file mode 100644
index 00000000..26c69b96
--- /dev/null
+++ b/app/components/validated-input/types/-themes/uikit/checkbox-group.js
@@ -0,0 +1 @@
+export { default } from "ember-validated-form/components/validated-input/types/-themes/uikit/checkbox-group";
diff --git a/app/components/validated-input/types/checkbox-group.js b/app/components/validated-input/types/checkbox-group.js
new file mode 100644
index 00000000..3715a102
--- /dev/null
+++ b/app/components/validated-input/types/checkbox-group.js
@@ -0,0 +1 @@
+export { default } from "ember-validated-form/components/validated-input/types/checkbox-group";
diff --git a/testem.js b/testem.js
index 44f1a159..8710e47a 100644
--- a/testem.js
+++ b/testem.js
@@ -4,7 +4,7 @@ module.exports = {
test_page: "tests/index.html?hidepassed",
disable_watching: true,
launch_in_ci: ["Chrome"],
- launch_in_dev: ["Chrome"],
+ launch_in_dev: [],
browser_start_timeout: 120,
browser_args: {
Chrome: {
diff --git a/tests/dummy/app/templates/docs/components/validated-input.md b/tests/dummy/app/templates/docs/components/validated-input.md
index 1ca11127..26e86fd9 100644
--- a/tests/dummy/app/templates/docs/components/validated-input.md
+++ b/tests/dummy/app/templates/docs/components/validated-input.md
@@ -42,7 +42,7 @@ two arguments: `update(value, changeset)`.
**autocomplete ``**
Binding to the [`` `autocomplete` attribute](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input#attr-autocomplete).
-The supported field types are `checkbox`, `radioGroup`, `select`, `textarea`
+The supported field types are `checkbox`, `checkboxGroup`, `radioGroup`, `select`, `textarea`
and any type that can be specified on an `` element. This addon does
not much more than translating `{{f.input type='select'}}` to
`{{one-way-select}}` or `{{f.input type='text'}}` to ``
@@ -189,3 +189,24 @@ localize your labels using an internationalization addon like
{{demo.snippet 'translations.js' label='locales/fr/translations.js'}}
{{/docs-demo}}
-->
+
+### Checkbox group
+
+This component renders a list of `` elements.
+
+
+{{#docs-demo as |demo|}}
+ {{#demo.example name='checkbox-group-template.hbs'}}
+ {{#validated-form model=(changeset (hash shape=null)) as |f|}}
+ {{f.input
+ type = 'checkboxGroup'
+ label = 'Shapes'
+ name = 'shape'
+ options = (array (hash key='t' label='Triangle') (hash key='s' label='Square') (hash key='c' label='Circle'))
+ }}
+ {{/validated-form}}
+ {{/demo.example}}
+
+ {{demo.snippet 'checkbox-group-template.hbs'}}
+{{/docs-demo}}
+
diff --git a/tests/integration/components/validated-input-test.js b/tests/integration/components/validated-input-test.js
index 462e137a..79ab7b96 100644
--- a/tests/integration/components/validated-input-test.js
+++ b/tests/integration/components/validated-input-test.js
@@ -1,4 +1,4 @@
-import { render, click, settled } from "@ember/test-helpers";
+import { render, click, fillIn, settled } from "@ember/test-helpers";
import Changeset from "ember-changeset";
import { hbs } from "ember-cli-htmlbars";
import { setupRenderingTest } from "ember-qunit";
@@ -52,6 +52,20 @@ module("Integration | Component | validated input", function (hooks) {
assert.dom("input").hasValue("Max");
});
+ test("it calls on-update if given", async function (assert) {
+ this.set("model", new Changeset({ firstName: "Max" }));
+ this.set("update", (value, changeset) => {
+ changeset.set("firstName", value.toUpperCase());
+ });
+ await render(
+ hbs`{{validated-input name="firstName" model=model on-update=update}}`
+ );
+
+ await fillIn("input", "foo");
+
+ assert.dom("input").hasValue("FOO");
+ });
+
test("it renders inputs with value even if model is defined", async function (assert) {
this.set("model", new Changeset({ firstName: "Max" }));
diff --git a/tests/integration/components/validated-input/types/-themes/bootstrap/checkbox-group-test.js b/tests/integration/components/validated-input/types/-themes/bootstrap/checkbox-group-test.js
new file mode 100644
index 00000000..a8088c30
--- /dev/null
+++ b/tests/integration/components/validated-input/types/-themes/bootstrap/checkbox-group-test.js
@@ -0,0 +1,26 @@
+import { render } from "@ember/test-helpers";
+import setupConfigTest from "dummy/tests/helpers/setup-config-test";
+import { hbs } from "ember-cli-htmlbars";
+import { setupRenderingTest } from "ember-qunit";
+import { module, test } from "qunit";
+
+module(
+ "Integration | Component | validated-input/types/-themes/bootstrap/checkbox-group",
+ function (hooks) {
+ setupRenderingTest(hooks);
+ setupConfigTest(hooks, { theme: "bootstrap" });
+
+ test("it renders", async function (assert) {
+ await render(hbs`
+ {{validated-input/types/-themes/bootstrap/checkbox-group
+ options = (array (hash key='t' label='Triangle') (hash key='s' label='Square'))
+ update=(action (mut value))
+ }}
+ `);
+
+ assert.dom("div.custom-control.custom-checkbox").exists();
+ assert.dom("input").hasClass("custom-control-input");
+ assert.dom("label").hasClass("custom-control-label");
+ });
+ }
+);
diff --git a/tests/integration/components/validated-input/types/-themes/uikit/checkbox-group-test.js b/tests/integration/components/validated-input/types/-themes/uikit/checkbox-group-test.js
new file mode 100644
index 00000000..70a66775
--- /dev/null
+++ b/tests/integration/components/validated-input/types/-themes/uikit/checkbox-group-test.js
@@ -0,0 +1,34 @@
+import { render } from "@ember/test-helpers";
+import setupConfigTest from "dummy/tests/helpers/setup-config-test";
+import { setupRenderingTest } from "ember-qunit";
+import hbs from "htmlbars-inline-precompile";
+import { module, test } from "qunit";
+
+module(
+ "Integration | Component | validated-input/types/-themes/uikit/checkbox-group",
+ function (hooks) {
+ setupRenderingTest(hooks);
+ setupConfigTest(hooks, { theme: "uikit" });
+
+ test("it renders", async function (assert) {
+ this.set("options", [
+ {
+ key: "opt1",
+ label: "Option 1",
+ },
+ {
+ key: "opt2",
+ label: "Option 2",
+ },
+ ]);
+
+ await render(
+ hbs`{{validated-input/types/-themes/uikit/checkbox-group options=options update=(action (mut value))}}`
+ );
+
+ assert.dom("label > input").exists();
+ assert.dom("input").hasClass("uk-checkbox");
+ assert.dom("label").hasClass("uk-form-label");
+ });
+ }
+);
diff --git a/tests/integration/components/validated-input/types/checkbox-group-test.js b/tests/integration/components/validated-input/types/checkbox-group-test.js
new file mode 100644
index 00000000..61aeb6eb
--- /dev/null
+++ b/tests/integration/components/validated-input/types/checkbox-group-test.js
@@ -0,0 +1,27 @@
+import { render } from "@ember/test-helpers";
+import { setupRenderingTest } from "ember-qunit";
+import hbs from "htmlbars-inline-precompile";
+import { module, test } from "qunit";
+
+module(
+ "Integration | Component | validated-input/types/checkbox-group",
+ function (hooks) {
+ setupRenderingTest(hooks);
+
+ test("it renders", async function (assert) {
+ this.set("options", [
+ { key: 1, label: 1 },
+ { key: 2, label: 2 },
+ ]);
+
+ await render(hbs`
+ {{validated-input/types/checkbox-group
+ options=options
+ update=(action (mut value))
+ }}
+ `);
+
+ assert.dom("input[type=checkbox]").exists({ count: 2 });
+ });
+ }
+);