summaryrefslogtreecommitdiff
path: root/src/web/app/common/views/components/switch.vue
blob: ffbab843eaf7fba7ac9c9985cfe2c22b9fd0e59f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
<template>
<div
	class="mk-switch"
	:class="{ disabled, checked }"
	role="switch"
	:aria-checked="checked"
	:aria-disabled="disabled"
	@click="switchValue"
	@mouseover="mouseenter"
>
	<input
		type="checkbox"
		@change="handleChange"
		ref="input"
		:disabled="disabled"
		@keydown.enter="switchValue"
	>
	<span class="button">
		<span :style="{ transform }"></span>
	</span>
	<span class="label">
		<span :aria-hidden="!checked">{{ text }}</span>
		<p :aria-hidden="!checked">
			<slot></slot>
		</p>
	</span>
</div>
</template>

<script lang="ts">
import Vue from 'vue';
export default Vue.extend({
	props: {
		value: {
			type: Boolean,
			default: false
		},
		disabled: {
			type: Boolean,
			default: false
		},
		text: String
	},/*
	created() {
		if (!~[true, false].indexOf(this.value)) {
			this.$emit('input', false);
		}
	},*/
	computed: {
		checked(): boolean {
			return this.value;
		},
		transform(): string {
			return this.checked ? 'translate3d(20px, 0, 0)' : '';
		}
	},
	watch: {
		value() {
			(this.$refs.input as any).checked = this.checked;
		}
	},
	mounted() {
		(this.$refs.input as any).checked = this.checked;
	},
	methods: {
		mouseenter() {
			(this.$el).style.transition = 'all 0s';
		},
		handleChange() {
			(this.$el).style.transition = 'all 0.3s';
			this.$emit('input', !this.checked);
			this.$emit('change', !this.checked);
			this.$nextTick(() => {
				// set input's checked property
				// in case parent refuses to change component's value
				(this.$refs.input as any).checked = this.checked;
			});
		},
		switchValue() {
			!this.disabled && this.handleChange();
		}
	}
});
</script>

<style lang="stylus" scoped>
@import '~const.styl'

.mk-switch
	display flex
	margin 12px 0
	cursor pointer
	transition all 0.3s

	> *
		user-select none

	&.disabled
		opacity 0.6
		cursor not-allowed

	&.checked
		> .button
			background-color $theme-color
			border-color $theme-color

		> .label
			> span
				color $theme-color

		&:hover
			> .label
				> span
					color darken($theme-color, 10%)

			> .button
				background darken($theme-color, 10%)
				border-color darken($theme-color, 10%)

	&:hover
		> .label
			> span
				color #2e3338

		> .button
			background #ced2da
			border-color #ced2da

	> input
		position absolute
		width 0
		height 0
		opacity 0
		margin 0

	> .button
		display inline-block
		margin 0
		width 40px
		min-width 40px
		height 20px
		min-height 20px
		background #dcdfe6
		border 1px solid #dcdfe6
		outline none
		border-radius 10px
		transition inherit

		> *
			position absolute
			top 1px
			left 1px
			border-radius 100%
			transition transform 0.3s
			width 16px
			height 16px
			background-color #fff

	> .label
		margin-left 8px
		display block
		font-size 15px
		cursor pointer
		transition inherit

		> span
			display block
			line-height 20px
			color #4a535a
			transition inherit

		> p
			margin 0
			//font-size 90%
			color #9daab3

</style>