🎑提前祝大家中秋快乐,教你做一个【中秋花灯许愿】💖的网站

简介: 为了参加掘金社区的中秋征文活动专门策划了这么一个小站。但是物尽其用,我最近自研了一套Web网站基于扫小程序码登录的机制,正好借这个小站测试一下。

用CSS绘制一个荷花灯


本小节所示内容来自codepen


首先用svg来绘制一个花瓣


<svg viewBox='0 0 72 200'>
    <path d='M0,100 C0,66.6666667 12,33.3333333 36,0 C60,33.3333333 72,66.6666667 72,100 C72,133.333333 60,166.666667 36,200 C12,166.666667 0,133.333333 0,100 Z'></path>
</svg>


渲染出来它长这个样子


image.png



由于还得玩点花里胡哨的动画,所以我们不能在svg里把所有花瓣都画出来,而是需要在不同的dom节点里画,然后对dom应用动画。


比如下面这个三片花瓣


<div class='lotus'>
  <ul>
    <li>
      <svg viewBox='0 0 72 200'>
        <path d='M0,100 C0,66.6666667 12,33.3333333 36,0 C60,33.3333333 72,66.6666667 72,100 C72,133.333333 60,166.666667 36,200 C12,166.666667 0,133.333333 0,100 Z'></path>
      </svg>
    </li>
    <li>
      <svg viewBox='0 0 72 200'>
        <path d='M0,100 C0,66.6666667 12,33.3333333 36,0 C60,33.3333333 72,66.6666667 72,100 C72,133.333333 60,166.666667 36,200 C12,166.666667 0,133.333333 0,100 Z'></path>
      </svg>
    </li>
    <li>
      <svg viewBox='0 0 72 200'>
        <path d='M0,100 C0,66.6666667 12,33.3333333 36,0 C60,33.3333333 72,66.6666667 72,100 C72,133.333333 60,166.666667 36,200 C12,166.666667 0,133.333333 0,100 Z'></path>
      </svg>
    </li>
  </ul>
</div>


使用transform,结合rotateZrotateXrotateY将花瓣旋转指定角度


image.png


transform: rotateZ(0deg) rotateX(0deg) rotateY(0deg);


花瓣现在有点少,我们多加三层,每一层花瓣的倾斜角度都比上一层大一点


image.png


继续升级,花瓣颜色得是红里透白呀。


<svg class='gradient'>
  <defs>
    <radialGradient cx='50%' cy='5%' fx='50%' fy='5%' id='gradient' r='110%'>
      <stop offset='0%' stop-color='#ffffff'></stop>
      <stop offset='100%' stop-color='#ff0000'></stop>
    </radialGradient>
  </defs>
</svg>


image.png


放个俯视视角,方便大家更好的理解


image.png


莲花灯我们已经完成,现在写一个简单的页面,让莲花灯在静谧的湖面上飘起来~


让花灯转起来


和刚才花瓣倾斜的方法一样,用transform,但这次我们创建一个css动画animation


@keyframes rotate {
  from {
    transform: rotateX(0deg) rotateZ(0deg);
  }
  to {
    transform: rotateX(0deg) rotateZ(360deg);
  }
}


image.png


加上rotateX


@keyframes rotate {
  from {
    transform: rotateX(70deg) rotateZ(0deg);
  }
  to {
    transform: rotateX(70deg) rotateZ(360deg);
  }
}


image.png


漂在水面上


其实现在让花灯产生位移,就很有漂在水面的感觉了


image.png


当然我们可以再细节一点,从右下角漂到左上角,并加入一些近大远小的视觉差,这个我们后面再完善。


静谧的湖面


作为一个写意流程序员,重要的是意境,要合理的激发用户的想象力,不要什么都靠视觉


哈哈哈


哈哈哈哈


哈哈哈哈哈


主要原因是我试了几种波光粼粼的湖面,搭起来太难看了,还不如不要,就此作罢


云开发建站


为什么推荐大家用uniCloud


因为DCloud打钱了??


真没有!


其实用任意免费的云开发服务都可以,我看重的就是免费,免费静态网页托管,免费CDN....说白了就是一分钱不用掏,就可以搭建一个网站。还不是一个,uniCloud阿里云免费服务空间最多可以创建50个。


扫小程序码登录


我自研了一个基于扫小程序码登录Web网站的机制,会另起文章再给大家分享。目前我还没见过其他有扫小程序码扫码登录的网站,如果大家知道的话,请在评论区发给我瞅瞅。


新建心愿


image.png

if(event.action==="create"){
        //使用JWT对用户身份鉴权,心愿和用户的openid绑定
        try{
                var payload  = utils.verifyToken(event.token);
        }catch(err){
                return utils.responseData(4001,"token校验失败");
        }
        //UGC内容走微信小程序的内容安全审查接口
        var res = await msgSecCheck(payload.openid,event.content);
        if(res.result.suggest!="pass"){
                return utils.responseData(1,"该内容无法通过内容安全审查");
        }
        //在数据表(集合)wish中插入数据
        var content  =  event.content;
        await db.collection("wish").add({
                owner:payload.openid,
                content:content,
                createtime:Date.now()
        })
        //统一响应格式
        return utils.responseData(0,"发布成功",{
                content:content,
                color:color
        });
}


随机获取心愿


这里我使用聚合操作aggregate随机获取100条心愿,然后在前端显示。


else if(event.action==="get100"){
        var dbRes = await db.collection("wish")
        .aggregate()//进入聚合操作
        .sample({
            size: 100//随机取100条数据
        })
        .project({
            owner:0//返回的字段里去除owner字段
        })
        .end();//结束聚合操作
        return utils.responseData(0,"",dbRes.data);
}


我们应该基于以下两点保护用户的openid


  • 用户不应该了解什么是openid


  • 如果万一要知道,最多只能知道自己的,决不允许知道他人的openid


所以我们在返回的数据中使用field来剔除掉指定字段


心愿点赞


image.png


uniCloud的数据库是MongoDB,我接触这个的时间也不长。设计这个点赞系统废了不少脑细菌。-_-!


  • 点赞/取消赞


else if(event.action==="like"){
    //赞肯定是需要用户登录的,所以要鉴权,代码同上为节省空间被我删掉了
    //先查询是否有指定id的心愿,查不到就返回错误
    dbRes =  await db.collection("wish").doc(event.wishid).get();
    if(dbRes.affectdDocs<=0)return utils.responseData(1,"心愿id错误");
    //先查询有没有点赞记录
    dbRes = await db.collection("likes").where({
            owner:dbCmd.eq(payload.openid),
            wishid:dbCmd.eq(event.wishid)
    }).limit(1).get();
    if(dbRes.affectedDocs>0){
            //赞过了,取消赞
            await db.collection("likes").doc(dbRes.data[0]._id).remove();
            return utils.responseData(0,"",{liked:false});
    }else{
            //没赞过,新建赞记录
            await db.collection("likes").add({
                    wishid:event.wishid,
                    owner:payload.openid
            });
            return utils.responseData(0,"",{liked:true});
    }
}


由于加入了点赞系统,所以在查询心愿数据的时候,我们还需要通过联表查询来获取点赞总数和当前用户是否已点赞


const $ = db.command.aggregate;
var dbRes = await db.collection("wish")
.aggregate()
.sample({
        size: 100
})
.lookup({
        from:'likes',//需要联合查询的表名 
        localField:'_id',//当前表中的哪个字段去匹配
        foreignField:'wishid',//外部表中的哪个字段
        as:'likes'//新字段名称用于赋值返回结果
})
.addFields({
        "likecount":$.size("$likes"),//获取数组的长度,也就是点赞总数
        "liked": payload.openid?$.size($.filter({
                input:'$likes',
                as:"item",
                cond:$.eq(["$$item.owner",payload.openid])
        })):false
})
.project({
        owner:0,
        likes:0
})
.end()


这里还有个复杂的聚合表达式,我单独拿出来和大家解释下


  • 第一部分


$.filter({
        input:'$likes',
        as:"item",
        cond:$.eq(["$$item.owner",payload.openid])
})


刚才我们已经通过联表查询获得了字段名为likes的数组,现在我们要从中取得指定openid是否存在其中,也就是某用户对该条心愿是否点过赞了。这里返回的结果仍然是数组,所以在第二部分,我们继续对它使用size命令,由于每个人只允许对一条心愿点赞一次,所以我们会获得 0,1,用于


  • 第二部分


payload.openid?$.size(filter):false


由于我们允许不登录也可以访问网站,所以要对请求时用户还没登录,也就是没有token解码的情况兼容。有openid才去做filter,没有就直接给false。


至此我们就完成了一个给大家中秋许愿的小网站的前后端开发,把思路打开,你还可以把它变成是树洞,留言板,心愿墙.....期待大家的创意


后记


这个许愿网站是专门为掘金中秋征文活动而开发的,从设计到前后端开发大概一共花了2天左右,在这个case中我也得以深入的学习了一下MongoDB的联表查询,更重要的是可以让大家帮我测试一下扫小程序码登录网站的机制。


如果测试下来没什么问题,我后续还会基于此开发很多有意思的东西。



相关文章
|
前端开发 JavaScript
中秋之美——html+css+js制作中秋网页
中秋之美——html+css+js制作中秋网页
704 0
|
9月前
|
Linux 定位技术 iOS开发
【Python实现坦克大战,带你找回童年的快乐】附源码
【Python实现坦克大战,带你找回童年的快乐】附源码
237 0
|
存储 前端开发 JavaScript
你小子!过年了,写了一个拼图小游戏来拼掘金兔年礼盒,来玩玩不?
你小子!过年了,写了一个拼图小游戏来拼掘金兔年礼盒,来玩玩不?
215 2
|
Python
Pythyon|当中秋遇上国庆
Pythyon|当中秋遇上国庆
213 0
|
区块链 C# 开发工具
520快给你喜欢的女生发个表白软件吧!【手把手教学】
520快给你喜欢的女生发个表白软件吧!【手把手教学】
169 0
|
SQL JavaScript Java
搭建情人节表白网站(超详细过程,包教包会)
网站的搭建其实在七夕的时候就已经弄好了,只是当时不会搭建,然后就放了好几个月,偶然发现情人节快到了,遂重新搭建了这个网站,不过说实话除了网站还真想不出有啥能够体现程序员的特长,你说各种代码,c,c++,java即时给你做出不错的GUI界面,人家还不一定会打开,网站最容易了,有手(机)就行,下面是它的效果展示视频,觉得好的记得三连哦
454 0
搭建情人节表白网站(超详细过程,包教包会)
想过七夕?->女朋友陪你玩游戏(C)(上)
三子棋 七夕到了,没有女朋友怎么办?没事~ 写一个小游戏,让女朋友陪你玩三子棋~
想过七夕?->女朋友陪你玩游戏(C)(上)
想过七夕?->女朋友陪你玩游戏(C)(中)
在一定数值范围内,游戏随机生成一个数字,玩家输入比随机数大的数字时,程序会提醒,数值大于目标数,反之亦然。通过不断试错,最终找出目标数字。此游戏可以重复玩。
想过七夕?->女朋友陪你玩游戏(C)(中)
想过七夕?->女朋友陪你玩游戏(C)(下)
七夕到了,没有女朋友怎么办?没事~ 写一个小游戏,让女朋友陪你玩扫雷~
想过七夕?->女朋友陪你玩游戏(C)(下)
|
编译器 Python
圣诞节来了,怎能还没有圣诞树呢 快来为心爱的她送上专属的圣诞礼物叭~
圣诞节来了,怎能还没有圣诞树呢 快来为心爱的她送上专属的圣诞礼物叭~
185 0
圣诞节来了,怎能还没有圣诞树呢 快来为心爱的她送上专属的圣诞礼物叭~