1. 程式人生 > >Asp.net 如何實現微信公眾號授權登入

Asp.net 如何實現微信公眾號授權登入

第一個類:封裝好微信配置檔案

using System;

using System.Collections.Generic;

using System.Linq;

using System.Net;

using System.Web;

using Newtonsoft.Json;

/// <summary>

/// 微信公眾號配置檔案

/// </summary>

/// 

public class Config : System.Web.SessionState.IRequiresSessionState

{

    /// <summary>

    /// 開發者ID

    /// </summary>

    public static string AppId { get; set; }

    /// <summary>

    /// 開發者密碼

    /// </summary>

    public static string AppSecret { get; set; }

    /// <summary>

    /// APP access_token 7200秒更新一次

    /// </summary>

    public static string access_token

    {

        get

        {

            try

            {

                if (HttpContext.Current.Session["$access_token"] == null)

                {

                    using (WebClient wc = new WebClient())

                    {

                        wc.Encoding = System.Text.Encoding.UTF8;

                        string ret = wc.DownloadString(string.Format("https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={0}&secret={1}", Config.AppId, Config.AppSecret));

                        Dictionary<string, object> o = JsonConvert.DeserializeObject<Dictionary<string, object>>(ret);

                        if (o.ContainsKey("errmsg")) return o["errmsg"].ToString();

                        HttpContext.Current.Session.Add("$access_token", o["access_token"].ToString());

                        HttpContext.Current.Session.Timeout = 7200;

                        return o["access_token"].ToString();

                    }

                }

                return HttpContext.Current.Session["$access_token"].ToString();

            }

            catch

            {

                throw;

            }

        }

    }

}

第二個類:封裝好HTTP請求類

using System;

using System.Collections.Specialized;

using System.Net;

/// <summary>

/// 封裝好HTTP請求方便呼叫

/// </summary>

public class HttpService

{

    public static string Get(string url)

    {

        using (WebClient c = new WebClient())

        {

            try

            {

                c.Encoding = System.Text.Encoding.UTF8;

                return c.DownloadString(url);

            }

            catch (Exception)

            {

                throw;

            }

        }

    }

    public static string GetPost(string url, string parameters)

    {

        string ret = "";

        using (WebClient c = new WebClient())

        {

            try

            {

                c.Encoding = System.Text.Encoding.UTF8;

                byte[] data = System.Text.Encoding.UTF8.GetBytes(parameters);

                c.Headers.Add("Content-Type", "application/x-www-form-urlencoded");

                c.Headers.Add("ContentLength", parameters.Length.ToString());

                ret = System.Text.Encoding.UTF8.GetString(c.UploadData(url, "POST", data));

                return ret;

            }

            catch (Exception)

            {

                throw;

            }

        }

    }

    public static string GetPost(string url, NameValueCollection nv)

    {

        using (WebClient c = new WebClient())

        {

            try

            {

                c.Encoding = System.Text.Encoding.UTF8;

                return System.Text.Encoding.UTF8.GetString(c.UploadValues(url, "POST", nv));

            }

            catch (Exception)

            {

                throw;

            }

        }

    }

}

第三個類:封裝好微信基礎頁,方便其他頁面繼承呼叫

using System;

using System.Collections.Generic;

using System.Linq;

using System.Web;

using Newtonsoft.Json;

/// <summary>

/// WeiXin 的摘要說明

/// </summary>

public abstract class WeiXinPage : System.Web.UI.Page, System.Web.SessionState.IRequiresSessionState

{

    protected sealed override void OnPreInit(EventArgs e)

    {

        if (isGuest)

        {

            getWxUser();

        }

        base.OnPreInit(e);

    }

    /// <summary>

    /// 是否訪客身份 true:是 false:否

    /// </summary>

    public static bool isGuest

    {

        get

        {

            return HttpContext.Current.Session[_user] == null ? true : false;

        }

    }

    private static string _user { get { return "$wxuser"; } }

    private static string _code { get { return "$wxcode"; } }

    private static string _AbsoluteUri { get { return HttpContext.Current.Request.Url.AbsoluteUri; } }

    /// <summary>

    /// 該路徑剔除微信Query的URL用於頁面重定向獲取最新的使用者資訊

    /// </summary>

    private static string _RawUrl

    {

        get

        {

            string url = HttpContext.Current.Request.Url.AbsoluteUri;

            url = url.Replace(url.Substring(url.IndexOf("code"), url.IndexOf("STATE") + 5 - url.IndexOf("code")), "");

            return url;

        }

    }

    /// <summary>

    /// 獲取微信使用者資訊

    /// </summary>

    private static Dictionary<string, object> getWxUser()

    {

        string code = HttpContext.Current.Request.QueryString.Get("code") ?? "",

               url = "",

               ret = "";

        try

        {

            object user = HttpContext.Current.Session[_user];

            if (code.Length == 0 && user == null)

            {

                /* 第一步: 使用者同意授權,獲取code*/

                JumpUrl();

                return null;

            }

            if (user != null) return JsonConvert.DeserializeObject<Dictionary<string, object>>(HttpContext.Current.Session[_user].ToString());

            #region /*如果session中不存在user資訊則對接微信介面獲取使用者資訊*/

            /* 第二步:通過code換取網頁授權access_token */

            url = string.Format("https://api.weixin.qq.com/sns/oauth2/access_token?appid={0}&secret={1}&code={2}&grant_type=authorization_code"

                           , Config.AppId

                           , Config.AppSecret

                           , code);

            ret = HttpService.Get(url);

            Dictionary<string, string> o = JsonConvert.DeserializeObject<Dictionary<string, string>>(ret);

            /* 第三步:根據access_token和openid拉取使用者資訊 */

            url = string.Format("https://api.weixin.qq.com/sns/userinfo?access_token={0}&openid={1}&lang=zh_CN"

                                , o["access_token"]

                                , o["openid"]);

            ret = HttpService.Get(url);

            /*儲存使用者到Session*/

            HttpContext.Current.Session[_user] = ret;

            #endregion

            return JsonConvert.DeserializeObject<Dictionary<string, object>>(HttpContext.Current.Session[_user].ToString());

        }

        catch

        {

            throw;

        }

    }

    /// <summary>

    /// 跳轉至微信授權登入頁

    /// </summary>

    private static void JumpUrl()

    {

        string url = string.Format("https://open.weixin.qq.com/connect/oauth2/authorize?appid={0}&redirect_uri={1}&response_type=code&scope=snsapi_userinfo&state=STATE#wechat_redirect"

                                  , Config.AppId

                                  , _AbsoluteUri);

        HttpContext.Current.Response.Redirect(url, false);

    }

    /// <summary>

    /// 微信使用者資訊

    /// </summary>

    public Dictionary<string, object> Data

    {

        get

        {

            return getWxUser();

        }

    }

}

演示:

public partial class _Default : WeiXinPage

{

    protected override void OnLoad(EventArgs e)

    {

        Response.Write(JsonConvert.SerializeObject(Data));

        base.OnLoad(e);

    }

}