一聚教程网:一个值得你收藏的教程网站

最新下载

热门教程

asp.net利用api获取用户内外网IP地址与所在城市

时间:2013-05-30 编辑:简简单单 来源:一聚教程网


1、内网IP直接在后台取
2、外网IP可以通过新浪API http://counter.sina.com.cn/ip 取得,原来也可以返回城市的,后台不知道什么原因,只能返回IP了
3、所在城市通过百度API http://api.map.baidu.com/location/ip?ak=&ip=取得,但是这个不会返回外网IP所以我就两个一起用了,挺蛋疼的。
以上在客户端去访问相应的API又存在一个跨域的问题,通过调查发现百度API支持JSONP,可以很好的解决跨域的问题,新浪API不支持但它返回一个变量,我们可以直接把新浪API写在页面srcipt中即可取得相应变量。
技术都应该没问题了,那我们开始写吧。

具体实现
第一步:在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 = "建筑材料管理系统";
            ViewBag.EnName = "Engineering Material Mangange System";
            return View();
        }
    }
}

类要用AllowAnonymous属性修饰,才能保证未登陆也能够访问。


第二步:添加对应的View,添加~/Views/Login/Index.cshtml,代码如下

 

 代码如下 复制代码
@{
    ViewBag.Title = "登录系统";
    Layout = null;
}


   
        @ViewBag.Title
       
       
       
       
       
       
   
   
       

           

               
               
@ViewBag.CnName

               
@ViewBag.EnName

               

               
                   
                       
                       
                   
                   
                       
                       
                   
                   
                       
                       
                   
                   
                       
                   
               
用户名:
密码:word" class="login" data-bind="value:form.password" />
checkbox" data-bind="checked:form.remember" />系统记住我

                           
                           
                       

           

       

   

1、脚本的最后一个即添加新浪API获取外网IP信息,它返回的数据格式为
var ILData = new Array("117.30.94.103","保留地址", "", "", ""); if (typeof(ILData_callback) != "undefined") { ILData_callback(); }
它其实也有一个callback函数,和JSONP类似,但函数名是固定的,并且没有传递数据。我们可以直接访问ILData[0]取得外网IP。
2、上面html中的data-bind=””写法为knouckoutjs的写法,用于绑定到viewModel的属性

 

第三步:创建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("登陆成功正在跳转,请稍候...");
                    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("正在登陆处理,请稍候...");
            },
            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部分其实可以不放到viewModel中。
1、$.getJSON即为JSONP的访问,其中加上了参数callback=?,jquery会自动处理成当前的回调函数,即跨域成功后会自动回调当前函数并传入数据。我们用viewModel中的form.city接收请求的数据中的城市信息。
2、最后一句ko.applyBindings(new viewModel())即实现了页面和viewModel的绑定,至此,前台全部完成。接下来写登陆处理doAction,还是放在LoginController中,访问地址为/login/doAction。

第四步:在LoginController中添加doAction的方法返回JSON数据。代码如下:

 代码如下 复制代码


        public JsonResult DoAction(JObject request)
        {
            var UserCode = request.Value("usercode");
            var Password = request.Value("password");

            if (String.IsNullOrEmpty(UserCode) || String.IsNullOrEmpty(Password))
                return Json(new { status="error",message="用户名或密码不能为空!" });

            var service = new sys_userService();
            var result = service.GetModel(ParamQuery.Instance()
                            .AndWhere("UserCode", UserCode)
                            .AndWhere("Password", ZEncypt.MD5(Password))
                            .AndWhere("IsEnable",true));

            if (result == null || String.IsNullOrEmpty(result.UserCode))
                return Json(new { status="error",message="用户名或密码不正确!" });

            var loginer = new LoginerBase { UserCode = result.UserCode,UserName=result.UserName };
            FormsAuth.SignIn(loginer.UserCode, loginer, 60 * 8);       //调用框架中的登陆机制 登陆时间设置为8小时
 
            new sys_loginHistoryService().AppendLoginHistory(request);  //登陆后添加登陆履历
            service.UpdateUserLoginCountAndDate(UserCode);              //登陆后更新用户登陆次数及时间
            MmsService.LoginHandler(this);                              //其它的业务处理

            return Json(new { status="success",message="登陆成功!" },JsonRequestBehavior.DenyGet);
        }

接收参数定义为JObject对象比较方便取得请求数据,这当中用到了两个数据服务,一个为sys_userService和sys_loginHistoryService,数据服务中的GetModel是服务基类中已有的,但AppendLoginHistory和UpdateUserLoginCountAndDate要自己写代码如下

 代码如下 复制代码


using System;
using System.Collections.Generic;
using System.Text;
using Zephyr.Core;
namespace Zephyr.Models
{
    public class sys_userService : ServiceBase
    {
        public void UpdateUserLoginCountAndDate(string UserCode)
        {
            db.Sql(@"
update sys_user
set LoginCount = isnull(LoginCount,0) + 1
   ,LastLoginDate = getdate()
where UserCode = @0 "
                ,UserCode).Execute();
        }
    }}
using System;
using System.Collections.Generic;
using System.Text;
using Zephyr.Core;
using Zephyr.Utils;

namespace Zephyr.Models
{
    public class sys_loginHistoryService : ServiceBase
    {
        public void AppendLoginHistory(JObject request)
        {
            var lanIP = ZHttp.ClientIP;
            var hostName = ZHttp.IsLanIP(lanIP) ? ZHttp.ClientHostName : string.Empty; //如果是内网就获取,否则出错获取不到,且影响效率

            var UserCode = request.Value("usercode");
            var UserName = MmsHelper.GetUserName();            var IP = request.Value("ip");
            var City = request.Value("city");
            if (IP!=lanIP)
                IP = string.Format("{0}/{1}", IP, lanIP).Trim('/').Replace("::1", "localhost");//拼接内外网IP            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", item).AutoMap(x=>x.ID).Execute();
        }
    }}

至此已大功告成!

热门栏目