网站搜索框使用微信扫码

背景

客户要求可以直接识别标签二维码对某些仪器设备进行管理,类似于淘宝搜索框可以直接拍照搜索商品一样。前面已经做了一个网页调用摄像头识别二维码的功能,此功能有两个缺陷:

  • 识别需要先拍照,不能直接识别
  • 识别率低,尤其是拍照抖动,二维码内容稍微多一点或二维码小一点就识别不了。

以上两点相加就相当不好用了。

使用微信扫码

鉴于我们的系统已经集成到了微信公众号,所以准备调用微信扫码(只有在微信内打开的页面才能使用微信扫码)。 参照官方文档

绑定域名

打开公众号设置

选择功能设置

添加JS接口安全域名,注意你添加的域名目录下面上次图中微信指定的文件,如果文件没在根目录,域名需要填写到子目录

引入JS文件

使用vs2019创建默认的asp.net mvc项目 在index.cshtml页面中使用的页面引用js <script src="http://res.wx.qq.com/open/js/jweixin-1.1.0.js"></script>

前端主要代码

$(function () {
    /**
     * 判断是否微信内部打开页面
     * */
    function isWeiXin() {
        var ua = window.navigator.userAgent.toLowerCase();
        console.log(ua);//mozilla/5.0 (iphone; cpu iphone os 9_1 like mac os x) applewebkit/601.1.46 (khtml, like gecko)version/9.0 mobile/13b143 safari/601.1
        if (ua.match(/MicroMessenger/i) == 'micromessenger') {
            return true;
        }
        else {
            return false;
        }
    }
    /**
     * 只有微信内部页面打开才能用
     * */
    if (isWeiXin()) {
        $.ajax({
            type: "post",
            url: "/Home/GetSingDataAsync",
            data: {
                "url": kk.split('#')[0]
            },
            dataType: "json",
            success: function (data) {
                alert(JSON.stringify(data));
                wx.config({
                    debug: true, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。
                    appId: data.appId, // 必填,公众号的唯一标识
                    timestamp: data.timestamp, // 必填,生成签名的时间戳
                    nonceStr: data.nonceStr, // 必填,生成签名的随机串
                    signature: data.signature,// 必填,签名,见附录1
                    jsApiList: [
                        'checkJsApi',
                        'startRecord',
                        'stopRecord',
                        'translateVoice',
                        'scanQRCode',// 微信扫一扫接口
                        'openCard'
                    ] // 必填,需要使用的JS接口列表,所有JS接口列表见附录2
                });
                wx.error(function (res) {
                    alert("出错了:" + res.errMsg);//这个地方的好处就是wx.config配置错误,会弹出窗口哪里错误,然后根据微信文档查询即可。
                });
                wx.ready(function () {
                    wx.checkJsApi({
                        jsApiList: ['scanQRCode'],
                        success: function (res) {
                            //扫描二维码
                            wx.scanQRCode({
                                needResult: 1, // 默认为0,扫描结果由微信处理,1则直接返回扫描结果,
                                scanType: ["qrCode"], // 可以指定扫二维码还是一维码,默认二者都有
                                success: function (res) {
                                    var result = res.resultStr; // 当needResult 为 1 时,扫码返回的结果
                                    alert(result);//因为我这边是扫描后有个链接,然后跳转到该页面
                                },
                                error: function () {
                                    console.log('123');
                                }
                            });
                        }
                    });
                });
            },
            error: function (url) {
                alert("An error occurred!");
            }
        });
    } else {
        alert("请使用微信打开");
    }
})  

后端代码

public class HomeController : Controller
{
    //把APPID和APP_SECRET换成你自己的
    private const string APPID = "******";
    private const string APP_SECRET = "******";
    //为了调试方便我这里第一次把Token和Ticket获取到之后就写死了,应该写入缓存(7200s过期)
    private static string Token = "24_cQsz9scwyXLnPaAes5JlfHTfuQ2e3Iw5L8JyWfUpQiMnTk4IToOTZ7dP0Fv190ZHTy5ST--jeuDzYwoUj_hvhSHDX288YYLYVcrmvMzRPwld8ccTTzWGNTKZz53jYKDy5f8U1E886msDPsrwORGbAJABET";
    private static string Ticket = "HoagFKDcsGMVCIY2vOjf9qZA_fkPP3enjnT58qu16hzZN-3kwAP0NK6jgQM0jyAc0sK8cxaGkT9_DSgp6cHCpw";
    public ActionResult Index()
    {

        return View();
    }

    public ActionResult About()
    {
        ViewBag.Message = "Your application description page.";

        return View();
    }

    public ActionResult Contact()
    {
        ViewBag.Message = "Your contact page.";

        return View();
    }


    public async Task GetToken()
    {
        await GetTicketAsync();
    }
    //获取token和ticket
    private async Task<string> GetTicketAsync()
    {

        var tokenUrl = $"https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={APPID}&secret={APP_SECRET}";

        var client = new System.Net.WebClient();
        client.Encoding = Encoding.UTF8;
        client.Headers.Add("Content-Type", "Application/x-www-form-urlencoded");
        var responseData = client.UploadData(tokenUrl, "POST", new byte[0]);
        var responseText = Encoding.UTF8.GetString(responseData);
        var token = JsonConvert.DeserializeAnonymousType(responseText, new { access_token = "", expires_in = "" });


        Token = token.access_token;
        var ticketUrl = $"https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token={Token}&type=jsapi";
        var ticResponseData = client.UploadData(ticketUrl, "POST", new byte[0]);
        var ticResponseText = Encoding.UTF8.GetString(ticResponseData);
        var ticket = JsonConvert.DeserializeAnonymousType(ticResponseText, new { errcode = "", errmsg = "", ticket = "", expires_in = "" });
        Ticket = ticket.ticket;
        return "";
    }
    //获取签名字符串
    public async Task<string> GetSingDataAsync(string url)
    {

        var sign = new SignData();
        sign.appId = APPID;
        sign.nonceStr = Create_nonce_str();
        sign.timestamp = Create_timestamp();
        //var url = Request.Url.AbsoluteUri;
        if (url.IndexOf('#') > 0)
        {
            url = url.Substring(0, url.IndexOf('#'));
        }
        sign.url = url;
        var string1 = "jsapi_ticket=" + Ticket +
                "&noncestr=" + sign.nonceStr +
                "&timestamp=" + sign.timestamp +
                "&url=" + sign.url;

        //var string1 = GetTestSign();

        var sha1 = SHA1.Create();
        sign.signature = ByteToHex(sha1.ComputeHash(Encoding.UTF8.GetBytes(string1)));
        return JsonConvert.SerializeObject(sign);
    }
    //测试签名字符串,和微信官方提供的一样,用来测试签名方法是否正确
    private string GetTestSign()
    {
        var nonceStr = "Wm3WZYTPz0wzccnW";
        var ticket = "sM4AOVdWfPE4DxkXGEs8VMCPGGVi4C3VM0P37wVUCFvkVAy_90u5h9nbSlYy3-Sl-HhTdfl2fzFy1AOcHKP7qg";
        var timestamp = "1414587457";
        var url = "http://mp.weixin.qq.com?params=value";
        var string1 = "jsapi_ticket=" + ticket +
                "&noncestr=" + nonceStr +
                "&timestamp=" + timestamp +
                "&url=" + url;
        return string1;
    }

    /// <summary>
    /// 随机字符串
    /// </summary>
    /// <returns></returns>
    private string Create_nonce_str()
    {
        return Guid.NewGuid().ToString().Substring(0, 8);
    }
    /// <summary>
    /// 时间戳
    /// </summary>
    /// <returns></returns>
    private string Create_timestamp()
    {
        return (DateTime.Now.Ticks / 100000000).ToString();
    }
    private string ByteToHex(byte[] hash)
    {
        var sb = new StringBuilder();
        foreach (var b in hash)
        {
            sb.Append(b.ToString("x2"));
        }
        return sb.ToString();
    }
}

代码已上传github

我的博客即将同步至腾讯云+社区,邀请大家一同入驻:https://cloud.tencent.com/developer/support-plan?invite_code=1jyzgtqyi05ky

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏二狗的DBA之路

MySQL中意向锁的备忘

https://www.zhihu.com/question/51513268/answer/127777478

7810
来自专栏Java研发军团

SpringBoot+Shiro+Redis共享Session实例

在单机版的Springboot+Shiro的基础上,这次实现共享Session。

37720
来自专栏相约机器人

三四行代码打造元学习核心,PyTorch元学习库L2L现已开源

learn2learn 是一个用于实现元学习的 Pytorch 库,我们只需要加几行高层 API,就能为一般的机器学习流程添加元学习能力。例如在元学习 MNIS...

8510
来自专栏一猿小讲

7行代码搞定WEB服务

作为一个 Java 程序猿,写代码久了,各种技术也就都尝试了一个遍。先从 SSH1(Spring、Struts1、Hibernate)摸爬滚打转变到 SSH2(...

10220
来自专栏Linux知识积累

使用ab命令对网站进行压力测试

这是apache自带的一个网站压力测试工具,是ApacheBench的缩写,我们可以使用这个工具来对我们的网站进行压力测试

8830
来自专栏后端开发你必须学会的干货

一个 WebSocket 服务器是如何开发出来的?

WebSocket 协议是为了解决 http 协议的无状态、短连接(通常是)和服务端无法主动给客户端推送数据等问题而开发的新型协议,其通信基础也是基于 TCP。...

12620
来自专栏分享/效率/工具/软件

Prometheus使用[笔记]

58330
来自专栏IT码农

java之@Controller和@RestController以及@GetMapping和@PostMapping接收参数的格式使用

一、1.使用@Controller 注解,在对应的方法上,视图解析器可以解析return 的jsp,html页面,并且跳转到相应页面

35220
来自专栏江湖安得便相忘

Swoft 2 框架概览

最近在工作中使用到了一款swoole的衍生框架 Swoft ,虽然会有一些小毛病,但整体上还是不错的,安利一下,说不定就用上了呢?

27520
来自专栏分享/效率/工具/软件

kali安装KatanaFramework

个人博客:https://suveng.github.io/blog/​​​​​​​

8930

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励