干货满满!如何优雅简洁地实现时钟翻牌器(支持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> </p>
<p>双十一剁手节过去了,大家应该在很多网页中看到了数字翻牌的效果吧,比如倒计时、 数字增长等。相信很多人都已经自己独立实现过了,我也在网上看了一些demo,发现HTML结构大多比较复杂,用了4个并列的标签来放置前后两个“牌”。本文就来讲解下,如何进一步精简HTML,让结构简单,让JS方法封装得易使用。先来看看最终效果:</p>
<p> </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> </p>
<p>每个翻牌的HTML结构(精简至2个并列标签):</p>
<pre><code class="hljs bash copyable" lang="bash"><div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<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"><!-- 翻牌的外框 -->
<div class=<span class="hljs-string">"flip down">
<!-- 位于前面的纸牌 -->
<div class=<span class="hljs-string">"digital front number0"></div>
<!-- 位于后面的纸牌 -->
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<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> </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> </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> </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> </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> </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> </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> </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> </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> </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> </p>
<p>关于基线(baseline)的计算,确实很麻烦,我也在这里绕了很久。其实理解line-height:0可以换个角度,会更容易理解,请看下图:</p>
<p> </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> </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> </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> </p>
<h4 class="heading" data-id="heading-6">1.4 设置纸牌的层叠关系</h4>
<p>首先,先看下“向下翻牌”的视频演示,直观感受下每个纸片的层级关系:</p>
<p> </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> </p>
<p>按照实物图就可以确定每张纸片的z-index:</p>
<p> </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> </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> </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> </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> </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> </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> </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> </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> </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> </p>
<p>可以看到,“向下翻”主要涉及两个元素的动画:</p>
<ol>
<li>前面纸牌的上半部向下翻转180度。</li>
<li>后面纸牌的下半部(目前已翻转上去)向下翻转180度恢复原状态。</li>
</ol>
<p> </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> </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> </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> </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> </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> </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">+ <div class=<span class="hljs-string">"single-demo">
M <div class=<span class="hljs-string">"flip down" id=<span class="hljs-string">"flip">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
+ </div>
+ <div class=<span class="hljs-string">"btn-con">
+ <button id=<span class="hljs-string">"btn1">向下翻+1</button>
+ <button id=<span class="hljs-string">"btn2">向上翻-1</button>
+ </div>
<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 >= 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 <= 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> </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> </p>
<p>这段Javascript代码很冗余,重复代码很多。在实际产品中,都是多个数字牌,这种方式显然无法应对。下一章节,我们来说下如何优雅的封装,以不变应万变。</p>
<h3 class="heading" data-id="heading-12">3 翻牌时钟的实现</h3>
<p>先看下最终效果:</p>
<p> </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> </p>
<h4 class="heading" data-id="heading-13">3.1 HTML构建</h4>
<p>HTML代码如下:</p>
<pre><code class="hljs bash copyable" lang="bash"><div class=<span class="hljs-string">"clock" id=<span class="hljs-string">"clock">
<div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<em>:</em>
<div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<em>:</em>
<div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
<div class=<span class="hljs-string">"flip down">
<div class=<span class="hljs-string">"digital front number0"></div>
<div class=<span class="hljs-string">"digital back number1"></div>
</div>
</div>
<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> </p>
<img class="lazyload inited" src="data:image/svg+xml;utf8,<?xml version="1.0"?><svg xmlns="http://www.w3.org/2000/svg" version="1.1" width="850" height="254"></svg>" 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> </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(() => {
// 还原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 < 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> 输出:<code>2019-06-02 08:30:37</code></p>
<p><code>yy-m-d h:i:s</code> 输出:<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 => 2019
yy => 19
y => 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 => 08,h => 8
* 但是,当数字>=10时,无论格式为一位还是多位,不做截取,这是与年份格式化不一致的地方
* 例如: 下午15时,hh => 15, h => 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> </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> </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 < 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> </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> </p>
<h3 class="heading" data-id="heading-18">4 Vue & 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]