微信JS-SDK的运用

做这个的需求是要判断用户是否分享了这个链接?我们是要从浏览器端分享这个链接,但是又没有直接判断用户是否发送了,只有这个链接是在微信里面打开的才有接口可以调用是否分享成功了,所以,想到的一个办法就是给需要分享的页面生成二维码信息,用户扫一扫后引导用户到这个页面,此时这个页面也是在微信端打开的,就可以诱导(词不太好)用户分享链接了。
微信JS-SDK提供了很全的接口,运用场景还是很多的,比如:分享到朋友圈、QQ、QQ空间等,在这里,就不一一举例了,相信官方文档应该比我说的详细点。

官方文档

大致说下流程吧:

1. 绑定域名
2. 引入文件
3. 后台代码编写
4. 配置各种接口
5. 测试

<span id="01">绑定域名</span>:

登录到网页微信公众号里面,在公众号设置-功能设置下的JS接口安全域名那配置,上面的说明需要看一下,每月就三次机会,珍惜一点!
域名是要备案的才可以的,不支持IP访问!
(题外话:我是用的ngrok做的内网穿透,可以用外网访问到本地的项目,方便调试。

<span id="02">引入文件</span>:

引入文件

<span id="03">后台代码编写</span>:

获取accesstoken和jsticket

代码说话比较有说服力...
因为是测试阶段,之前是直接存cookie里了,后来发现调试需要用手机微信调试,电脑端是不行的。后来改用session了,后期还可以优化的,测试阶段就不注重这些了。

微信的接口每日有请求限制,放session里保存90分钟,微信那边请求到的accesstoken和jsticket2小时自动失效,这里保存90分钟等失效后再重新获取。

只要访问下面的这个地址,就会获取到accesstoken和jsticket,放session中保存。

import java.io.IOException;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.ResponseBody;

import com.udbac.weixin.WxApiClient;


@Controller
@RequestMapping("/wxapi")
public class WxApiController {

    /**
     * 获取accesstoken和jsticket
     * @param request
     * @return
     * @throws IOException 
     */
    @RequestMapping(value = "/message/AccessToken", method = RequestMethod.GET)
    public @ResponseBody void doGet(HttpServletRequest request,HttpServletResponse response) throws IOException {
            
            HttpSession session = request.getSession();
            session.setMaxInactiveInterval(90*60);
            if(session.getAttribute("AccessToken") == null){
                session.setAttribute("AccessToken", WxApiClient.getAccessToken());
            }
            if(session.getAttribute("ticket") == null){
                session.setAttribute("ticket", WxApiClient.getTicket(session.getAttribute("AccessToken").toString()));
            }
            //获取浏览器访问访问服务器时传递过来的cookie数组
//          Cookie[] cookies = request.getCookies();
//          //如果用户是第一次访问,那么得到的cookies将是null
//          if (cookies!=null) {
//              for (int i = 0; i < cookies.length; i++) {
//                  Cookie cookie = cookies[i];
//                  if (cookie.getName().equals("AccessToken")) {
//                      String accessToken = cookie.getValue();
//                      System.out.println("浏览器中存在cookie:"+accessToken);
//                  }
//                  
//                  if (cookie.getName().equals("ticket")) {
//                      System.out.println("浏览器中存在ticket:"+cookie.getValue());
//                  }
//              }
//          }else {
//              //设置cookie的过期时间为1个半小时
//              Cookie cookie = new Cookie("AccessToken", WxApiClient.getAccessToken());//创建一个cookie,cookie的名字是lastAccessTime
//              Cookie ticketCookie = new Cookie("ticket", WxApiClient.getTicket(cookie.getValue()));
//              cookie.setMaxAge(90*60);
//              cookie.setPath("/");
//              ticketCookie.setMaxAge(90*60);
//              ticketCookie.setPath("/");
//              //将cookie对象添加到response对象中,这样服务器在输出response对象中的内容时就会把cookie也输出到客户端浏览器
//              response.addCookie(cookie);
//              response.addCookie(ticketCookie);
//              System.out.println("新添加的cookie:"+cookie.getValue());
//              System.out.println("新添加的ticket:"+ticketCookie.getValue());
//          }
            
            
            System.out.println("cookie:"+session.getAttribute("AccessToken"));
            System.out.println("ticket:"+session.getAttribute("ticket"));
    }
    

怎么获取到的accesstoken和jsticket?

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.ProtocolException;
import java.net.URL;
import java.net.URLConnection;

import javax.json.JsonObject;

import net.sf.json.JSONObject;

/**
 * Title:WxApiClient 
 * Description:
 * @author root
 * @date 2017年06月14日15:09:00
 */
public class WxApiClient {

    // APPID
    public static String appid = "公众号的appId";

    // APPSECRET
    public static String appsecret = "公众号的appSecret";

    // 获取AccessToken的接口的地址
    public static String access_token_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid="
            + appid + "&secret=" + appsecret;

    // jsapi票据接口地址
    public static String jsapi_ticket_url = "http://api.weixin.qq.com/cgi-bin/ticket/getticket?type=jsapi&access_token=%s";

    // 通过获取Token的接口得到accesstoken
    public static String getAccessToken() {
        String message = getMessage(access_token_url);
        JSONObject jsonObject = JSONObject.fromObject(message);
        String accessToken = (String) jsonObject.get("access_token");
        return accessToken;
    }

    /**
     * 通过accesstoken换取ticket票据
     * @param accessToken
     * @return
     */
    public static String getTicket(String accessToken){
        String jsTicket = getMessage(String.format(jsapi_ticket_url,accessToken));
        JSONObject jsonObject = JSONObject.fromObject(jsTicket);
        String ticket = (String)jsonObject.get("ticket");
        return ticket;
    } 
    
    /**
     * GET请求获取返回信息
     * 
     * @param urlPath
     * @return
     */
    public static String getMessage(String urlPath) {
        try {
            URL url = new URL(urlPath);
            URLConnection rulConnection = url.openConnection();
            HttpURLConnection httpUrlConnection = (HttpURLConnection) rulConnection;
            httpUrlConnection.setDoOutput(true);
            httpUrlConnection.setDoInput(true);
            httpUrlConnection.setUseCaches(false);
            httpUrlConnection.setRequestProperty("Content-type", "application/x-java-serialized-object");
            httpUrlConnection.setRequestMethod("GET");
            // httpUrlConnection.connect();
            OutputStream outStrm = httpUrlConnection.getOutputStream();
            ObjectOutputStream objOutputStrm = new ObjectOutputStream(outStrm);
            objOutputStrm.flush();
            objOutputStrm.close();
            // InputStream inStrm = httpUrlConnection.getInputStream();

            int code = httpUrlConnection.getResponseCode();
            // System.out.println("code " + code);
            // System.out.println(httpUrlConnection.getResponseMessage());

            // 读取响应内容
            String sCurrentLine = "";
            String sTotalString = "";
            if (code == 200) {
                java.io.InputStream is = httpUrlConnection.getInputStream();
                BufferedReader reader = new BufferedReader(new InputStreamReader(is));
                while ((sCurrentLine = reader.readLine()) != null)
                    if (sCurrentLine.length() > 0)
                        sTotalString = sTotalString + sCurrentLine.trim();
            } else {
                sTotalString = "远程服务器连接失败,错误代码:" + code;

            }
            // System.out.println("response:" + sTotalString);
            return sTotalString;

        } catch (MalformedURLException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (ProtocolException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        return urlPath;
    }

}

有了accesstoken和ticket了,就能生成权限验证配置(我习惯说成是签名)了。
就是官方文档中提到的下面这些:

wx.config({  
    debug: true, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,  
    可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。  
    appId: '', // 必填,公众号的唯一标识  
    timestamp: , // 必填,生成签名的时间戳  
    nonceStr: '', // 必填,生成签名的随机串  
    signature: '',// 必填,签名,见附录1  
    jsApiList: [] // 必填,需要使用的JS接口列表,所有JS接口列表见附录2  
});

上面的这些马上就能获取到了,看看怎么实现它的

import java.util.Map;

import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;

import com.udbac.weixin.WxSign;

import net.sf.json.JSONObject;

/**
 * js_sdk测试链接分享
 * Description:
 * @author root
 * @date 2017年6月14日 下午3:37:43
 */

@Controller
@RequestMapping("/wxtest")
public class TestShareController {
    
    /**
     * 生成需要的签名
     * @param request
     * @param url  //实际需要分享的URL
     * @return
     */
    @ResponseBody
    @RequestMapping(value="/sign",method=RequestMethod.POST)
    public JSONObject sign(HttpServletRequest request,@RequestParam("url") String url){
        JSONObject jsonObject = new JSONObject();
        String jsapi_ticket = null;
//      Cookie[] cookies = request.getCookies();
//      for (int i = 0; i < cookies.length; i++) {
//          Cookie cookie = cookies[i];
//          if(cookie.getName().equals("ticket")){
//              jsapi_ticket = cookie.getValue();
//          }
//      }
        HttpSession session = request.getSession();
        jsapi_ticket = session.getAttribute("ticket").toString();
        Map<String, String> sign = WxSign.sign(jsapi_ticket, url);
        for (Map.Entry entry : sign.entrySet()) {
            jsonObject.put(entry.getKey(), entry.getValue());
//            System.out.println(entry.getKey() + ", " + entry.getValue());
        }
        System.out.println(jsonObject);
        return jsonObject;
    }
}

当我们从页面ajax访问这个请求的时候,注意是先执行上面的/api/message/AccessToken请求,获取到accesstoken和jsticket了才能从页面获取这个签名规则,测试阶段,可以自己优化一下就行了。
访问成功后,会得到我们需要的签名信息,如下:

{"timestamp":"1497445752","nonceStr":"0870902e-2aea-4558-b380-3d04110f9700",  
"jsapi_ticket":"sM4AOVdWfPE4DxkXGEs8VFgEz7OrZJykcz39ZYlhg8AimcNz9J0kSWLNMaNGIjVSyOS5kPqiIiqXUTSXLVQkgw",  
"signature":"50f2a127e61dd78724fd547fd0677d9e41d70f9c",  
"url":"http://qq1216028459.tunnel.2bdata.com/stt/js_sdk/wx.html"}

<span id="04">配置各种接口</span>:

获取到的这些就是上面config中需要配置的了,看下面的代码吧:

<body>
    
还是写点什么吧~

<script type="text/javascript">
<!-- 可配置成公共请求,为所需要分享的页面生成签名规则 -->
$.post('../wxtest/sign',{'url':window.location.href},function(res){
    console.log(res);
    
    //配置
    wx.config({
        debug: true, // 开启调试模式,调用的所有api的返回值会在客户端alert出来,若要查看传入的参数,可以在pc端打开,参数信息会通过log打出,仅在pc端时才会打印。
        appId: '公众号的appId', // 必填,公众号的唯一标识
        timestamp: res.timestamp, // 必填,生成签名的时间戳
        nonceStr: res.nonceStr, // 必填,生成签名的随机串
        signature: res.signature,// 必填,签名,见附录1
        jsApiList: ['onMenuShareTimeline'] // 必填,需要使用的JS接口列表,所有JS接口列表见附录2
    });
    
    
    wx.ready(function() {  
        shareTime();  
    });  
    
     var shareImg = "http://tva1.sinaimg.cn/crop.0.0.640.640.50/ade8471cjw8eqtysan1p2j20hs0htdhu.jpg";  
    function shareTime() {  
        wx.onMenuShareTimeline({
            title: '我是标题啊', // 分享标题
            link: window.location.href, // 分享链接,该链接域名或路径必须与当前页面对应的公众号JS安全域名一致
            imgUrl: shareImg, // 分享图标
            success: function (e) { 
                // 用户确认分享后执行的回调函数
                $.toast("分享成功");
            },
            cancel: function (e) { 
                // 用户取消分享后执行的回调函数
                $.toast("分享失败", "cancel");
            }
        });
    }  
});

</script>
</body>
</html>

上面的wx.read是官方文档中提到的,是先执行config,再执行ready。
上面的有一个sign类,是官网提供给我们开发者的签名规则。

在文档的下面:

这个是我改了WxSign类名的。

import java.util.UUID;
import java.util.Map;
import java.util.HashMap;
import java.util.Formatter;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.io.UnsupportedEncodingException;  

public class WxSign {
    public static void main(String[] args) {
        String jsapi_ticket = "jsapi_ticket";

        // 注意 URL 一定要动态获取,不能 hardcode
        String url = "http://example.com";
        Map<String, String> ret = sign(jsapi_ticket, url);
        for (Map.Entry entry : ret.entrySet()) {
            System.out.println(entry.getKey() + ", " + entry.getValue());
        }
    };

    public static Map<String, String> sign(String jsapi_ticket, String url) {
        Map<String, String> ret = new HashMap<String, String>();
        String nonce_str = create_nonce_str();
        String timestamp = create_timestamp();
        String string1;
        String signature = "";

        //注意这里参数名必须全部小写,且必须有序
        string1 = "jsapi_ticket=" + jsapi_ticket +
                  "&noncestr=" + nonce_str +
                  "&timestamp=" + timestamp +
                  "&url=" + url;
        System.out.println(string1);

        try
        {
            MessageDigest crypt = MessageDigest.getInstance("SHA-1");
            crypt.reset();
            crypt.update(string1.getBytes("UTF-8"));
            signature = byteToHex(crypt.digest());
        }
        catch (NoSuchAlgorithmException e)
        {
            e.printStackTrace();
        }
        catch (UnsupportedEncodingException e)
        {
            e.printStackTrace();
        }

        ret.put("url", url);
        ret.put("jsapi_ticket", jsapi_ticket);
        ret.put("nonceStr", nonce_str);
        ret.put("timestamp", timestamp);
        ret.put("signature", signature);

        return ret;
    }

    private static String byteToHex(final byte[] hash) {
        Formatter formatter = new Formatter();
        for (byte b : hash)
        {
            formatter.format("%02x", b);
        }
        String result = formatter.toString();
        formatter.close();
        return result;
    }

    private static String create_nonce_str() {
        return UUID.randomUUID().toString();
    }

    private static String create_timestamp() {
        return Long.toString(System.currentTimeMillis() / 1000);
    }
}

到这步完了,基本就算完事了,具体该看怎么测试了。

<span id="05">测试</span>:

在微信上先访问wxapi/message/AccessToken,这就得到了accesstoken和ticket,再访问咱需要分享的这个页面,就配置好了签名信息,就算wx.config下面需要的信息。

正常进入到页面后,会有一些微信自带的提示,这是因为config里面配置了debug:true。
当我们点击这个页面右上角分享菜单的时候,我们点击分享到朋友圈,会出来下面这个界面

分享中

跟我们页面上配置的一模一样,O(∩_∩)O哈哈~

测试点击取消,出来下面这个提示

fail

这个提示就是上面配置的we ui给我们封装好的提示样式,还是很不错的吧。

测试点击发送,那样就到你的朋友圈里了

success

发送和取消就是上面页面中的封装的回调函数,用户是否发送了或者是取消分享了的回调后该怎么处理,这些知道怎么做了吧,各有各的需求了哈。

ok,完毕!
我的博客文章地址:http://www.hanyz.cn/2017/06/14/%E5%BE%AE%E4%BF%A1JS-SDK%E7%9A%84%E8%BF%90%E7%94%A8/

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

推荐阅读更多精彩内容