Skip to content

Commit

Permalink
Merge pull request #1875 from nextcloud/enh/checkbox-radio
Browse files Browse the repository at this point in the history
Add CheckboxRadio component
  • Loading branch information
ChristophWurst authored Apr 27, 2021
2 parents ec28b3c + c7e5b37 commit 2caf2de
Show file tree
Hide file tree
Showing 3 changed files with 374 additions and 0 deletions.
347 changes: 347 additions & 0 deletions src/components/CheckboxRadio/CheckboxRadio.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,347 @@
<!--
- @copyright Copyright (c) 2021 John Molakvoæ <skjnldsv@protonmail.com>
-
- @author John Molakvoæ <skjnldsv@protonmail.com>
-
- @license GNU AGPL version 3 or any later version
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU Affero General Public License as
- published by the Free Software Foundation, either version 3 of the
- License, or (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU Affero General Public License for more details.
-
- You should have received a copy of the GNU Affero General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
-
-->

<docs>

### General description

This is a standard input checkbox/radio design

### Standard checkbox
```vue
<template>
<div>
<CheckboxRadio :checked.sync="sharingEnabled">Enable sharing</CheckboxRadio>
<CheckboxRadio :checked.sync="sharingEnabled" :disabled="true">Enable sharing</CheckboxRadio>
<br>
sharingEnabled: {{ sharingEnabled }}
</div>
</template>
<script>
export default {
data() {
return {
sharingEnabled: false,
}
}
}
</script>
```

### Standard radio set
```vue
<template>
<div>
<CheckboxRadio :checked.sync="sharingPermission" value="r" name="sharing_permission_radio" type="radio">Default permission read</CheckboxRadio>
<CheckboxRadio :checked.sync="sharingPermission" value="rw" name="sharing_permission_radio" type="radio">Default permission read+write</CheckboxRadio>
<br>
sharingPermission: {{ sharingPermission }}
</div>
</template>
<script>
export default {
data() {
return {
sharingPermission: 'r',
}
}
}
</script>
```

### Standard checkbox set
```vue
<template>
<div>
<CheckboxRadio :disabled="true" :checked.sync="sharingPermission" value="r" name="sharing_permission">Permission read</CheckboxRadio>
<CheckboxRadio :checked.sync="sharingPermission" value="w" name="sharing_permission">Permission write</CheckboxRadio>
<CheckboxRadio :checked.sync="sharingPermission" value="d" name="sharing_permission">Permission delete</CheckboxRadio>
<br>
sharingPermission: {{ sharingPermission }}
</div>
</template>
<script>
export default {
data() {
return {
sharingPermission: ['r', 'd'],
}
}
}
</script>
```

</docs>

<template>
<element :is="wrapperElement"
:class="{
'checkbox-radio--checked': isChecked,
'checkbox-radio--disabled': disabled,
'checkbox-radio--indeterminate': indeterminate,
}"
class="checkbox-radio">
<input :id="id"
:checked="isChecked"
:disabled="disabled"
:indeterminate="indeterminate"
:name="name"
:type="type"
:value="value"
class="checkbox-radio__input"
@change="onToggle">

<label :for="id" class="checkbox-radio__label">
<icon :is="checkboxRadioIconElement"
:size="24"
class="checkbox-radio__icon"
title=""
decorative />

<!-- @slot The checkbox/radio label -->
<slot />
</label>
</element>
</template>

<script>
import CheckboxBlankOutline from 'vue-material-design-icons/CheckboxBlankOutline'
import MinusBox from 'vue-material-design-icons/MinusBox'
import CheckboxMarked from 'vue-material-design-icons/CheckboxMarked'
import RadioboxMarked from 'vue-material-design-icons/RadioboxMarked'
import RadioboxBlank from 'vue-material-design-icons/RadioboxBlank'

import GenRandomId from '../../utils/GenRandomId'
import l10n from '../../mixins/l10n'

export const TYPE_CHECKBOX = 'checkbox'
export const TYPE_RADIO = 'radio'

export default {
name: 'CheckboxRadio',

mixins: [l10n],

props: {

/**
* Unique id attribute of the input
*/
id: {
type: String,
default: () => 'checkbox-radio-' + GenRandomId(),
validator: id => id.trim() !== '',
},

/**
* Input name. Required for radio, optional for checkbox
*/
name: {
type: String,
default: null,
},

/**
* Type of the input. Checkbox or radio
*/
type: {
type: String,
default: 'checkbox',
validator: type => type === TYPE_CHECKBOX || type === TYPE_RADIO,
},

/**
* Checked state. To be used with `:value.sync`
*/
checked: {
type: [Boolean, Array, String],
default: false,
},

/**
* Value to be synced on check
*/
value: {
type: String,
default: null,
},

/**
* Disabled state
*/
disabled: {
type: Boolean,
default: false,
},

/**
* Indeterminate state
*/
indeterminate: {
type: Boolean,
default: false,
},

/**
* Wrapping element tag
*/
wrapperElement: {
type: String,
default: 'span',
},
},

computed: {
/**
* Check if that entry is checked
* If value is defined, we use that as the checked value
* If not, we expect true/false in this.checked
* @returns {boolean}
*/
isChecked() {
if (this.value !== null) {
if (Array.isArray(this.checked)) {
return [...this.checked].indexOf(this.value) > -1
}
return this.checked === this.value
}
return this.checked === true
},

/**
* Returns the proper Material icon depending on the select case
* @returns {Component}
*/
checkboxRadioIconElement() {
if (this.type === 'radio') {
if (this.isChecked) {
return RadioboxMarked
}
return RadioboxBlank
}

// Checkbox
if (this.indeterminate) {
return MinusBox
}
if (this.isChecked) {
return CheckboxMarked
}
return CheckboxBlankOutline
},
},

mounted() {
if (this.name && this.type === TYPE_CHECKBOX) {
if (!Array.isArray(this.checked)) {
throw new Error('When using groups of checkboxes, the updated value will be an array')
}
}
},

methods: {
onToggle() {
if (this.disabled) {
return
}

// If this is a radio, there can only be one value
if (this.type === TYPE_RADIO) {
this.$emit('update:checked', this.value)
return
}

// If the initial value was a boolean, let's keep it that way
if (typeof this.checked === 'boolean') {
this.$emit('update:checked', !this.isChecked)
return
}

// Dispatch the checked values as an array if multiple, or single value otherwise
const values = this.getInputsSet()
.filter(input => input.checked)
.map(input => input.value)
this.$emit('update:checked', values)
},

/**
* Get the input set based on this name
* @returns {Node[]}
*/
getInputsSet() {
return [...document.getElementsByName(this.name)]
},
},
}
</script>

<style lang="scss" scoped>
$spacing: 4px;

.checkbox-radio {
display: flex;

&__input {
position: fixed;
z-index: -1;
top: -5000px;
left: -5000px;
opacity: 0;
}

&__label {
display: flex;
align-items: center;
user-select: none;
height: $clickable-area;
border-radius: $clickable-area;
padding: 0 $icon-margin;
margin: 0 #{-$icon-margin};

&, * {
cursor: pointer;
}
}

&__icon {
margin-right: $spacing;
// Remove the left margin of material design icons to align text
margin-left: -2px;
color: var(--color-primary-element);
}

&--disabled &__label {
opacity: $opacity_disabled;
.checkbox-radio__icon {
color: var(--color-text-light)
}
}

&:not(&--disabled) {
.checkbox-radio__input:hover + .checkbox-radio__label,
.checkbox-radio__input:focus + .checkbox-radio__label {
background-color: var(--color-primary-light);
}
}
}

</style>
25 changes: 25 additions & 0 deletions src/components/CheckboxRadio/index.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
/**
* @copyright Copyright (c) 2021 John Molakvoæ <skjnldsv@protonmail.com>
*
* @author John Molakvoæ <skjnldsv@protonmail.com>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/

import CheckboxRadio from './CheckboxRadio'

export default CheckboxRadio
2 changes: 2 additions & 0 deletions src/components/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,7 @@ import AppSidebarTab from './AppSidebarTab'
import Avatar from './Avatar'
import Breadcrumb from './Breadcrumb'
import Breadcrumbs from './Breadcrumbs'
import CheckboxRadio from './CheckboxRadio'
import ColorPicker from './ColorPicker'
import Content from './Content'
import DatetimePicker from './DatetimePicker'
Expand Down Expand Up @@ -96,6 +97,7 @@ export {
Avatar,
Breadcrumb,
Breadcrumbs,
CheckboxRadio,
ColorPicker,
Content,
DatetimePicker,
Expand Down

0 comments on commit 2caf2de

Please sign in to comment.