Asp.net MVC利用knockoutjs實(shí)現(xiàn)登陸并記錄用戶的內(nèi)外網(wǎng)IP及所在城市(推薦)
前言
前面第一篇開了頭個(gè),現(xiàn)在想先從登陸寫起,但感覺還有很多東西應(yīng)該放在前面寫,比如
1、MVC及Web API的Route配置,Web API的Route配置如何支持命名空間
2、如何配置Filters(實(shí)現(xiàn)安全驗(yàn)證、錯(cuò)誤處理等等)
3、自定義Filters、HttpRouteConstraint、ModelBinder及HttpParameterBinding等
這些問題在我開發(fā)過程中都有碰到,但感覺每一點(diǎn)都要說太多了。如果有需要到時(shí)候再回過頭來寫。
需求
還是老樣子,我們先要明白要登陸實(shí)現(xiàn)哪些東西:
1、登陸頁(yè)面(用戶名、密碼、記住我、登陸按鈕、重置按鈕)
2、消息顯示(比如 錯(cuò)誤時(shí)顯示某某錯(cuò)誤,登陸時(shí)顯示正在登陸,登陸成功顯示正在跳轉(zhuǎn)等)
3、登陸處理(驗(yàn)證、登陸、正在登陸時(shí)禁用表單、更新用戶登陸次數(shù)及時(shí)間、添加登陸履歷其中要包括用戶的內(nèi)網(wǎng)IP外網(wǎng)IP還有所在城市、其它業(yè)務(wù)處理)
4、成功跳轉(zhuǎn)
實(shí)現(xiàn)效果
在實(shí)現(xiàn)之前我們先看看實(shí)現(xiàn)出來的效果截圖
登陸頁(yè)面
跳轉(zhuǎn)頁(yè)面
登陸履歷
需求分析及實(shí)現(xiàn)
需求中基本都好實(shí)現(xiàn),只有登陸履歷中要記錄內(nèi)外網(wǎng)IP及所在城市要考慮一下。在asp.NET中取得客戶端內(nèi)外網(wǎng)IP還是比較麻煩的,而要取得所在城市就基本不可能了,所以我們只好考慮借助第三方api去實(shí)現(xiàn)了。
1、內(nèi)網(wǎng)IP直接在后臺(tái)取
2、外網(wǎng)IP可以通過新浪API http://counter.sina.com.cn/ip 取得,原來也可以返回城市的,后臺(tái)不知道什么原因,只能返回IP了
3、所在城市通過百度API http://api.map.baidu.com/location/ip?ak=&ip=取得,但是這個(gè)不會(huì)返回外網(wǎng)IP所以我就兩個(gè)一起用了,挺蛋疼的。
以上在客戶端去訪問相應(yīng)的API又存在一個(gè)跨域的問題,通過調(diào)查發(fā)現(xiàn)百度API支持JSONP,可以很好的解決跨域的問題,新浪API不支持但它返回一個(gè)變量,我們可以直接把新浪API寫在頁(yè)面srcipt中即可取得相應(yīng)變量。
技術(shù)都應(yīng)該沒問題了,那我們開始寫吧。
具體實(shí)現(xiàn)
第一步:在MVC中新建LoginController添加如下代碼
using System; using System.Web.Mvc; using Newtonsoft.Json; using Newtonsoft.Json.Linq; using Zephyr.Core; using Zephyr.Models; using Zephyr.Web.Areas.Mms.Common; namespace Zephyr.Controllers { [AllowAnonymous] public class LoginController : Controller { public ActionResult Index() { ViewBag.CnName = "建筑材料管理系統(tǒng)"; ViewBag.EnName = "Engineering Material Mangange System"; return View(); } } }
類要用AllowAnonymous屬性修飾,才能保證未登陸也能夠訪問。
第二步:添加對(duì)應(yīng)的View,添加~/Views/Login/Index.cshtml,代碼如下
@{ ViewBag.Title = "登錄系統(tǒng)"; Layout = null; } <!doctype html> <html> <head> <title>@ViewBag.Title</title> <link href="~/Content/css/page/login.css" rel="stylesheet" type="text/css" /> <script src="~/Content/js/jquery/jquery-1.8.1.min.js"></script> <script src="~/Content/js/core/json2.js"></script> <script src="~/Content/js/core/knockout-2.2.1.js"></script> <script src="~/Content/js/viewModel/login.js"></script> <script src="http://counter.sina.com.cn/ip"></script> </head> <body> <div class="second_body"> <form data-bind="submit:loginClick"> <div class="logo"><img src="/Content/images/login/logo.png" alt="" /></div> <div class="title-zh">@ViewBag.CnName</div> <div class="title-en" style="@ViewBag.EnNameStyle">@ViewBag.EnName</div> <div class="message" data-bind="html:message"></div> <table border="0" style="width:300px;"> <tr> <td style="padding-bottom: 5px;width:55px;">用戶名:</td> <td colspan="2"><input type="text" class="login" data-bind="value:form.usercode" /></td> </tr> <tr> <td class="lable" style="letter-spacing: 0.5em; vertical-align: middle">密碼:</td> <td colspan="2"><input type="password" class="login" data-bind="value:form.password" /></td> </tr> <tr> <td></td> <td colspan="2"><input type="checkbox" data-bind="checked:form.remember" /><span>系統(tǒng)記住我</span></td> </tr> <tr> <td colspan="3" style="text-align:center"> <input type="submit" value="登錄" class="login_button" /> <input type="button" value="重置" class="reset_botton" data-bind="click:resetClick" /> </td> </tr> </table> </form> </div> </body> </html>
1、腳本的最后一個(gè)即添加新浪API獲取外網(wǎng)IP信息,它返回的數(shù)據(jù)格式為
var ILData = new Array("117.30.94.103","保留地址", "", "", ""); if (typeof(ILData_callback) != "undefined") { ILData_callback(); }
它其實(shí)也有一個(gè)callback函數(shù),和JSONP類似,但函數(shù)名是固定的,并且沒有傳遞數(shù)據(jù)。我們可以直接訪問ILData[0]取得外網(wǎng)IP。
2、上面html中的data-bind=””寫法為knouckoutjs的寫法,用于綁定到viewModel的屬性
第三步:創(chuàng)建ViewModel
var viewModel = function () { var self = this; this.form = { usercode: ko.observable(), password: ko.observable(), remember:ko.observable(false), ip: null, city: null }; this.message = ko.observable(); this.loginClick = function (form) { $.ajax({ type: "POST", url: "/login/doAction", data: ko.toJSON(self.form), dataType: "json", contentType: "application/json", success: function (d) { if (d.status == 'success') { self.message("登陸成功正在跳轉(zhuǎn),請(qǐng)稍候..."); window.location.href = '/'; } else { self.message(d.message); } }, error: function (e) { self.message(e.responseText); }, beforeSend: function () { $(form).find("input").attr("disabled", true); self.message("正在登陸處理,請(qǐng)稍候..."); }, complete: function () { $(form).find("input").attr("disabled", false); } }); }; this.resetClick = function () { self.form.usercode(""); self.form.password(""); self.form.remember(false); }; this.init = function () { self.form.ip = ILData[0]; $.getJSON("http://api.map.baidu.com/location/ip?ak=F454f8a5efe5e577997931cc01de3974&callback=?", function (d) { self.form.city = d.content.address; }); if (top != window) top.window.location = window.location; }; this.init(); }; $(function () { ko.applyBindings(new viewModel());});
定義viewModel,其屬性包括from表單信息,message提示信息,loginClick登陸,resetClick重置。其中的init部分其實(shí)可以不放到viewModel中。
1、$.getJSON即為JSONP的訪問,其中加上了參數(shù)callback=?,jQuery會(huì)自動(dòng)處理成當(dāng)前的回調(diào)函數(shù),即跨域成功后會(huì)自動(dòng)回調(diào)當(dāng)前函數(shù)并傳入數(shù)據(jù)。我們用viewModel中的form.city接收請(qǐng)求的數(shù)據(jù)中的城市信息。
2、最后一句ko.applyBindings(new viewModel())即實(shí)現(xiàn)了頁(yè)面和viewModel的綁定,至此,前臺(tái)全部完成。接下來寫登陸處理doAction,還是放在LoginController中,訪問地址為/login/doAction。
第四步:在LoginController中添加doAction的方法返回JSON數(shù)據(jù)。代碼如下:
public JsonResult DoAction(JObject request) { var message = new sys_userService().Login(request); return Json(message, JsonRequestBehavior.DenyGet); }
然后在service層中處理
using System; using System.Collections.Generic; using Zephyr.Core; using System.Dynamic; using Newtonsoft.Json.Linq; using Newtonsoft.Json; using Zephyr.Utils; using Zephyr.Web.Areas.Mms.Common; namespace Zephyr.Models { public class sys_userService : ServiceBase<sys_user> { public object Login(JObject request) { var UserCode = request.Value<string>("usercode"); var Password = request.Value<string>("password"); //用戶名密碼檢查 if (String.IsNullOrEmpty(UserCode) || String.IsNullOrEmpty(Password)) return new { status = "error", message = "用戶名或密碼不能為空!" }; //用戶名密碼驗(yàn)證 var result = this.GetModel(ParamQuery.Instance() .AndWhere("UserCode", UserCode) .AndWhere("Password", Password) .AndWhere("IsEnable", true)); if (result == null || String.IsNullOrEmpty(result.UserCode)) return new { status = "error", message = "用戶名或密碼不正確!" }; //調(diào)用框架中的登陸機(jī)制 var loginer = new LoginerBase { UserCode = result.UserCode, UserName = result.UserName }; FormsAuth.SignIn(loginer.UserCode, loginer, 60 * 8); //登陸后處理 this.UpdateUserLoginCountAndDate(UserCode); //更新用戶登陸次數(shù)及時(shí)間 this.AppendLoginHistory(request); //添加登陸履歷 MmsService.LoginHandler(request); //MMS系統(tǒng)的其它的業(yè)務(wù)處理 //返回登陸成功 return new { status = "success", message = "登陸成功!" }; } //更新用戶登陸次數(shù)及時(shí)間 public void UpdateUserLoginCountAndDate(string UserCode) { db.Sql(@" update sys_user set LoginCount = isnull(LoginCount,0) + 1 ,LastLoginDate = getdate() where UserCode = @0 " , UserCode).Execute(); } //添加登陸履歷 public void AppendLoginHistory(JObject request) { var lanIP = ZHttp.ClientIP; var hostName = ZHttp.IsLanIP(lanIP) ? ZHttp.ClientHostName : string.Empty; //如果是內(nèi)網(wǎng)就獲取,否則出錯(cuò)獲取不到,且影響效率 var UserCode = request.Value<string>("usercode"); var UserName = MmsHelper.GetUserName(); var IP = request.Value<string>("ip"); var City = request.Value<string>("city"); if (IP != lanIP) IP = string.Format("{0}/{1}", IP, lanIP).Trim('/').Replace("::1", "localhost"); var item = new sys_loginHistory(); item.UserCode = UserCode; item.UserName = UserName; item.HostName = hostName; item.HostIP = IP; item.LoginCity = City; item.LoginDate = DateTime.Now; db.Insert<sys_loginHistory>("sys_loginHistory", item).AutoMap(x => x.ID).Execute(); } } }
接收參數(shù)定義為JObject對(duì)象比較方便取得請(qǐng)求數(shù)據(jù),數(shù)據(jù)服務(wù)中的GetModel是服務(wù)基類中已有的方法,這當(dāng)中用到了兩個(gè)函數(shù),一個(gè)為UpdateUserLoginCountAndDate為更新用戶登陸次數(shù)及時(shí)間的處理,另一個(gè)AppendLoginHistory添加登陸履歷。至此已大功告成!
以上所述是小編給大家介紹的Asp.net MVC利用knockoutjs實(shí)現(xiàn)登陸并記錄用戶的內(nèi)外網(wǎng)IP及所在城市(推薦),希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- knockoutjs模板實(shí)現(xiàn)樹形結(jié)構(gòu)列表
- 使用asp.net mvc,boostrap及knockout.js開發(fā)微信自定義菜單編輯工具(推薦)
- KnockoutJS 3.X API 第四章之表單textInput、hasFocus、checked綁定
- KnockoutJS 3.X API 第四章之?dāng)?shù)據(jù)控制流foreach綁定
- KnockoutJS 3.X API 第四章之?dāng)?shù)據(jù)控制流with綁定
- KnockoutJS 3.X API 第四章之事件event綁定
- KnockoutJS 3.X API 第四章之表單value綁定
- BootstrapTable+KnockoutJS相結(jié)合實(shí)現(xiàn)增刪改查解決方案(三)兩個(gè)Viewmodel搞定增刪改查
- KnockoutJS數(shù)組比較算法實(shí)例詳解
相關(guān)文章
.NET 刷新頁(yè)面防止表單二次提交的實(shí)現(xiàn)方法
頁(yè)面上按鈕是服務(wù)器控件,現(xiàn)在刷新頁(yè)面要防止按鈕事件重復(fù)執(zhí)行。這篇文章給大家?guī)砹?net刷新頁(yè)面防止表單二次提交的實(shí)現(xiàn)方法,非常不錯(cuò),感興趣的朋友一起看看吧2016-09-09JavaScript用JQuery呼叫Server端方法實(shí)現(xiàn)代碼與參考語(yǔ)法
從Javascript客戶端用JQuery呼叫Server端的方法,這也是一個(gè)大膽的嘗試,本人做了演示動(dòng)畫以及參考語(yǔ)法,感興趣的朋友可以參考下,希望本人對(duì)你有所幫助2013-01-01ASP.NET中內(nèi)嵌頁(yè)面代碼的一個(gè)問題
如果頁(yè)面繼承了一個(gè)父頁(yè)面,而且當(dāng)前頁(yè)面用new覆蓋了父頁(yè)面的某個(gè)屬性2010-02-02asp.net獲取當(dāng)前網(wǎng)址url的各種屬性(文件名、參數(shù)、域名 等)的代碼
用 asp.net獲取當(dāng)前網(wǎng)頁(yè)地址的一些信息。2010-07-07asp.net+jquery ajax無刷新登錄的實(shí)現(xiàn)方法
asp.net+jquery ajax無刷新登錄的實(shí)現(xiàn)方法,需要的朋友可以參考一下2013-06-06三層+存儲(chǔ)過程實(shí)現(xiàn)分頁(yè)示例代碼
本文為大家介紹下三層+存儲(chǔ)過程分頁(yè)的具體實(shí)現(xiàn),前后臺(tái)及存儲(chǔ)過程代碼如下,感興趣的朋友可以參考下哈2013-06-06ASP.net中Core自定義View查找位置的實(shí)例代碼
在本篇文章里小編給大家分享的是關(guān)于ASP.net中Core自定義View查找位置的實(shí)例代碼,需要的朋友們可以學(xué)習(xí)下。2020-04-04