低调点现实点 發表於 2019-12-4 17:18:00

干货满满!如何优雅简洁地实现时钟翻牌器(支持JS/Vue/React)

<h1 class="article-title" data-v-0526462d=""><img class="lazyload inited loaded" style="font-size: 14px" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9e0d96aa2d32d?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9e0d96aa2d32d?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="940" data-height="400"></h1>
<div class="article-content" data-v-0526462d="" data-id="5ddb22a2e51d4523412df8c5">
<p>&nbsp;</p>
<p>双十一剁手节过去了,大家应该在很多网页中看到了数字翻牌的效果吧,比如倒计时、 数字增长等。相信很多人都已经自己独立实现过了,我也在网上看了一些demo,发现HTML结构大多比较复杂,用了4个并列的标签来放置前后两个“牌”。本文就来讲解下,如何进一步精简HTML,让结构简单,让JS方法封装得易使用。先来看看最终效果:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c6f769228554?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c6f769228554?imageslim" data-width="538" data-height="158">
<p>&nbsp;</p>
<p>每个翻牌的HTML结构(精简至2个并列标签):</p>
<pre><code class="hljs bash copyable" lang="bash">&lt;div class=<span class="hljs-string">"flip down"&gt;
    &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
    &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
&lt;/div&gt;
<span class="copy-code-btn">复制代码</span></span></span></span></code></pre>
<p>本次分享含有很多小技巧,灵活使用能够提升技术水平和工作效率,具体包括以下知识点:</p>
<p>知识点1:<code>:before :after</code>伪元素的使用</p>
<p>知识点2:line-height: 0的妙用</p>
<p>知识点3:transform-origin和perspective</p>
<p>知识点4:backface-visibility</p>
<p>知识点5:时间格式化函数的实现</p>
<p>Let's do it!</p>
<h3 class="heading" data-id="heading-0">1 翻牌的构建</h3>
<h4 class="heading" data-id="heading-1">1.1 基本结构</h4>
<p>首先解释下HTML的结构:</p>
<pre><code class="hljs bash copyable" lang="bash">&lt;!-- 翻牌的外框 --&gt;
&lt;div class=<span class="hljs-string">"flip down"&gt;
    &lt;!-- 位于前面的纸牌 --&gt;
    &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
    &lt;!-- 位于后面的纸牌 --&gt;
    &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
&lt;/div&gt;
<span class="copy-code-btn">复制代码</span></span></span></span></code></pre>
<p>【说明】</p>
<p>flip: 纸牌的外框</p>
<p>down:表示向下翻牌动效,还有对于的up。后面章节会具体讲解。</p>
<p>front: 表示位于前面的纸牌</p>
<p>back: 表示位于后面的纸牌</p>
<p>number*: 表示纸牌上的数字</p>
<p>flip的CSS代码如下:</p>
<pre><code class="hljs bash copyable" lang="bash">.flip {
    display: inline-block;
    position: relative;
    width: 60px;
    height: 100px;
    line-height: 100px;
    border: solid 1px <span class="hljs-comment">#000;
    border-radius: 10px;
    background: <span class="hljs-comment">#fff;
    font-size: 66px;
    color: <span class="hljs-comment">#fff;
    box-shadow: 0 0 6px rgba(0, 0, 0, .5);
    text-align: center;
    font-family: <span class="hljs-string">"Helvetica Neue"
}
<span class="copy-code-btn">复制代码</span></span></span></span></span></code></pre>
<p>这段代码很基础,就不再详细解释了。眼尖的同学可能发现了,为什么要设置background为#fff(白色)呢?最终效果明明是黑的。留个疑问,下一小节就会明白了。</p>
<p>基本结构的效果是这样的:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c702b7cde275?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c702b7cde275?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="816" data-height="260">
<p>&nbsp;</p>
<h4 class="heading" data-id="heading-2">1.2 构建纸牌并用伪元素拆分上下两部分</h4>
<p>由于每个纸牌是上下对折、翻转的,所以每个纸牌要拆分成上下两部分。可是HTML中每个纸牌只有一个标签,怎么拆分成两个呢?这里就用到了before和after伪元素。</p>
<h4 class="heading" data-id="heading-3">知识点1: 伪元素的使用</h4>
<p>先看代码:</p>
<pre><code class="hljs bash copyable" lang="bash">.flip .digital:before,
.flip .digital:after {
    content: <span class="hljs-string">"";
    position: absolute;
    left: 0;
    right: 0;
    background: <span class="hljs-comment">#000;
    overflow: hidden;
}

.flip .digital:before {
    top: 0;
    bottom: 50%;
    border-radius: 10px 10px 0 0;
}

.flip .digital:after {
    top: 50%;
    bottom: 0;
    border-radius: 0 0 10px 10px;
}
<span class="copy-code-btn">复制代码</span></span></span></code></pre>
<p>:before和:after在digital内部生成了两个伪元素,其中,before用来生成纸牌的“上半张”,after用来生成纸牌的“下半张”。</p>
<p>因此,before“上半张”为从“顶部(<code>top: 0</code>)”到“距底一半(<code>bottom: 50%</code>)”的部分,顶部两侧为圆角。</p>
<p>同理,after“下半张”为“距顶一半(<code>top: 50%</code>)”到“底部(<code>bottom: 0</code>)”的部分,底部两侧为圆角。</p>
<p>注意代码中的<code>content: ""</code>不能省略,否则伪元素是不显示的。</p>
<p>效果如下:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c72d80216cf9?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c72d80216cf9?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="814" data-height="258">
<p>&nbsp;</p>
<blockquote>
<p>回答上一章节的问题,为什么底层设置background为白色?</p>
<p>答案很简单,元素内部的纸片边角和外层边角之间会有一点点的缝隙,这个缝隙会露出底部的白色,从视觉效果上看,更加具有立体感。</p>
</blockquote>
<p>然后,为上下部分中间添加一条水平折线。</p>
<pre><code class="hljs bash copyable" lang="bash">    .flip .digital:before,
    .flip .digital:after {
      content: <span class="hljs-string">"";
      position: absolute;
      left: 0;
      right: 0;
      background: <span class="hljs-comment">#000;
      overflow: hidden;
+       box-sizing: border-box;
    }
    ...(略)
    .flip .digital:before {
      top: 0;
      bottom: 50%;
      border-radius: 10px 10px 0 0;
+       border-bottom: solid 1px <span class="hljs-comment">#666;
    }
<span class="copy-code-btn">复制代码</span></span></span></span></code></pre>
<p>外层flip添加<code>box-sizing: border-box</code>保证了下边框不会影响元素的原有高度。</p>
<p>效果如下:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c73974f71d34?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c73974f71d34?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="814" data-height="258">
<p>&nbsp;</p>
<p>到这里,我们可以认为是4个小纸片,分别是:</p>
<ol>
<li>前上:.digital.front:before</li>
<li>前下:.digital.front:after</li>
<li>后上:.digital.back:before</li>
<li>后下:.digital.back:after</li>
</ol>
<p>由于重叠在一起,只能看到一张纸牌。而看到的这个纸牌是后面(back)的纸牌,为什么呢?因为back的HTML写在了front的后面。不过没关系,后面我们会通过z-index来重新调整层叠顺序,先不着急。</p>
<h4 class="heading" data-id="heading-4">1.3 为纸牌添加文字</h4>
<p>还记的刚才的<code>content: ""</code>吗?纸牌的文字显示就用到了这个。</p>
<p>先通过CSS定义好0~9的数字:</p>
<pre><code class="hljs bash copyable" lang="bash">.flip .number0:before,
.flip .number0:after {
    content: <span class="hljs-string">"0";
}

.flip .number1:before,
.flip .number1:after {
    content: <span class="hljs-string">"1";
}

.flip .number2:before,
.flip .number2:after {
    content: <span class="hljs-string">"2";
}
...(略)
.flip .number9:before,
.flip .number9:after {
    content: <span class="hljs-string">"9";
}
<span class="copy-code-btn">复制代码</span></span></span></span></span></code></pre>
<p>现在效果如下:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c745cc2e5c7d?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c745cc2e5c7d?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="816" data-height="258">
<p>&nbsp;</p>
<p>可以很明显的看到两个问题:</p>
<ol>
<li>本应该在后面的back纸牌跑到了前面(z-index问题)</li>
<li>下半张纸牌的文字应该只显示下半部分。</li>
</ol>
<p>先来解决问题2,这里就涉及到了第二个知识点。</p>
<h4 class="heading" data-id="heading-5">知识点2:line-height: 0的妙用</h4>
<p>提到文字的显示,肯定会想到基线(baseline),可能你也曾经看过这个图:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c75b4dba9eda?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c75b4dba9eda?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="1000" data-height="360">
<p>&nbsp;</p>
<p>关于基线(baseline)的计算,确实很麻烦,我也在这里绕了很久。其实理解line-height:0可以换个角度,会更容易理解,请看下图:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7613684cea7?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7613684cea7?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="1280" data-height="556">
<p>&nbsp;</p>
<p>当line-height为200px,每行文字高度为200px,文字在200px高度的行间区域垂直居中;</p>
<p>当line-height为100px,每行文字高度为100px,文字在100px高度的行间区域垂直居中;</p>
<p>当line-height为0时,行间距为0,中线的位置也为0,所以文字只有下半部分留在容器内。</p>
<p>利用line-height:0的特性,就可以很轻易实现“下半张”纸牌只显示文字的下半部分,并且与“上半张”纸牌很好的衔接在一起。</p>
<p>在代码中设置line-height为0:</p>
<pre><code class="hljs bash copyable" lang="bash">    .flip .digital:after {
      top: 50%;
      bottom: 0;
      border-radius: 0 0 10px 10px;
+       line-height: 0;
    }
<span class="copy-code-btn">复制代码</span></code></pre>
<p>效果如下:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c786015a697f?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c786015a697f?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="664" data-height="214">
<p>&nbsp;</p>
<h4 class="heading" data-id="heading-6">1.4 设置纸牌的层叠关系</h4>
<p>首先,先看下“向下翻牌”的视频演示,直观感受下每个纸片的层级关系:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/25/16e9ff8d4ac29027?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/25/16e9ff8d4ac29027?imageslim" data-width="306" data-height="374">
<p>&nbsp;</p>
<p>按照实物图就可以确定每张纸片的z-index:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7a5c77bd2ad?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7a5c77bd2ad?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="1024" data-height="768">
<p>&nbsp;</p>
<p>添加以下CSS代码:</p>
<pre><code class="hljs bash copyable" lang="bash">/*向下翻*/
.flip.down .front:before {
    z-index: 3;
}

.flip.down .back:after {
    z-index: 2;
}

.flip.down .front:after,
.flip.down .back:before {
    z-index: 1;
}
<span class="copy-code-btn">复制代码</span></code></pre>
<p>现在效果如下:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7b6b1cc839d?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7b6b1cc839d?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="664" data-height="214">
<p>&nbsp;</p>
<p>咦?怎么不对?别着急,这是因为我们只设置了层级,但是没有把后面纸牌的下半部翻转上去。</p>
<p>添加翻转代码:</p>
<pre><code class="hljs bash copyable" lang="bash">    .flip.down .back:after {
      z-index: 2;
+       transform-origin: 50% 0%;
+       transform: perspective(160px) rotateX(180deg);
    }
<span class="copy-code-btn">复制代码</span></code></pre>
<p>这里涉及到了知识点3。</p>
<h4 class="heading" data-id="heading-7">知识点3:transform-origin和perspective</h4>
<p><code>transform-origin</code>是元素旋转的基本点。</p>
<p><code>transform-origin: 50% 0%;</code>表示将旋转基本点设置在横轴的中点,纵轴的顶点位置,如下图所示:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7c65e9d5856?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7c65e9d5856?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="664" data-height="216">
<p>&nbsp;</p>
<p><code>perspective(160px)</code>可以理解为立体透视图的景深。在本次分享的效果中,我们的视角是正对牌面,并且纸牌位于视角中间。所以 transform-origin的第一个值(X轴位置)为50%。</p>
<p><code>rotateX(180deg)</code>表示以X轴进行翻转,对应这里就是上下翻转。这里已经通过transform-origin的第二个参数(Y轴位置:0%)将X轴放在了元素顶部。</p>
<p>基于以上设置,已经可以正常显示了,如下图:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7d5b209a0a2?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c7d5b209a0a2?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="664" data-height="214">
<p>&nbsp;</p>
<p>同理,“向上翻”也需要进行设置下。大家可以自己折两个纸片,参照上面的方法,应该很容易实现。这里不再重复讲解,直接放上代码,大家可以对比下哪里不同:</p>
<pre><code class="hljs bash copyable" lang="bash">/*向上翻*/
.flip.up .front:after {
    z-index: 3;
}

.flip.up .back:before {
    z-index: 2;
    transform-origin: 50% 100%;
    transform: perspective(160px) rotateX(-180deg);
}

.flip.up .front:before,
.flip.up .back:after {
    z-index: 1;
}
<span class="copy-code-btn">复制代码</span></code></pre>
<h3 class="heading" data-id="heading-8">2 翻牌动画的实现</h3>
<p>现在纸片都已摆好了,剩下的就是实现CSS3动画,以及JS交互控制。</p>
<h4 class="heading" data-id="heading-9">2.1 CSS3翻牌动画</h4>
<p>我们还是以“向下翻”为例,再来看下之前的实物翻牌视频:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/25/16e9ff8d4ac29027?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/25/16e9ff8d4ac29027?imageslim" data-width="306" data-height="374">
<p>&nbsp;</p>
<p>可以看到,“向下翻”主要涉及两个元素的动画:</p>
<ol>
<li>前面纸牌的上半部向下翻转180度。</li>
<li>后面纸牌的下半部(目前已翻转上去)向下翻转180度恢复原状态。</li>
</ol>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/25/16ea12ead0c4c6be?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/25/16ea12ead0c4c6be?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="850" data-height="310">
<p>&nbsp;</p>
<p>直接上代码:</p>
<pre><code class="hljs bash copyable" lang="bash">.flip.down.go .front:before {
    transform-origin: 50% 100%;
    animation: frontFlipDown 0.6s ease-in-out both;
    box-shadow: 0 -2px 6px rgba(255, 255, 255, 0.3);
}

.flip.down.go .back:after {
    animation: backFlipDown 0.6s ease-in-out both;
}

@keyframes frontFlipDown {
    0% {
      transform: perspective(160px) rotateX(0deg);
    }

    100% {
      transform: perspective(160px) rotateX(-180deg);
    }
}

@keyframes backFlipDown {
    0% {
      transform: perspective(160px) rotateX(180deg);
    }

    100% {
      transform: perspective(160px) rotateX(0deg);
    }
}
<span class="copy-code-btn">复制代码</span></code></pre>
<p>以上代码涉及的知识点和原理没有新的东西,都已经讲解过了,就不详述了。box-shadow是为了给纸片的上边缘加一点白光,视觉效果更好一点。否则在翻转的时候,跟后面元素都是黑色,融在一起了。看看现在的效果:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c851bbb41c44?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c851bbb41c44?imageslim" data-width="538" data-height="158">
<p>&nbsp;</p>
<p>显示不正常!为什么?因为前排上半部纸片的z-index最高,所以它在翻转到下半部的时候仍然遮挡住了其他纸片。怎么优雅的解决?超级简单,来看看第四个知识点:</p>
<h4 class="heading" data-id="heading-10">知识点4:backface-visibility</h4>
<p>backface-visibility表示元素的背面是否可见,默认为visible(可见)。</p>
<p>这里的需求是,当前面上半部纸片翻转到一半的时候(90度)进入不可见状态。而纸牌翻转90度以后,正好是显露元素背面的开始,所以将backface-visibility设置为hidden即可完美解决!</p>
<p>修改代码如下:</p>
<pre><code class="hljs bash copyable" lang="bash">    .flip.down.go .front:before {
      transform-origin: 50% 100%;
      animation: frontFlipDown 0.6s ease-in-out both;
      box-shadow: 0 -2px 6px rgba(255, 255, 255, 0.3);
+       backface-visibility: hidden;
    }
<span class="copy-code-btn">复制代码</span></code></pre>
<p>现在效果很完美!</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c8848ea63f43?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c8848ea63f43?imageslim" data-width="538" data-height="158">
<p>&nbsp;</p>
<p>大家可以试着自己实现向上翻转效果,代码直接放出:</p>
<pre><code class="hljs bash copyable" lang="bash">.flip.up.go .front:after {
    transform-origin: 50% 0;
    animation: frontFlipUp 0.6s ease-in-out both;
    box-shadow: 0 2px 6px rgba(255, 255, 255, 0.3);
    backface-visibility: hidden;
}

.flip.up.go .back:before {
    animation: backFlipUp 0.6s ease-in-out both;
}
@keyframes frontFlipUp {
    0% {
      transform: perspective(160px) rotateX(0deg);
    }

    100% {
      transform: perspective(160px) rotateX(180deg);
    }
}

@keyframes backFlipUp {
    0% {
      transform: perspective(160px) rotateX(-180deg);
    }

    100% {
      transform: perspective(160px) rotateX(0deg);
    }
}
<span class="copy-code-btn">复制代码</span></code></pre>
<h4 class="heading" data-id="heading-11">2.2 JS实现翻牌交互</h4>
<p>现在我们来实现一个简单的交互。需求是:</p>
<ol>
<li>点击“+”,向下翻牌,数字+1</li>
<li>点击“-”,向上翻牌,数字-1</li>
</ol>
<p>首先,修改下HTML:</p>
<pre><code class="hljs bash copyable" lang="bash">+   &lt;div class=<span class="hljs-string">"single-demo"&gt;
M       &lt;div class=<span class="hljs-string">"flip down" id=<span class="hljs-string">"flip"&gt;
            &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
                &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
          &lt;/div&gt;
+        &lt;/div&gt;
+   &lt;div class=<span class="hljs-string">"btn-con"&gt;
+       &lt;button id=<span class="hljs-string">"btn1"&gt;向下翻+1&lt;/button&gt;
+       &lt;button id=<span class="hljs-string">"btn2"&gt;向上翻-1&lt;/button&gt;
+   &lt;/div&gt;
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></code></pre>
<p>配套的CSS如下,仅为了demo好看,无实际作用:</p>
<pre><code class="hljs bash copyable" lang="bash">.single-demo {
    margin: 50px auto;
    padding: 30px;
    width: 600px;
    text-align: center;
    border: solid 1px <span class="hljs-comment">#999;
}
<span class="copy-code-btn">复制代码</span></span></code></pre>
<p>Javascript代码如下:</p>
<pre><code class="hljs bash copyable" lang="bash">var flip = document.getElementById(<span class="hljs-string">'flip')
var backNode = document.querySelector(<span class="hljs-string">'.back')
var frontNode = document.querySelector(<span class="hljs-string">'.front')
var btn = document.getElementById(<span class="hljs-string">'btn')
btn1.addEventListener(<span class="hljs-string">'click', <span class="hljs-function"><span class="hljs-title">function() {
    flipDown();
})
btn2.addEventListener(<span class="hljs-string">'click', <span class="hljs-function"><span class="hljs-title">function() {
    flipUp();
})
// 当前数字
var count = 0
// 是否正在翻转(防止翻转未结束就进行下一次翻转)
var isFlipping = <span class="hljs-literal">false

// 向下翻转+1
<span class="hljs-keyword">function <span class="hljs-function"><span class="hljs-title">flipDown() {
    // 如果处于翻转中,则不执行
    <span class="hljs-keyword">if (isFlipping) {
      <span class="hljs-built_in">return <span class="hljs-literal">false
    }
    // 设置前牌的文字
    frontNode.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'digital front number' + count)
    // 计算后牌文字(越界判断)
    var nextCount = count &gt;= 9 ? 0 : (count + 1)
    // 设置后牌的文字
    backNode.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'digital back number' + nextCount)
    // 添加go,执行翻转动画
    flip.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'flip down go')
    // 将翻转态设置为<span class="hljs-literal">true
    isFlipping = <span class="hljs-literal">true
    // 翻转结束后,恢复状态
    <span class="hljs-built_in">setTimeout(<span class="hljs-function"><span class="hljs-title">function() {
      // 去掉go
      flip.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'flip down')
      // 将翻转态设置为<span class="hljs-literal">false
      isFlipping = <span class="hljs-literal">false
      // 设置前牌文字为+1后的数字
      frontNode.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'digital front number' + nextCount)
      // 更新当前文字
      count = nextCount
    }, 1000)
}
// 向上翻转-1(同理,注释略)
<span class="hljs-keyword">function <span class="hljs-function"><span class="hljs-title">flipUp() {
    <span class="hljs-keyword">if (isFlipping) {
      <span class="hljs-built_in">return <span class="hljs-literal">false
    }
    frontNode.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'digital front number' + count)
    var nextCount = count &lt;= 0 ? 9 : (count - 1)
    backNode.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'digital back number' + nextCount)
    flip.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'flip up go')
    isFlipping = <span class="hljs-literal">true
    <span class="hljs-built_in">setTimeout(<span class="hljs-function"><span class="hljs-title">function() {
      flip.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'flip up')
      isFlipping = <span class="hljs-literal">false
      frontNode.setAttribute(<span class="hljs-string">'class', <span class="hljs-string">'digital front number' + nextCount)
      count = nextCount
    }, 1000)
}
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>先看下交互效果:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c8e33b2cb7c0?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c8e33b2cb7c0?imageslim" data-width="538" data-height="167">
<p>&nbsp;</p>
<p>这段Javascript代码很冗余,重复代码很多。在实际产品中,都是多个数字牌,这种方式显然无法应对。下一章节,我们来说下如何优雅的封装,以不变应万变。</p>
<h3 class="heading" data-id="heading-12">3 翻牌时钟的实现</h3>
<p>先看下最终效果:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c6f769228554?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c6f769228554?imageslim" data-width="538" data-height="158">
<p>&nbsp;</p>
<h4 class="heading" data-id="heading-13">3.1 HTML构建</h4>
<p>HTML代码如下:</p>
<pre><code class="hljs bash copyable" lang="bash">&lt;div class=<span class="hljs-string">"clock" id=<span class="hljs-string">"clock"&gt;
    &lt;div class=<span class="hljs-string">"flip down"&gt;
      &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
      &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
    &lt;/div&gt;
    &lt;div class=<span class="hljs-string">"flip down"&gt;
      &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
      &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
    &lt;/div&gt;
    &lt;em&gt;:&lt;/em&gt;
    &lt;div class=<span class="hljs-string">"flip down"&gt;
      &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
      &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
    &lt;/div&gt;
    &lt;div class=<span class="hljs-string">"flip down"&gt;
      &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
      &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
    &lt;/div&gt;
    &lt;em&gt;:&lt;/em&gt;
    &lt;div class=<span class="hljs-string">"flip down"&gt;
      &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
      &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
    &lt;/div&gt;
    &lt;div class=<span class="hljs-string">"flip down"&gt;
      &lt;div class=<span class="hljs-string">"digital front number0"&gt;&lt;/div&gt;
      &lt;div class=<span class="hljs-string">"digital back number1"&gt;&lt;/div&gt;
    &lt;/div&gt;
&lt;/div&gt;
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>CSS代码如下(之前章节的CSS代码请保留):</p>
<pre><code class="hljs bash copyable" lang="bash">.clock {
    text-align: center;
}

.clock em {
    display: inline-block;
    line-height: 102px;
    font-size: 66px;
    font-style: normal;
    vertical-align: top;
}
<span class="copy-code-btn">复制代码</span></code></pre>
<p>效果如下,剩下的就是JS部分了。</p>
<p>&nbsp;</p>
<img class="lazyload inited" src="data:image/svg+xml;utf8,&lt;?xml version=&quot;1.0&quot;?&gt;&lt;svg xmlns=&quot;http://www.w3.org/2000/svg&quot; version=&quot;1.1&quot; width=&quot;850&quot; height=&quot;254&quot;&gt;&lt;/svg&gt;" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c8f5acb2f607?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="850" data-height="254">
<p>&nbsp;</p>
<h4 class="heading" data-id="heading-14">3.2 构建Flipper类</h4>
<p>将每个翻牌封装成类,这样在应对多个翻牌的时候,可以方便的通过new Flipper()去独立控制每个翻牌对象。</p>
<p>类的实现代码如下:</p>
<pre><code class="hljs bash copyable" lang="bash"><span class="hljs-keyword">function Flipper(config) {
    // 默认配置
    this.config = {
      // 时钟模块的节点
      node: null,
      // 初始前牌文字
      frontText: <span class="hljs-string">'number0',
      // 初始后牌文字
      backText: <span class="hljs-string">'number1',
      // 翻转动画时间(毫秒,与翻转动画CSS 设置的animation-duration时间要一致)
      duration: 600,
    }
    // 节点的原本class,与html对应,方便后面添加/删除新的class
    this.nodeClass = {
      flip: <span class="hljs-string">'flip',
      front: <span class="hljs-string">'digital front',
      back: <span class="hljs-string">'digital back'
    }
    // 覆盖默认配置
    Object.assign(this.config, config)
    // 定位前后两个牌的DOM节点
    this.frontNode = this.config.node.querySelector(<span class="hljs-string">'.front')
    this.backNode = this.config.node.querySelector(<span class="hljs-string">'.back')
    // 是否处于翻牌动画过程中(防止动画未完成就进入下一次翻牌)
    this.isFlipping = <span class="hljs-literal">false
    // 初始化
    this._init()
}
Flipper.prototype = {
    constructor: Flipper,
    // 初始化
    _init: <span class="hljs-function"><span class="hljs-title">function() {
      // 设置初始牌面字符
      this._setFront(this.config.frontText)
      this._setBack(this.config.backText)
    },
    // 设置前牌文字
    _setFront: <span class="hljs-keyword">function(className) {
      this.frontNode.setAttribute(<span class="hljs-string">'class', this.nodeClass.front + <span class="hljs-string">' ' + className)
    },
    // 设置后牌文字
    _setBack: <span class="hljs-keyword">function(className) {
      this.backNode.setAttribute(<span class="hljs-string">'class', this.nodeClass.back + <span class="hljs-string">' ' + className)
    },
    _flip: <span class="hljs-keyword">function(<span class="hljs-built_in">type, front, back) {
      // 如果处于翻转中,则不执行
      <span class="hljs-keyword">if (this.isFlipping) {
            <span class="hljs-built_in">return <span class="hljs-literal">false
      }
      // 设置翻转状态为<span class="hljs-literal">true
      this.isFlipping = <span class="hljs-literal">true
      // 设置前牌文字
      this._setFront(front)
      // 设置后牌文字
      this._setBack(back)
      // 根据传递过来的<span class="hljs-built_in">type设置翻转方向
      <span class="hljs-built_in">let flipClass = this.nodeClass.flip;
      <span class="hljs-keyword">if (<span class="hljs-built_in">type === <span class="hljs-string">'down') {
            flipClass += <span class="hljs-string">' down'
      } <span class="hljs-keyword">else {
            flipClass += <span class="hljs-string">' up'
      }
      // 添加翻转方向和执行动画的class,执行翻转动画
      this.config.node.setAttribute(<span class="hljs-string">'class', flipClass + <span class="hljs-string">' go')
      // 根据设置的动画时间,在动画结束后,还原class并更新前牌文字
      <span class="hljs-built_in">setTimeout(() =&gt; {
            // 还原class
            this.config.node.setAttribute(<span class="hljs-string">'class', flipClass)
            // 设置翻转状态为<span class="hljs-literal">false
            this.isFlipping = <span class="hljs-literal">false
            // 将前牌文字设置为当前新的数字,后牌因为被前牌挡住了,就不用设置了。
            this._setFront(back)
      }, this.config.duration)
    },
    // 下翻牌
    flipDown: <span class="hljs-keyword">function(front, back) {
      this._flip(<span class="hljs-string">'down', front, back)
    },
    // 上翻牌
    flipUp: <span class="hljs-keyword">function(front, back) {
      this._flip(<span class="hljs-string">'up', front, back)
    }
}
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>可以注意到,Flipper的传参只接受一个对象形式的参数config,使用对象的方式向函数传参有很多优点:</p>
<ol>
<li>参数语义化,方便理解</li>
<li>不用在意参数顺序</li>
<li>传参的增删和顺序调整不会影响业务代码的使用</li>
</ol>
<p>使用Object.assign方法,可将传递进来的config参数覆盖默认参数。传递的config中没有的属性,则使用默认配置。当然,这种方式只适用于浅拷贝。</p>
<blockquote>
<p>关于prototype,以及为什么要设置constructor,请阅读我的另一篇文章《一张刮刮卡竟包含这么多前端知识点》第4.1章节,已经讲解得很详细了。</p>
</blockquote>
<p>代码逻辑请阅读注释。</p>
<h4 class="heading" data-id="heading-15">3.3 实现时钟业务逻辑</h4>
<p>接下来的工作就是将js与dom进行绑定。</p>
<p>请看代码:</p>
<blockquote>
<p>这段代码一定要放在Flipper类代码的下面,Flipper.prototype一定要在业务逻辑代码之前执行,否则会报错找不到Flipper内部方法。</p>
</blockquote>
<pre><code class="hljs bash copyable" lang="bash">// 定位时钟模块
<span class="hljs-built_in">let clock = document.getElementById(<span class="hljs-string">'clock')
// 定位6个翻板
<span class="hljs-built_in">let flips = clock.querySelectorAll(<span class="hljs-string">'.flip')
// 获取当前时间
<span class="hljs-built_in">let now = new Date()
// 格式化当前时间,例如现在是20:30:10,则输出<span class="hljs-string">"203010"字符串
<span class="hljs-built_in">let nowTimeStr = formatDate(now, <span class="hljs-string">'hhiiss')
// 格式化下一秒的时间
<span class="hljs-built_in">let nextTimeStr = formatDate(new Date(now.getTime() + 1000), <span class="hljs-string">'hhiiss')
// 定义牌板数组,用来存储6个Flipper翻板对象
<span class="hljs-built_in">let flipObjs = []
<span class="hljs-keyword">for (<span class="hljs-built_in">let i = 0; i &lt; flips.length; i++) {
    // 创建6个Flipper实例,初始化并存入flipObjs
    flipObjs.push(new Flipper({
      // 每个Flipper实例按数组顺序与翻板DOM的顺序一一对应
      node: flips,
      // 按数组顺序取时间字符串对应位置的数字
      frontText: <span class="hljs-string">'number' + nowTimeStr,
      backText: <span class="hljs-string">'number' + nextTimeStr
    }))
}
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>代码逻辑不难,请阅读注释。比较值得分享的是其中的时间格式化函数formatDate。</p>
<h4 class="heading" data-id="heading-16">知识点5:时间格式化函数的实现</h4>
<p>为了方便业务使用,实现一个时间格式化方法,这个方法在很多其他业务中都会使用到,具有很普遍的实用价值。</p>
<p>需求是通过输入日期时间格式要求,输出对应的字符串。</p>
<p>例如:</p>
<p><code>yyyy-mm-dd hh:ii:ss</code>&nbsp;输出:<code>2019-06-02 08:30:37</code></p>
<p><code>yy-m-d h:i:s</code>&nbsp;输出:<code>19-6-2 8:30:37</code></p>
<p>先看代码:</p>
<pre><code class="hljs bash copyable" lang="bash">//正则格式化日期
<span class="hljs-keyword">function formatDate(date, dateFormat) {
    /* 单独格式化年份,根据y的字符数量输出年份
   * 例如:yyyy =&gt; 2019
            yy =&gt; 19
            y =&gt; 9
   */
    <span class="hljs-keyword">if (/(y+)/.test(dateFormat)) {
      dateFormat = dateFormat.replace(RegExp.<span class="hljs-variable">$1, (date.getFullYear() + <span class="hljs-string">'').substr(4 - RegExp.<span class="hljs-variable">$1.length));
    }
    // 格式化月、日、时、分、秒
    <span class="hljs-built_in">let o = {
      <span class="hljs-string">'m+': date.getMonth() + 1,
      <span class="hljs-string">'d+': date.getDate(),
      <span class="hljs-string">'h+': date.getHours(),
      <span class="hljs-string">'i+': date.getMinutes(),
      <span class="hljs-string">'s+': date.getSeconds()
    };
    <span class="hljs-keyword">for (<span class="hljs-built_in">let k <span class="hljs-keyword">in o) {
      <span class="hljs-keyword">if (new RegExp(`(<span class="hljs-variable">${k})`).test(dateFormat)) {
            // 取出对应的值
            <span class="hljs-built_in">let str = o + <span class="hljs-string">'';
            /* 根据设置的格式,输出对应的字符
             * 例如: 早上8时,hh =&gt; 08,h =&gt; 8
             * 但是,当数字&gt;=10时,无论格式为一位还是多位,不做截取,这是与年份格式化不一致的地方
             * 例如: 下午15时,hh =&gt; 15, h =&gt; 15
             */
            dateFormat = dateFormat.replace(RegExp.<span class="hljs-variable">$1, (RegExp.<span class="hljs-variable">$1.length === 1) ? str : padLeftZero(str));
      }
    }
    <span class="hljs-built_in">return dateFormat;
};

//日期时间补零
<span class="hljs-keyword">function padLeftZero(str) {
    <span class="hljs-built_in">return (<span class="hljs-string">'00' + str).substr(str.length);
}
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>代码逻辑请阅读注释,这里再补充下“日期时间补零padLeftZero”函数的说明。由于月、日、时、分、秒最多为2位数,所以这里只考虑最多补一个0的情况。</p>
<p>原理是:不管数字是几位,先在前面补两个0,再根据原数字的位数进行截取,最终输出固定为两位的补零数字</p>
<p>例如:数字"16"是两位数,先补两个0变成"0016",再从该字符串的索引开始截取(2=原数字的位数),由于字符串索引从开始,所以对应字符串的第3位,输出结果仍为"16。</p>
<p>同理,数字"8"是1位数,先补两个0变成"008",再从该字符串的索引开始截取(1=原数字的位数),即从第2位开始截取,输出"08"。</p>
<p>这样就实现了补零的功能。</p>
<p>现在看下效果,已经可以正确显示当前时间了。</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c968b524b368?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c968b524b368?imageView2/0/w/1280/h/960/format/webp/ignore-error/1" data-width="1158" data-height="360">
<p>&nbsp;</p>
<h4 class="heading" data-id="heading-17">3.4 运行时钟</h4>
<p>万事俱备,只差加个定时器让时钟翻动起来。</p>
<pre><code class="hljs bash copyable" lang="bash"><span class="hljs-built_in">setInterval(<span class="hljs-function"><span class="hljs-title">function() {
    // 获取当前时间
    <span class="hljs-built_in">let now = new Date()
    // 格式化当前时间
    <span class="hljs-built_in">let nowTimeStr = formatDate(new Date(now.getTime() - 1000), <span class="hljs-string">'hhiiss')
    // 格式化下一秒时间
    <span class="hljs-built_in">let nextTimeStr = formatDate(now, <span class="hljs-string">'hhiiss')
    // 将当前时间和下一秒时间逐位对比
    <span class="hljs-keyword">for (<span class="hljs-built_in">let i = 0; i &lt; flipObjs.length; i++) {
      // 如果前后数字没有变化,则直接跳过,不翻牌
      <span class="hljs-keyword">if (nowTimeStr === nextTimeStr) {
            <span class="hljs-built_in">continue
      }
      // 传递前后牌的数字,进行向下翻牌动画
      flipObjs.flipDown(<span class="hljs-string">'number' + nowTimeStr, <span class="hljs-string">'number' + nextTimeStr)
    }
}, 1000)
<span class="copy-code-btn">复制代码</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>
<p>这段代码逻辑很简单了,主要就是进行前后时间字符串的对比,然后设置纸牌并翻转。最终效果:</p>
<p>&nbsp;</p>
<img class="lazyload inited loaded" src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c6f769228554?imageslim" alt="" data-src="https://user-gold-cdn.xitu.io/2019/11/24/16e9c6f769228554?imageslim" data-width="538" data-height="158">
<p>&nbsp;</p>
<h3 class="heading" data-id="heading-18">4 Vue &amp; React封装</h3>
<p>由于篇幅有限,这里不再详述,原理都是一样的,只是利用Vue和React的API和语法进行封装。</p>
<p>原生JavaScript、Vue、React三个版本的演示源码请到我的github下载:</p>
<p>github.com/Yuezi32/fli…</p>
<p>本次分享讲解了如何优雅地实现结构简单的翻牌时钟,并对JS进行了科学高效的封装。其中也涉及到了CSS3的一些知识点和技巧。希望能对大家的工作有所帮助。</p>
</div><br><br>
来源:https://www.cnblogs.com/zhengpan/p/11984168.html
頁: [1]
查看完整版本: 干货满满!如何优雅简洁地实现时钟翻牌器(支持JS/Vue/React)