如何造一个移动端的联动选择器(三)

之前写了一篇 MultiPicker -『为移动端而生』的自定义多级联动选择器,得到了很多人的关注。鉴于很多人对这种手写插件的过程很好奇,所以写了几篇文章,来说说它的成长史~

这一阶段,你将会了解到页面dom的实现手势算法的技巧

在阅读本文之前,确保你有稍微看过 MultiPicker 的源码 喔~

点击查看源码 ,也可以在** npm **上找到他们:

回顾上集:

如何造一个移动端的联动选择器(一)

如何造一个移动端的联动选择器(二)


四、日期选择器 和 自定义 JSON 选择器 的联动差别

思考第5个问题:『如果说滑动手势是它们之间的共同点,那它们之间又有什么区别?』

一个最明显的区别就是,日期选择器可以在多级之间反复调整,而自定义JSON 选择器只能从高级联动往下调整。

比如,在用日期选择器选择生日的时候。不小心操作失误了,选择成了1994-1-16

我想要修改年份为1995。当我滑动第一级联动时,后面的联动是不会改变的。


但是当我选择城市的时候,如果我选择了北京,下面的联动等级一定会全部配合 “北京” 这个高级联动,向下自适应改变。

选择了广东,下面的联动等级也一定会全部配合 “广东” 这个高级联动,向下自适应改变。

所以为了区别这两个不一样的联动场景,出现了两套不一样的联动算法。

五、日期选择器 的联动算法

思考第6个问题:『如何协调用户设置的时间点和实际时间点之间的联系?』

前面说到,用户如果设置了【月日时分】这四个时间单位的话,他可能会输入beginTime:[3,27,12,12], endTime 和 recentTime 也是类似。但是计算机如何快速识别这个开始时间,其实就是[2016, 3, 27, 12, 12] 呢?

如果把用户设置的时间点称为【虚拟时间】,而计算机能够处理的完整时间点称为【实际时间】,这个问题就简化了许多。

我做了一个小技巧,就是在我判断用户参数合法性的同时,把用户作为参数传入的【虚拟时间】( 如 :beginTime、endTime、 recentTime),转变成一个代码能够快速识别的【真实时间】(如:begin_time、end_time、 recent_time)。

另外,idxArrmaxHeightdistance 对应下标的值是和【虚拟时间】对应下标的值保持一致。

思考第7个问题:『如何计算联动数据,才能做到在多级之间反复调整?』

在我最新的重构算法中,我的解决方案是:

当ul被滑动时,就从最高级的联动开始【递归调用】。被递归的函数叫做checkRange

实现步骤如下:

① 每次touchend的时候,会先将当前滑动的结果保存,再调用checkRange(0);

checkRange会根据你的参数,直接设置下一级联动应该有的数据范围:

③ 判断好下一级的数据范围后,需要判断是否滑动到了开始时间(即最顶),或结束时间(即最底):

这里的loop是自己封装的 for 循环,一定要理解这里的dir到底是如何计算的。

④ 判断好dir的值之后,就需要对前面第②步生成的数据范围进行调整:

如果滑到了开始时间的分界点,需要处理min的值;

如果滑到了结束时间的分界点,需要处理max的值;

处理好后,再调用 initRangeArr 更新dom。

⑤ 在initRangeArr中更新dom之后,需要配合数据,调整好 ul 的translate3d。通过一系列的计算,得到targetLong的值,用来设置translate3d。并且同步好所有控制结果的数据,不仅仅是更新recent_timeresultArr,还需要更新 maxHeightdistance

⑥ 然后递归调用checkRange

PS:注意区分【虚拟时间】【真实时间】的下标含义哦。

【虚拟时间】的下标是指,在界面上的每个ul的下标,比如有三个ul,那么就是 [0, 1, 2];

【真实时间】的下标是指,【0:年】【1:月】【2:日】.... 以此类推。

六、自定义JSON选择器 的联动算法

思考第8个问题:『如何确定下一级的联动级数呢?』

由于前面规范了自定义JSON的格式,所以,如果判断下一级联动的级数问题,就转化成为了【如何计算下一级 child 的深度】问题。

我的解决方案是:迭代调用 checkArrDeep 来,就能判断是否还有子联动,从而计算深度。

具体实现步骤如下:

① 先传入一个需要计算深度的对象给 checkArrDeep,判断如果还有child则迭代,并计算深度。

② 生成所有子联动对应的 li,并更新dom,同时把子联动的 translate3d 都设置成 (0, 0, 0);

这步和 日期选择器 有着本质的不同,这里的子联动一定会更新,并且所有子联动一定会 translate3d(0 ,0 ,0);

PS: 这里有一个小技巧,就是一定要记录在更新联动之前的上一次联动的级数。我用ulCount 来记录上一次联动的级数,在完成所有更新操作之后,最后更新ulCount,以便下次迭代使用。

③ 如果增加了联动级数(需要ulCount来判断),则为新增加的联动 ul 绑定新的touch事件。如果减少了联动级数,则清除dom。

④ 重新设置联动的宽度,并同步更新所有计算联动会使用到的数据,包括ulCount:

七、用户可以自定义callback

思考第9个问题:『如何确定用户想要什么样的数据格式呢?』

最完美的解决方案,就是让用户自己写callback,自己解决所有的数据格式问题。

用户可以在回调中 拼接自己想要的字符串构造后台想要的json格式。很大程度上的增加了灵活度。

至此, 『日期选择器 - DateSelector』『自定义json选择器 - MulitiPicker』就算完成了。


写在最后

Github地址:『为移动端而生』的自定义多级联动选择器

到此,DOM的实现和手势操作已经实现。

预知后话,后两天见分晓

我是嘉宝Appian,一个卖萌出家的算法妹纸。

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

推荐阅读更多精彩内容