element 源码学习四 —— color-picker 源码学习

在 element ui 中最让我好奇的组件之一就是 color-picker 着色器组件。这里还是通过几个问题来学习一下如何实现着色器的。

源码地址

在前几篇博客中说起过 element 组件都位于 package 目录下,那么本次学习的颜色选择器就是在 package/color-picker 目录中。
简单说下目录结构:

目录结构

  • src 源码文件夹
    • components 组件文件夹
      • alpha-slider.vue 透明度选择器
      • hue-slider.vue 色调选择器
      • picker-dropdown.vue 下拉界面(几个选择器的组合)
      • sv-panel 颜色选择器
    • color.js 颜色处理逻辑
    • draggable.js 选择器拖动效果逻辑
    • main.vue color-picker 的整体界面实现。
  • cooking.conf.js cooking 配置
  • index.js index文件,用于导出组件
  • package.json 组件信息配置文件

下面通过问答解决问题的方式来学习 color-picker 组件。

回答几个源码问题

整体组件的结构是怎样的?

从整体结构来看,color-picker 的结构其实是多个组件的组合而成的。

  • 显示颜色结果的 span选择颜色的下拉框组成整体的 color-picker 组件;
  • 其中下拉框由以下组件组合而成;
    • 3个颜色选择器
    • 1个input
    • 1个清空button
    • 1个确定button
组件结构
结构图

选择器的背景颜色变化是如何实现的?

3 个颜色选择器都是由 CSS3 的线性渐变效果 linear-gradient() 来实现的。下面是简化版~

    <style>
        .div01 {
            width: 27px;
            height: 350px;
            background: linear-gradient(to bottom, red 0, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, red 100%);
        }

        .bg-white {
            width: 450px;
            height: 350px;
            position: absolute;
            background: linear-gradient(to right, #fff, rgba(255, 255, 255, 0));
        }

        .bg-black {
            width: 450px;
            height: 350px;
            position: absolute;
            background: linear-gradient(to top, #000, transparent);
        }

        .div02 {
            width: 450px;
            height: 350px;
            position: relative;
            background: rgb(213, 0, 255);
        }

        .div03 {
            height: 27px;
            width: 450px;
            background: linear-gradient(to right, rgba(213, 0, 255, 0) 0%, rgb(213, 0, 255) 100%);
        }
    </style>

    <div class="div02">
        <div class="bg-white"></div>
        <div class="bg-black"></div>
    </div>
    <div class="div01"></div>
    <div class="div03"></div>

最终结果如图所示:
显示结果

原来看似复杂的颜色选择器知识用了几个渐变就组合出来了,CSS 真的很强大!

如何计算并获取选中的色值?

颜色结果的计算逻辑都在 color.js 中了,来简单看下代码。

// hsv 转 hsl
const hsv2hsl = function(hue, sat, val) {};

// 是否为 1.0
const isOnePointZero = function(n) {};

// 是否为百分比
const isPercentage = function(n) {};

// Take input from [0, n] and return it as [0, 1]
const bound01 = function(value, max) {};

// 十进制转十六进制
const INT_HEX_MAP = { 10: 'A', 11: 'B', 12: 'C', 13: 'D', 14: 'E', 15: 'F' };

// 转为十六进制颜色值
const toHex = function({ r, g, b }) {};

// 十六进制转十进制
const HEX_INT_MAP = { A: 10, B: 11, C: 12, D: 13, E: 14, F: 15 };

// 解析十六进制
const parseHexChannel = function(hex) {};

// hsl 转 hsv
const hsl2hsv = function(hue, sat, light) {};

// rgb 转 hsv
const rgb2hsv = function(r, g, b) {};


// hsv 转 rgb
const hsv2rgb = function(h, s, v) {};

export default class Color {
  constructor(options) {
    this._hue = 0;
    this._saturation = 100;
    this._value = 100;
    this._alpha = 100;

    this.enableAlpha = false;
    this.format = 'hex';
    this.value = '';

    options = options || {};

    for (let option in options) {
      if (options.hasOwnProperty(option)) {
        this[option] = options[option];
      }
    }

    this.doOnChange();
  }
  // 设置属性值
  set(prop, value) {
    if (arguments.length === 1 && typeof prop === 'object') {
      for (let p in prop) {
        if (prop.hasOwnProperty(p)) {
          this.set(p, prop[p]);
        }
      }

      return;
    }

    this['_' + prop] = value;
    this.doOnChange();
  }
  // 获取属性值 _hue
  get(prop) {
    return this['_' + prop];
  }
  // 颜色值转为 rgb 返回
  toRgb() {
    return hsv2rgb(this._hue, this._saturation, this._value);
  }
  // 格式化传入的值
  fromString(value) {
    if (!value) {
      this._hue = 0;
      this._saturation = 100;
      this._value = 100;

      this.doOnChange();
      return;
    }
    // 定义计算出结果后:赋值、改变。
    const fromHSV = (h, s, v) => {
      this._hue = Math.max(0, Math.min(360, h));
      this._saturation = Math.max(0, Math.min(100, s));
      this._value = Math.max(0, Math.min(100, v));

      this.doOnChange();
    };

    /* 颜色变化逻辑,最后都会转为 HSV 三个值执行 fromHSV 方法 */
  }

  // 更具计算结果定义当前颜色值 value
  doOnChange() {
    const { _hue, _saturation, _value, _alpha, format } = this;

    if (this.enableAlpha) {
      switch (format) {
        case 'hsl':
          const hsl = hsv2hsl(_hue, _saturation / 100, _value / 100);
          this.value = `hsla(${ _hue }, ${ Math.round(hsl[1] * 100) }%, ${ Math.round(hsl[2] * 100) }%, ${ _alpha / 100})`;
          break;
        case 'hsv':
          this.value = `hsva(${ _hue }, ${ Math.round(_saturation) }%, ${ Math.round(_value) }%, ${ _alpha / 100})`;
          break;
        default:
          const { r, g, b } = hsv2rgb(_hue, _saturation, _value);
          this.value = `rgba(${r}, ${g}, ${b}, ${ _alpha / 100 })`;
      }
    } else {
      switch (format) {
        case 'hsl':
          const hsl = hsv2hsl(_hue, _saturation / 100, _value / 100);
          this.value = `hsl(${ _hue }, ${ Math.round(hsl[1] * 100) }%, ${ Math.round(hsl[2] * 100) }%)`;
          break;
        case 'hsv':
          this.value = `hsv(${ _hue }, ${ Math.round(_saturation) }%, ${ Math.round(_value) }%)`;
          break;
        case 'rgb':
          const { r, g, b } = hsv2rgb(_hue, _saturation, _value);
          this.value = `rgb(${r}, ${g}, ${b})`;
          break;
        default:
          this.value = toHex(hsv2rgb(_hue, _saturation, _value));
      }
    }
  }
};

其中,将工具方法和计算颜色的具体方法隐藏了,只看具体逻辑。
其实 color.js 主要是定义了一个 Color 类,简单说下其中一些方法的作用:

  • set 用于设置 Color 中的变量。
  • get 用于获取 _hue _saturation _value _alpha 这四个值。
  • toRgb 方法将当前颜色的值(除了透明度)以 RGB 的形式返回。
  • fromString 方法将传入的颜色值解析成 HSV 格式,并赋值给 _hue _saturation _value_alpha
  • doOnChange 方法将会计算颜色值组成字符串传给 value

至此,Color 的大致功能就清晰了:解析传入的颜色值为 HSVA 格式分别表示为 _hue _saturation _value_alpha,并且组合成颜色字符串传给 value
现在需要把获取到的颜色值传给显示结果的 span,那么就从 main.vue 的中显示颜色结果的 <span> 标签开始看起。

<span class="el-color-picker__color-inner" :style="{ backgroundColor: displayedColor }"></span>

背景色调用了 displayedColor 这个 computed 属性:

    computed: {
      displayedColor() {
        if (!this.value && !this.showPanelColor) {
          return 'transparent';
        } else {
          const { r, g, b } = this.color.toRgb();
          return this.showAlpha
            ? `rgba(${ r }, ${ g }, ${ b }, ${ this.color.get('alpha') / 100 })`
            : `rgb(${ r }, ${ g }, ${ b })`;
        }
      },
  }

这里的 this.value 是 props 中传入的属性。如果没有传入 value 并且没有选择过颜色,那么显示透明色;
this.color 是 Color 类的实例化对象:

const color = new Color({
   enableAlpha: this.showAlpha,
   format: this.colorFormat
 });

所以,就调用了我们上面所说的 toRgb 方法,最后返回颜色结果。
至此,实现了颜色的计算、获取和显示。

颜色选择器如何获取和修改颜色值?

在下拉菜单中 hue-slider 组件获取色调(哪种颜色)、sv-panel 获取具体的颜色值、alpha-silder 获取透明度。
这三个组件通过 props 获取父级组件传递的的 color 对象来显示颜色。如果颜色选择器的选择块移动后,通过修改 color 值来实现颜色的修改。

颜色选择器的选择块如何实现

选择颜色的过程其实就是选择器位移发生变化的过程。下面是作者参照 element 做的一个在有限范围内任意移动选择器的 demo:

    <style>
        #container {
            width: 500px;
            height: 500px;
            position: relative;
            border: 1px solid black;
        }

        .drag {
            height: 4px;
            width: 4px;
            position: absolute;
            border-radius: 50%;
            border: 1px solid red;
            cursor: pointer;
        }
    </style>

    <div id="app">
        <div id="container" ref="container">
            <div class="drag" 
            :style="{
                top: cursorTop + 'px',
                left: cursorLeft + 'px'
            }"></div>
        </div>

    </div>

    <script>
        new Vue({
            el: "#app",
            data: {
                cursorLeft: 0,
                cursorTop: 0,
            },
            mounted() {
                draggable(this.$el, {
                    drag: (event) => {
                        this.handleDrag(event);
                    },
                    end: (event) => {
                        this.handleDrag(event);
                    }
                });

                this.update();
            },
            methods: {
                handleDrag(event) {
                    const container = this.$refs.container
                    const el = this.$el;
                    const rect = container.getBoundingClientRect();

                    let left = event.clientX - rect.left;
                    let top = event.clientY - rect.top;
                    left = Math.max(0, left);
                    left = Math.min(left, rect.width - 6);

                    top = Math.max(0, top);
                    top = Math.min(top, rect.height - 6);

                    this.cursorLeft = left;
                    this.cursorTop = top;
                }
            }
        })

        let isDragging = false;

        function draggable(element, options) {
            if (Vue.prototype.$isServer) return;
            const moveFn = function (event) {
                if (options.drag) {
                    options.drag(event);
                }
            };
            const upFn = function (event) {
                document.removeEventListener('mousemove', moveFn);
                document.removeEventListener('mouseup', upFn);
                document.onselectstart = null;
                document.ondragstart = null;

                isDragging = false;

                if (options.end) {
                    options.end(event);
                }
            };
            element.addEventListener('mousedown', function (event) {
                if (isDragging) return;
                document.onselectstart = function () { return false; };
                document.ondragstart = function () { return false; };

                document.addEventListener('mousemove', moveFn);
                document.addEventListener('mouseup', upFn);
                isDragging = true;

                if (options.start) {
                    options.start(event);
                }
            });
        }
    </script>

好吧,我知道代码太长了,要看效果请移步此处
选择器的逻辑如下:

  • 根据 props 传入的颜色值初次计算选择器的位置。
  • 拖动选择器,根据选择器位置、已知的 color 属性计算当前选择器位置的颜色结果。

也就是说做一个选择器需要的就是一个可拖动的选择器一套计算颜色的算法逻辑。比如在 sv-silder 中的算法逻辑如下:

// 计算 cursor 位置
this.cursorLeft = saturation * width / 100;
this.cursorTop = (100 - value) * height / 100;
// 计算颜色
this.color.set({
  saturation: left / rect.width * 100,
 value: 100 - top / rect.height * 100
});

其他两个选择器原理也是类似的~

最后

至此,我对 color-picker 的一些疑惑都解开了,也写了一些 demo 来玩玩。对该组件有了大致的理解了~不得不感叹作者对于 CSS 和 Vue 的掌握真的非常熟练。学到了不少东西,感谢开源社区给我们提供了那么多好东西给我们使用和学习~
再下一篇文章中我想探索下其他一些有趣的 element 组件,敬请期待!

打个广告

上海链家-链家上海研发中心需求大量前端、后端、测试,需要内推请将简历发送至 dingxiaojie001@ke.com

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,089评论 4 62
  • 额~~~ 不知道怎么了,突然就成了网红, 成为了所有人的儿子,汗-_-|| 我以前可是王子, 跟公主一吻定情, 怎...
    嘉宝与劳拉阅读 326评论 0 0
  • 就这样静静地躺在懒人河的怀中 在波纹起伏间随意徜徉 清风徐来亲吻着我们的肌肤 同时也带走了夏日的暑热 你可以选择抬...
    向上的蜗牛君阅读 200评论 0 0
  • 正则表达式是程序开发中一个重要的元素,它提供用来描述或匹配文本的字符串,如特定的字符、词或算式等。但在某些情况下,...
    sara_org阅读 1,119评论 1 5
  • 晚梦回休 人影依稀、花开渐瘦。 歌阕茶楼,风流无数、最是残烛白头。 月朗星稀,柏雪佳人,何似紫足翘首。 俱往矣,百...
    yida91阅读 143评论 0 0