Commit e7893a68ede3fef83accc70beabcd1ca6f4e9eff

Authored by 梁灏
1 parent dab39476

update ColorPicker

.eslintignore
1   -src/directives
2 1 \ No newline at end of file
  2 +src/directives
  3 +src/utils/throttle.js
3 4 \ No newline at end of file
... ...
... ... @@ -151,4 +151,26 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
151 151 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
152 152 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
153 153 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
154   -THE SOFTWARE.
155 154 \ No newline at end of file
  155 +THE SOFTWARE.
  156 +
  157 +The MIT License (MIT)
  158 +
  159 +Copyright (c) 2015 greyby
  160 +
  161 +Permission is hereby granted, free of charge, to any person obtaining a copy
  162 +of this software and associated documentation files (the "Software"), to deal
  163 +in the Software without restriction, including without limitation the rights
  164 +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  165 +copies of the Software, and to permit persons to whom the Software is
  166 +furnished to do so, subject to the following conditions:
  167 +
  168 +The above copyright notice and this permission notice shall be included in all
  169 +copies or substantial portions of the Software.
  170 +
  171 +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  172 +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  173 +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  174 +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  175 +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  176 +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  177 +SOFTWARE.
... ...
examples/routers/color-picker.vue
1 1 <template>
2 2 <div style="margin: 100px;">
  3 + {{ color.rgba }}
3 4 <!--<Input placeholder="请输入..." size="large" style="width: 50px;"></Input>-->
4 5 <!--<color-picker placement="bottom-start" size="large"></color-picker>-->
5 6 <!--<Date-picker type="date" placeholder="选择日期" size="large" style="width: 200px"></Date-picker>-->
6   - <color-picker recommend alpha placement="bottom" size="default"></color-picker>
  7 + <color-picker v-model="color" alpha :recommend="true" placement="bottom" size="default"></color-picker>
  8 + <color-picker v-model="color" :alpha="false" :recommend="false" placement="bottom" size="default"></color-picker>
7 9 <!--<Date-picker type="date" placeholder="选择日期" style="width: 200px"></Date-picker>-->
8 10 <!--<color-picker placement="bottom-start" size="small"></color-picker>-->
9 11 <!--<Date-picker type="date" placeholder="选择日期" size="small" style="width: 200px"></Date-picker>-->
... ... @@ -13,7 +15,30 @@
13 15 export default {
14 16 props: {},
15 17 data () {
16   - return {};
  18 + return {
  19 + color: {
  20 + hex: '#194d33',
  21 + hsl: {
  22 + h: 150,
  23 + s: 0.5,
  24 + l: 0.2,
  25 + a: 1
  26 + },
  27 + hsv: {
  28 + h: 150,
  29 + s: 0.66,
  30 + v: 0.30,
  31 + a: 1
  32 + },
  33 + rgba: {
  34 + r: 25,
  35 + g: 77,
  36 + b: 51,
  37 + a: 1
  38 + },
  39 + a: 1
  40 + }
  41 + };
17 42 },
18 43 computed: {},
19 44 methods: {}
... ...
package-lock.json
1 1 {
2 2 "name": "iview",
3   - "version": "2.0.0-rc.18",
  3 + "version": "2.1.0",
4 4 "lockfileVersion": 1,
5 5 "requires": true,
6 6 "dependencies": {
... ... @@ -15043,6 +15043,11 @@
15043 15043 "setimmediate": "1.0.5"
15044 15044 }
15045 15045 },
  15046 + "tinycolor2": {
  15047 + "version": "1.4.1",
  15048 + "resolved": "https://registry.npmjs.org/tinycolor2/-/tinycolor2-1.4.1.tgz",
  15049 + "integrity": "sha1-9PrTM0R7wLB9TcjpIJ2POaisd+g="
  15050 + },
15046 15051 "tmp": {
15047 15052 "version": "0.0.31",
15048 15053 "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.31.tgz",
... ...
package.json
... ... @@ -43,7 +43,8 @@
43 43 "async-validator": "^1.7.1",
44 44 "core-js": "^2.4.1",
45 45 "deepmerge": "^1.5.0",
46   - "popper.js": "^0.6.4"
  46 + "popper.js": "^0.6.4",
  47 + "tinycolor2": "^1.4.1"
47 48 },
48 49 "peerDependencies": {
49 50 "vue": "^2.4.1"
... ...
src/components/color-picker/alpha.vue 0 → 100644
  1 +<template>
  2 + <div class="ivu-color-picker-alpha">
  3 + <div class="ivu-color-picker-alpha-checkboard-wrap">
  4 + <div class="ivu-color-picker-alpha-checkerboard"></div>
  5 + </div>
  6 + <div class="ivu-color-picker-alpha-gradient" :style="{background: gradientColor}"></div>
  7 + <div class="ivu-color-picker-alpha-container" ref="container"
  8 + @mousedown="handleMouseDown"
  9 + @touchmove="handleChange"
  10 + @touchstart="handleChange">
  11 + <div class="ivu-color-picker-alpha-pointer" :style="{left: colors.a * 100 + '%'}">
  12 + <div class="ivu-color-picker-alpha-picker"></div>
  13 + </div>
  14 + </div>
  15 + </div>
  16 +</template>
  17 +<script>
  18 + export default {
  19 + name: 'Alpha',
  20 + props: {
  21 + value: Object,
  22 + onChange: Function
  23 + },
  24 + computed: {
  25 + colors () {
  26 + return this.value;
  27 + },
  28 + gradientColor () {
  29 + const rgba = this.colors.rgba;
  30 + const rgbStr = [rgba.r, rgba.g, rgba.b].join(',');
  31 + return 'linear-gradient(to right, rgba(' + rgbStr + ', 0) 0%, rgba(' + rgbStr + ', 1) 100%)';
  32 + }
  33 + },
  34 + methods: {
  35 + handleChange (e, skip) {
  36 + !skip && e.preventDefault();
  37 + const container = this.$refs.container;
  38 + const containerWidth = container.clientWidth;
  39 +
  40 + const xOffset = container.getBoundingClientRect().left + window.pageXOffset;
  41 + const pageX = e.pageX || (e.touches ? e.touches[0].pageX : 0);
  42 + const left = pageX - xOffset;
  43 +
  44 + let a;
  45 + if (left < 0) {
  46 + a = 0;
  47 + } else if (left > containerWidth) {
  48 + a = 1;
  49 + } else {
  50 + a = Math.round(left * 100 / containerWidth) / 100;
  51 + }
  52 +
  53 + if (this.colors.a !== a) {
  54 + this.$emit('change', {
  55 + h: this.colors.hsl.h,
  56 + s: this.colors.hsl.s,
  57 + l: this.colors.hsl.l,
  58 + a: a,
  59 + source: 'rgba'
  60 + });
  61 + }
  62 + },
  63 + handleMouseDown (e) {
  64 + this.handleChange(e, true);
  65 + window.addEventListener('mousemove', this.handleChange);
  66 + window.addEventListener('mouseup', this.handleMouseUp);
  67 + },
  68 + handleMouseUp () {
  69 + this.unbindEventListeners();
  70 + },
  71 + unbindEventListeners () {
  72 + window.removeEventListener('mousemove', this.handleChange);
  73 + window.removeEventListener('mouseup', this.handleMouseUp);
  74 + }
  75 + }
  76 + };
  77 +</script>
0 78 \ No newline at end of file
... ...
src/components/color-picker/color-picker.vue
... ... @@ -8,12 +8,14 @@
8 8 </div>
9 9 <Dropdown-menu slot="list">
10 10 <div :class="[prefixCls + '-picker']">
11   - <div :class="[prefixCls + '-picker-panel']"></div>
  11 + <div :class="[prefixCls + '-picker-panel']">
  12 + <Saturation v-model="saturationColors" @change="childChange"></Saturation>
  13 + </div>
12 14 <div :class="[prefixCls + '-picker-hue-slider']">
13   - <Slider v-model="hueNumber" :min="0" :max="255"></Slider>
  15 + <Hue v-model="saturationColors" @change="childChange"></Hue>
14 16 </div>
15 17 <div v-if="alpha" :class="[prefixCls + '-picker-alpha-slider']">
16   - <Slider v-model="alphaNumber" :min="0" :max="100"></Slider>
  18 + <Alpha v-model="saturationColors" @change="childChange"></Alpha>
17 19 </div>
18 20 <recommend-colors v-if="colors.length" :list="colors" :class="[prefixCls + '-picker-colors']"></recommend-colors>
19 21 <recommend-colors v-if="!colors.length && recommend" :list="recommendedColor" :class="[prefixCls + '-picker-colors']"></recommend-colors>
... ... @@ -23,22 +25,76 @@
23 25 </Dropdown>
24 26 </template>
25 27 <script>
  28 + import tinycolor from 'tinycolor2';
  29 +
26 30 import Dropdown from '../dropdown/dropdown.vue';
27 31 import DropdownMenu from '../dropdown/dropdown-menu.vue';
28   - import Slider from '../slider/slider.vue';
29 32 import RecommendColors from './recommend-colors.vue';
30 33 import Confirm from '../date-picker/base/confirm.vue';
  34 +
  35 + import Saturation from './saturation.vue';
  36 + import Hue from './hue.vue';
  37 + import Alpha from './alpha.vue';
  38 +
31 39 import { oneOf } from '../../utils/assist';
32 40  
33 41 const prefixCls = 'ivu-color-picker';
34 42 const inputPrefixCls = 'ivu-input';
35 43  
  44 + function _colorChange (data, oldHue) {
  45 + const alpha = data && data.a;
  46 + let color;
  47 +
  48 + // hsl is better than hex between conversions
  49 + if (data && data.hsl) {
  50 + color = tinycolor(data.hsl);
  51 + } else if (data && data.hex && data.hex.length > 0) {
  52 + color = tinycolor(data.hex);
  53 + } else {
  54 + color = tinycolor(data);
  55 + }
  56 +
  57 + if (color && (color._a === undefined || color._a === null)) {
  58 + color.setAlpha(alpha || 1);
  59 + }
  60 +
  61 + const hsl = color.toHsl();
  62 + const hsv = color.toHsv();
  63 +
  64 + if (hsl.s === 0) {
  65 + hsv.h = hsl.h = data.h || (data.hsl && data.hsl.h) || oldHue || 0;
  66 + }
  67 +
  68 + // when the hsv.v is less than 0.0164 (base on test)
  69 + // because of possible loss of precision
  70 + // the result of hue and saturation would be miscalculated
  71 + if (hsv.v < 0.0164) {
  72 + hsv.h = data.h || (data.hsv && data.hsv.h) || 0;
  73 + hsv.s = data.s || (data.hsv && data.hsv.s) || 0;
  74 + }
  75 +
  76 + if (hsl.l < 0.01) {
  77 + hsl.h = data.h || (data.hsl && data.hsl.h) || 0;
  78 + hsl.s = data.s || (data.hsl && data.hsl.s) || 0;
  79 + }
  80 +
  81 + return {
  82 + hsl: hsl,
  83 + hex: color.toHexString().toUpperCase(),
  84 + rgba: color.toRgb(),
  85 + hsv: hsv,
  86 + oldHue: data.h || oldHue || hsl.h,
  87 + source: data.source,
  88 + a: data.a || color.getAlpha()
  89 + };
  90 + }
  91 +
36 92 export default {
37 93 name: 'ColorPicker',
38   - components: { Dropdown, DropdownMenu, Slider, Confirm, RecommendColors },
  94 + components: { Dropdown, DropdownMenu, Confirm, RecommendColors, Saturation, Hue, Alpha },
39 95 props: {
40 96 value: {
41   - type: String
  97 + type: Object
42 98 },
43 99 alpha: {
44 100 type: Boolean,
... ... @@ -81,10 +137,8 @@
81 137 },
82 138 data () {
83 139 return {
  140 + val: _colorChange(this.value),
84 141 prefixCls: prefixCls,
85   - currentValue: this.value,
86   - hueNumber: 0,
87   - alphaNumber: 0,
88 142 recommendedColor: [
89 143 '#2d8cf0',
90 144 '#19be6b',
... ... @@ -110,6 +164,15 @@
110 164 };
111 165 },
112 166 computed: {
  167 + saturationColors: {
  168 + get () {
  169 + return this.val;
  170 + },
  171 + set (newVal) {
  172 + this.val = newVal;
  173 + this.$emit('input', newVal);
  174 + }
  175 + },
113 176 wrapClasses () {
114 177 return [
115 178 `${prefixCls}-rel`,
... ... @@ -129,8 +192,41 @@
129 192 ];
130 193 }
131 194 },
  195 + watch: {
  196 + value (newVal) {
  197 + this.val = _colorChange(newVal);
  198 + }
  199 + },
132 200 methods: {
  201 + childChange (data) {
  202 + this.colorChange(data);
  203 + },
  204 + colorChange (data, oldHue) {
  205 + this.oldHue = this.saturationColors.hsl.h;
  206 + this.saturationColors = _colorChange(data, oldHue || this.oldHue);
  207 + },
  208 + isValidHex (hex) {
  209 + return tinycolor(hex).isValid();
  210 + },
  211 + simpleCheckForValidColor (data) {
  212 + const keysToCheck = ['r', 'g', 'b', 'a', 'h', 's', 'l', 'v'];
  213 + let checked = 0;
  214 + let passed = 0;
133 215  
  216 + for (let i = 0; i < keysToCheck.length; i++) {
  217 + const letter = keysToCheck[i];
  218 + if (data[letter]) {
  219 + checked++;
  220 + if (!isNaN(data[letter])) {
  221 + passed++;
  222 + }
  223 + }
  224 + }
  225 +
  226 + if (checked === passed) {
  227 + return data;
  228 + }
  229 + }
134 230 }
135 231 };
136 232 </script>
137 233 \ No newline at end of file
... ...
src/components/color-picker/color.js deleted
src/components/color-picker/hue.vue 0 → 100644
  1 +<template>
  2 + <div class="ivu-color-picker-hue">
  3 + <div class="ivu-color-picker-hue-container" ref="container"
  4 + @mousedown="handleMouseDown"
  5 + @touchmove="handleChange"
  6 + @touchstart="handleChange">
  7 + <div class="ivu-color-picker-hue-pointer" :style="{top: 0, left: pointerLeft}">
  8 + <div class="ivu-color-picker-hue-picker"></div>
  9 + </div>
  10 + </div>
  11 + </div>
  12 +</template>
  13 +<script>
  14 + export default {
  15 + name: 'Hue',
  16 + props: {
  17 + value: Object
  18 + },
  19 + data () {
  20 + return {
  21 + oldHue: 0,
  22 + pullDirection: ''
  23 + };
  24 + },
  25 + computed: {
  26 + colors () {
  27 + const h = this.value.hsl.h;
  28 + if (h !== 0 && h - this.oldHue > 0) this.pullDirection = 'right';
  29 + if (h !== 0 && h - this.oldHue < 0) this.pullDirection = 'left';
  30 + this.oldHue = h;
  31 +
  32 + return this.value;
  33 + },
  34 + pointerLeft () {
  35 + if (this.colors.hsl.h === 0 && this.pullDirection === 'right') return '100%';
  36 + return (this.colors.hsl.h * 100) / 360 + '%';
  37 + }
  38 + },
  39 + methods: {
  40 + handleChange (e, skip) {
  41 + !skip && e.preventDefault();
  42 +
  43 + const container = this.$refs.container;
  44 + const containerWidth = container.clientWidth;
  45 +
  46 + const xOffset = container.getBoundingClientRect().left + window.pageXOffset;
  47 + const pageX = e.pageX || (e.touches ? e.touches[0].pageX : 0);
  48 + const left = pageX - xOffset;
  49 +
  50 + let h;
  51 + let percent;
  52 +
  53 + if (left < 0) {
  54 + h = 0;
  55 + } else if (left > containerWidth) {
  56 + h = 360;
  57 + } else {
  58 + percent = left * 100 / containerWidth;
  59 + h = (360 * percent / 100);
  60 + }
  61 +
  62 + if (this.colors.hsl.h !== h) {
  63 + this.$emit('change', {
  64 + h: h,
  65 + s: this.colors.hsl.s,
  66 + l: this.colors.hsl.l,
  67 + a: this.colors.hsl.a,
  68 + source: 'hsl'
  69 + });
  70 + }
  71 + },
  72 + handleMouseDown (e) {
  73 + this.handleChange(e, true);
  74 + window.addEventListener('mousemove', this.handleChange);
  75 + window.addEventListener('mouseup', this.handleMouseUp);
  76 + },
  77 + handleMouseUp () {
  78 + this.unbindEventListeners();
  79 + },
  80 + unbindEventListeners () {
  81 + window.removeEventListener('mousemove', this.handleChange);
  82 + window.removeEventListener('mouseup', this.handleMouseUp);
  83 + }
  84 + }
  85 + };
  86 +</script>
0 87 \ No newline at end of file
... ...
src/components/color-picker/saturation.vue 0 → 100644
  1 +<template>
  2 + <div class="ivu-color-picker-saturation-wrapper">
  3 + <div
  4 + class="ivu-color-picker-saturation"
  5 + :style="{background: bgColor}"
  6 + ref="container"
  7 + @mousedown="handleMouseDown">
  8 + <div class="ivu-color-picker-saturation--white"></div>
  9 + <div class="ivu-color-picker-saturation--black"></div>
  10 + <div class="ivu-color-picker-saturation-pointer" :style="{top: pointerTop, left: pointerLeft}">
  11 + <div class="ivu-color-picker-saturation-circle"></div>
  12 + </div>
  13 + </div>
  14 + </div>
  15 +</template>
  16 +<script>
  17 + import throttle from '../../utils/throttle';
  18 +
  19 + export default {
  20 + name: 'Saturation',
  21 + props: {
  22 + value: Object
  23 + },
  24 + data () {
  25 + return {};
  26 + },
  27 + computed: {
  28 + colors () {
  29 + return this.value;
  30 + },
  31 + bgColor () {
  32 + return `hsl(${this.colors.hsv.h}, 100%, 50%)`;
  33 + },
  34 + pointerTop () {
  35 + return (-(this.colors.hsv.v * 100) + 1) + 100 + '%';
  36 + },
  37 + pointerLeft () {
  38 + return this.colors.hsv.s * 100 + '%';
  39 + }
  40 + },
  41 + methods: {
  42 + throttle: throttle((fn, data) => {fn(data);}, 20,
  43 + {
  44 + 'leading': true,
  45 + 'trailing': false
  46 + }),
  47 + handleChange (e, skip) {
  48 + !skip && e.preventDefault();
  49 + const container = this.$refs.container;
  50 + const containerWidth = container.clientWidth;
  51 + const containerHeight = container.clientHeight;
  52 + const xOffset = container.getBoundingClientRect().left + window.pageXOffset;
  53 + const yOffset = container.getBoundingClientRect().top + window.pageYOffset;
  54 + const pageX = e.pageX || (e.touches ? e.touches[0].pageX : 0);
  55 + const pageY = e.pageY || (e.touches ? e.touches[0].pageY : 0);
  56 + let left = pageX - xOffset;
  57 + let top = pageY - yOffset;
  58 + if (left < 0) {
  59 + left = 0;
  60 + } else if (left > containerWidth) {
  61 + left = containerWidth;
  62 + } else if (top < 0) {
  63 + top = 0;
  64 + } else if (top > containerHeight) {
  65 + top = containerHeight;
  66 + }
  67 + const saturation = left / containerWidth;
  68 + let bright = -(top / containerHeight) + 1;
  69 + bright = bright > 0 ? bright : 0;
  70 + bright = bright > 1 ? 1 : bright;
  71 + this.throttle(this.onChange, {
  72 + h: this.colors.hsv.h,
  73 + s: saturation,
  74 + v: bright,
  75 + a: this.colors.hsv.a,
  76 + source: 'hsva'
  77 + });
  78 + },
  79 + onChange (param) {
  80 + this.$emit('change', param);
  81 + },
  82 + handleMouseDown () {
  83 + // this.handleChange(e, true)
  84 + window.addEventListener('mousemove', this.handleChange);
  85 + window.addEventListener('mouseup', this.handleChange);
  86 + window.addEventListener('mouseup', this.handleMouseUp);
  87 + },
  88 + handleMouseUp () {
  89 + this.unbindEventListeners();
  90 + },
  91 + unbindEventListeners () {
  92 + window.removeEventListener('mousemove', this.handleChange);
  93 + window.removeEventListener('mouseup', this.handleChange);
  94 + window.removeEventListener('mouseup', this.handleMouseUp);
  95 + }
  96 + }
  97 + };
  98 +</script>
0 99 \ No newline at end of file
... ...
src/styles/components/color-picker.less
... ... @@ -24,30 +24,31 @@
24 24 }
25 25  
26 26 &-picker{
27   - padding: 8px 8px 0;
  27 + padding: 4px 8px 0;
28 28 &-panel{
29 29 width: 200px;
30   - height: 200px;
31 30 margin: 0 auto;
32   - background: #47cb89;
33   - border-radius: 50%;
34   - }
35   - &-hue-slider{
36   -
  31 + box-sizing: initial;
  32 + position: relative;
37 33 }
38   - &-alpha-slider{
39   -
  34 + &-hue-slider, &-alpha-slider{
  35 + height: 10px;
  36 + margin-top: 8px;
  37 + position: relative;
40 38 }
41 39 &-colors{
42 40 margin-top: 8px;
  41 + overflow: hidden;
43 42 span{
44 43 display: inline-block;
45   - width: 18px;
46   - height: 18px;
  44 + width: 20px;
  45 + height: 20px;
  46 + float: left;
47 47 em{
48 48 display: block;
49 49 width: 16px;
50 50 height: 16px;
  51 + margin: 2px;
51 52 cursor: pointer;
52 53 border-radius: 2px;
53 54 box-shadow: inset 0 0 0 1px rgba(0,0,0,.15);
... ... @@ -58,4 +59,121 @@
58 59 margin-top: 8px;
59 60 }
60 61 }
  62 +
  63 + &-saturation{
  64 + &-wrapper{
  65 + width: 100%;
  66 + padding-bottom: 75%;
  67 + position: relative;
  68 + overflow: hidden;
  69 + }
  70 + &, &--white, &--black{
  71 + cursor: pointer;
  72 + position: absolute;
  73 + top: 0;
  74 + left: 0;
  75 + right: 0;
  76 + bottom: 0;
  77 + }
  78 + &--white{
  79 + background: linear-gradient(to right, #fff, rgba(255,255,255,0));
  80 + }
  81 + &--black{
  82 + background: linear-gradient(to top, #000, rgba(0,0,0,0));
  83 + }
  84 + &-pointer{
  85 + cursor: pointer;
  86 + position: absolute;
  87 + }
  88 + &-circle{
  89 + cursor: head;
  90 + width: 4px;
  91 + height: 4px;
  92 + box-shadow: 0 0 0 1.5px #fff, inset 0 0 1px 1px rgba(0,0,0,.3), 0 0 1px 2px rgba(0,0,0,.4);
  93 + border-radius: 50%;
  94 + transform: translate(-2px, -2px);
  95 + }
  96 + }
  97 +
  98 + &-hue{
  99 + position: absolute;
  100 + top: 0;
  101 + right: 0;
  102 + bottom: 0;
  103 + left: 0;
  104 + border-radius: 2px;
  105 + background: linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%);
  106 + &-container{
  107 + cursor: pointer;
  108 + margin: 0 2px;
  109 + position: relative;
  110 + height: 100%;
  111 + }
  112 + &-pointer{
  113 + z-index: 2;
  114 + position: absolute;
  115 + }
  116 + &-picker{
  117 + cursor: pointer;
  118 + margin-top: 1px;
  119 + width: 4px;
  120 + border-radius: 1px;
  121 + height: 8px;
  122 + box-shadow: 0 0 2px rgba(0, 0, 0, .6);
  123 + background: #fff;
  124 + transform: translateX(-2px);
  125 + }
  126 + }
  127 +
  128 + &-alpha{
  129 + position: absolute;
  130 + top: 0;
  131 + right: 0;
  132 + bottom: 0;
  133 + left: 0;
  134 + &-checkboard-wrap{
  135 + position: absolute;
  136 + top: 0;
  137 + right: 0;
  138 + bottom: 0;
  139 + left: 0;
  140 + overflow: hidden;
  141 + }
  142 + &-checkerboard{
  143 + position: absolute;
  144 + top: 0;
  145 + right: 0;
  146 + bottom: 0;
  147 + left: 0;
  148 + background: url();
  149 + }
  150 + &-gradient{
  151 + position: absolute;
  152 + top: 0;
  153 + right: 0;
  154 + bottom: 0;
  155 + left: 0;
  156 + }
  157 + &-container{
  158 + cursor: pointer;
  159 + position: relative;
  160 + z-index: 2;
  161 + height: 100%;
  162 + margin: 0 3px;
  163 + }
  164 + &-pointer{
  165 + z-index: 2;
  166 + position: absolute;
  167 + }
  168 + &-picker{
  169 + cursor: pointer;
  170 + width: 4px;
  171 + border-radius: 1px;
  172 + height: 8px;
  173 + box-shadow: 0 0 2px rgba(0, 0, 0, .6);
  174 + background: #fff;
  175 + margin-top: 1px;
  176 + transform: translateX(-2px);
  177 + }
  178 + }
61 179 }
62 180 \ No newline at end of file
... ...
src/utils/throttle.js 0 → 100644
  1 +/**
  2 + * lodash (Custom Build) <https://lodash.com/>
  3 + * Build: `lodash modularize exports="npm" -o ./`
  4 + * Copyright jQuery Foundation and other contributors <https://jquery.org/>
  5 + * Released under MIT license <https://lodash.com/license>
  6 + * Based on Underscore.js 1.8.3 <http://underscorejs.org/LICENSE>
  7 + * Copyright Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
  8 + */
  9 +
  10 +/** Used as the `TypeError` message for "Functions" methods. */
  11 +var FUNC_ERROR_TEXT = 'Expected a function';
  12 +
  13 +/** Used as references for various `Number` constants. */
  14 +var NAN = 0 / 0;
  15 +
  16 +/** `Object#toString` result references. */
  17 +var symbolTag = '[object Symbol]';
  18 +
  19 +/** Used to match leading and trailing whitespace. */
  20 +var reTrim = /^\s+|\s+$/g;
  21 +
  22 +/** Used to detect bad signed hexadecimal string values. */
  23 +var reIsBadHex = /^[-+]0x[0-9a-f]+$/i;
  24 +
  25 +/** Used to detect binary string values. */
  26 +var reIsBinary = /^0b[01]+$/i;
  27 +
  28 +/** Used to detect octal string values. */
  29 +var reIsOctal = /^0o[0-7]+$/i;
  30 +
  31 +/** Built-in method references without a dependency on `root`. */
  32 +var freeParseInt = parseInt;
  33 +
  34 +/** Detect free variable `global` from Node.js. */
  35 +var freeGlobal = typeof global == 'object' && global && global.Object === Object && global;
  36 +
  37 +/** Detect free variable `self`. */
  38 +var freeSelf = typeof self == 'object' && self && self.Object === Object && self;
  39 +
  40 +/** Used as a reference to the global object. */
  41 +var root = freeGlobal || freeSelf || Function('return this')();
  42 +
  43 +/** Used for built-in method references. */
  44 +var objectProto = Object.prototype;
  45 +
  46 +/**
  47 + * Used to resolve the
  48 + * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)
  49 + * of values.
  50 + */
  51 +var objectToString = objectProto.toString;
  52 +
  53 +/* Built-in method references for those with the same name as other `lodash` methods. */
  54 +var nativeMax = Math.max,
  55 + nativeMin = Math.min;
  56 +
  57 +/**
  58 + * Gets the timestamp of the number of milliseconds that have elapsed since
  59 + * the Unix epoch (1 January 1970 00:00:00 UTC).
  60 + *
  61 + * @static
  62 + * @memberOf _
  63 + * @since 2.4.0
  64 + * @category Date
  65 + * @returns {number} Returns the timestamp.
  66 + * @example
  67 + *
  68 + * _.defer(function(stamp) {
  69 + * console.log(_.now() - stamp);
  70 + * }, _.now());
  71 + * // => Logs the number of milliseconds it took for the deferred invocation.
  72 + */
  73 +var now = function() {
  74 + return root.Date.now();
  75 +};
  76 +
  77 +/**
  78 + * Creates a debounced function that delays invoking `func` until after `wait`
  79 + * milliseconds have elapsed since the last time the debounced function was
  80 + * invoked. The debounced function comes with a `cancel` method to cancel
  81 + * delayed `func` invocations and a `flush` method to immediately invoke them.
  82 + * Provide `options` to indicate whether `func` should be invoked on the
  83 + * leading and/or trailing edge of the `wait` timeout. The `func` is invoked
  84 + * with the last arguments provided to the debounced function. Subsequent
  85 + * calls to the debounced function return the result of the last `func`
  86 + * invocation.
  87 + *
  88 + * **Note:** If `leading` and `trailing` options are `true`, `func` is
  89 + * invoked on the trailing edge of the timeout only if the debounced function
  90 + * is invoked more than once during the `wait` timeout.
  91 + *
  92 + * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
  93 + * until to the next tick, similar to `setTimeout` with a timeout of `0`.
  94 + *
  95 + * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
  96 + * for details over the differences between `_.debounce` and `_.throttle`.
  97 + *
  98 + * @static
  99 + * @memberOf _
  100 + * @since 0.1.0
  101 + * @category Function
  102 + * @param {Function} func The function to debounce.
  103 + * @param {number} [wait=0] The number of milliseconds to delay.
  104 + * @param {Object} [options={}] The options object.
  105 + * @param {boolean} [options.leading=false]
  106 + * Specify invoking on the leading edge of the timeout.
  107 + * @param {number} [options.maxWait]
  108 + * The maximum time `func` is allowed to be delayed before it's invoked.
  109 + * @param {boolean} [options.trailing=true]
  110 + * Specify invoking on the trailing edge of the timeout.
  111 + * @returns {Function} Returns the new debounced function.
  112 + * @example
  113 + *
  114 + * // Avoid costly calculations while the window size is in flux.
  115 + * jQuery(window).on('resize', _.debounce(calculateLayout, 150));
  116 + *
  117 + * // Invoke `sendMail` when clicked, debouncing subsequent calls.
  118 + * jQuery(element).on('click', _.debounce(sendMail, 300, {
  119 + * 'leading': true,
  120 + * 'trailing': false
  121 + * }));
  122 + *
  123 + * // Ensure `batchLog` is invoked once after 1 second of debounced calls.
  124 + * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });
  125 + * var source = new EventSource('/stream');
  126 + * jQuery(source).on('message', debounced);
  127 + *
  128 + * // Cancel the trailing debounced invocation.
  129 + * jQuery(window).on('popstate', debounced.cancel);
  130 + */
  131 +function debounce(func, wait, options) {
  132 + var lastArgs,
  133 + lastThis,
  134 + maxWait,
  135 + result,
  136 + timerId,
  137 + lastCallTime,
  138 + lastInvokeTime = 0,
  139 + leading = false,
  140 + maxing = false,
  141 + trailing = true;
  142 +
  143 + if (typeof func != 'function') {
  144 + throw new TypeError(FUNC_ERROR_TEXT);
  145 + }
  146 + wait = toNumber(wait) || 0;
  147 + if (isObject(options)) {
  148 + leading = !!options.leading;
  149 + maxing = 'maxWait' in options;
  150 + maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;
  151 + trailing = 'trailing' in options ? !!options.trailing : trailing;
  152 + }
  153 +
  154 + function invokeFunc(time) {
  155 + var args = lastArgs,
  156 + thisArg = lastThis;
  157 +
  158 + lastArgs = lastThis = undefined;
  159 + lastInvokeTime = time;
  160 + result = func.apply(thisArg, args);
  161 + return result;
  162 + }
  163 +
  164 + function leadingEdge(time) {
  165 + // Reset any `maxWait` timer.
  166 + lastInvokeTime = time;
  167 + // Start the timer for the trailing edge.
  168 + timerId = setTimeout(timerExpired, wait);
  169 + // Invoke the leading edge.
  170 + return leading ? invokeFunc(time) : result;
  171 + }
  172 +
  173 + function remainingWait(time) {
  174 + var timeSinceLastCall = time - lastCallTime,
  175 + timeSinceLastInvoke = time - lastInvokeTime,
  176 + result = wait - timeSinceLastCall;
  177 +
  178 + return maxing ? nativeMin(result, maxWait - timeSinceLastInvoke) : result;
  179 + }
  180 +
  181 + function shouldInvoke(time) {
  182 + var timeSinceLastCall = time - lastCallTime,
  183 + timeSinceLastInvoke = time - lastInvokeTime;
  184 +
  185 + // Either this is the first call, activity has stopped and we're at the
  186 + // trailing edge, the system time has gone backwards and we're treating
  187 + // it as the trailing edge, or we've hit the `maxWait` limit.
  188 + return (lastCallTime === undefined || (timeSinceLastCall >= wait) ||
  189 + (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait));
  190 + }
  191 +
  192 + function timerExpired() {
  193 + var time = now();
  194 + if (shouldInvoke(time)) {
  195 + return trailingEdge(time);
  196 + }
  197 + // Restart the timer.
  198 + timerId = setTimeout(timerExpired, remainingWait(time));
  199 + }
  200 +
  201 + function trailingEdge(time) {
  202 + timerId = undefined;
  203 +
  204 + // Only invoke if we have `lastArgs` which means `func` has been
  205 + // debounced at least once.
  206 + if (trailing && lastArgs) {
  207 + return invokeFunc(time);
  208 + }
  209 + lastArgs = lastThis = undefined;
  210 + return result;
  211 + }
  212 +
  213 + function cancel() {
  214 + if (timerId !== undefined) {
  215 + clearTimeout(timerId);
  216 + }
  217 + lastInvokeTime = 0;
  218 + lastArgs = lastCallTime = lastThis = timerId = undefined;
  219 + }
  220 +
  221 + function flush() {
  222 + return timerId === undefined ? result : trailingEdge(now());
  223 + }
  224 +
  225 + function debounced() {
  226 + var time = now(),
  227 + isInvoking = shouldInvoke(time);
  228 +
  229 + lastArgs = arguments;
  230 + lastThis = this;
  231 + lastCallTime = time;
  232 +
  233 + if (isInvoking) {
  234 + if (timerId === undefined) {
  235 + return leadingEdge(lastCallTime);
  236 + }
  237 + if (maxing) {
  238 + // Handle invocations in a tight loop.
  239 + timerId = setTimeout(timerExpired, wait);
  240 + return invokeFunc(lastCallTime);
  241 + }
  242 + }
  243 + if (timerId === undefined) {
  244 + timerId = setTimeout(timerExpired, wait);
  245 + }
  246 + return result;
  247 + }
  248 + debounced.cancel = cancel;
  249 + debounced.flush = flush;
  250 + return debounced;
  251 +}
  252 +
  253 +/**
  254 + * Creates a throttled function that only invokes `func` at most once per
  255 + * every `wait` milliseconds. The throttled function comes with a `cancel`
  256 + * method to cancel delayed `func` invocations and a `flush` method to
  257 + * immediately invoke them. Provide `options` to indicate whether `func`
  258 + * should be invoked on the leading and/or trailing edge of the `wait`
  259 + * timeout. The `func` is invoked with the last arguments provided to the
  260 + * throttled function. Subsequent calls to the throttled function return the
  261 + * result of the last `func` invocation.
  262 + *
  263 + * **Note:** If `leading` and `trailing` options are `true`, `func` is
  264 + * invoked on the trailing edge of the timeout only if the throttled function
  265 + * is invoked more than once during the `wait` timeout.
  266 + *
  267 + * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
  268 + * until to the next tick, similar to `setTimeout` with a timeout of `0`.
  269 + *
  270 + * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
  271 + * for details over the differences between `_.throttle` and `_.debounce`.
  272 + *
  273 + * @static
  274 + * @memberOf _
  275 + * @since 0.1.0
  276 + * @category Function
  277 + * @param {Function} func The function to throttle.
  278 + * @param {number} [wait=0] The number of milliseconds to throttle invocations to.
  279 + * @param {Object} [options={}] The options object.
  280 + * @param {boolean} [options.leading=true]
  281 + * Specify invoking on the leading edge of the timeout.
  282 + * @param {boolean} [options.trailing=true]
  283 + * Specify invoking on the trailing edge of the timeout.
  284 + * @returns {Function} Returns the new throttled function.
  285 + * @example
  286 + *
  287 + * // Avoid excessively updating the position while scrolling.
  288 + * jQuery(window).on('scroll', _.throttle(updatePosition, 100));
  289 + *
  290 + * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.
  291 + * var throttled = _.throttle(renewToken, 300000, { 'trailing': false });
  292 + * jQuery(element).on('click', throttled);
  293 + *
  294 + * // Cancel the trailing throttled invocation.
  295 + * jQuery(window).on('popstate', throttled.cancel);
  296 + */
  297 +function throttle(func, wait, options) {
  298 + var leading = true,
  299 + trailing = true;
  300 +
  301 + if (typeof func != 'function') {
  302 + throw new TypeError(FUNC_ERROR_TEXT);
  303 + }
  304 + if (isObject(options)) {
  305 + leading = 'leading' in options ? !!options.leading : leading;
  306 + trailing = 'trailing' in options ? !!options.trailing : trailing;
  307 + }
  308 + return debounce(func, wait, {
  309 + 'leading': leading,
  310 + 'maxWait': wait,
  311 + 'trailing': trailing
  312 + });
  313 +}
  314 +
  315 +/**
  316 + * Checks if `value` is the
  317 + * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)
  318 + * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)
  319 + *
  320 + * @static
  321 + * @memberOf _
  322 + * @since 0.1.0
  323 + * @category Lang
  324 + * @param {*} value The value to check.
  325 + * @returns {boolean} Returns `true` if `value` is an object, else `false`.
  326 + * @example
  327 + *
  328 + * _.isObject({});
  329 + * // => true
  330 + *
  331 + * _.isObject([1, 2, 3]);
  332 + * // => true
  333 + *
  334 + * _.isObject(_.noop);
  335 + * // => true
  336 + *
  337 + * _.isObject(null);
  338 + * // => false
  339 + */
  340 +function isObject(value) {
  341 + var type = typeof value;
  342 + return !!value && (type == 'object' || type == 'function');
  343 +}
  344 +
  345 +/**
  346 + * Checks if `value` is object-like. A value is object-like if it's not `null`
  347 + * and has a `typeof` result of "object".
  348 + *
  349 + * @static
  350 + * @memberOf _
  351 + * @since 4.0.0
  352 + * @category Lang
  353 + * @param {*} value The value to check.
  354 + * @returns {boolean} Returns `true` if `value` is object-like, else `false`.
  355 + * @example
  356 + *
  357 + * _.isObjectLike({});
  358 + * // => true
  359 + *
  360 + * _.isObjectLike([1, 2, 3]);
  361 + * // => true
  362 + *
  363 + * _.isObjectLike(_.noop);
  364 + * // => false
  365 + *
  366 + * _.isObjectLike(null);
  367 + * // => false
  368 + */
  369 +function isObjectLike(value) {
  370 + return !!value && typeof value == 'object';
  371 +}
  372 +
  373 +/**
  374 + * Checks if `value` is classified as a `Symbol` primitive or object.
  375 + *
  376 + * @static
  377 + * @memberOf _
  378 + * @since 4.0.0
  379 + * @category Lang
  380 + * @param {*} value The value to check.
  381 + * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.
  382 + * @example
  383 + *
  384 + * _.isSymbol(Symbol.iterator);
  385 + * // => true
  386 + *
  387 + * _.isSymbol('abc');
  388 + * // => false
  389 + */
  390 +function isSymbol(value) {
  391 + return typeof value == 'symbol' ||
  392 + (isObjectLike(value) && objectToString.call(value) == symbolTag);
  393 +}
  394 +
  395 +/**
  396 + * Converts `value` to a number.
  397 + *
  398 + * @static
  399 + * @memberOf _
  400 + * @since 4.0.0
  401 + * @category Lang
  402 + * @param {*} value The value to process.
  403 + * @returns {number} Returns the number.
  404 + * @example
  405 + *
  406 + * _.toNumber(3.2);
  407 + * // => 3.2
  408 + *
  409 + * _.toNumber(Number.MIN_VALUE);
  410 + * // => 5e-324
  411 + *
  412 + * _.toNumber(Infinity);
  413 + * // => Infinity
  414 + *
  415 + * _.toNumber('3.2');
  416 + * // => 3.2
  417 + */
  418 +function toNumber(value) {
  419 + if (typeof value == 'number') {
  420 + return value;
  421 + }
  422 + if (isSymbol(value)) {
  423 + return NAN;
  424 + }
  425 + if (isObject(value)) {
  426 + var other = typeof value.valueOf == 'function' ? value.valueOf() : value;
  427 + value = isObject(other) ? (other + '') : other;
  428 + }
  429 + if (typeof value != 'string') {
  430 + return value === 0 ? value : +value;
  431 + }
  432 + value = value.replace(reTrim, '');
  433 + var isBinary = reIsBinary.test(value);
  434 + return (isBinary || reIsOctal.test(value))
  435 + ? freeParseInt(value.slice(2), isBinary ? 2 : 8)
  436 + : (reIsBadHex.test(value) ? NAN : +value);
  437 +}
  438 +
  439 +module.exports = throttle;
... ...