ToB企服应用市场:ToB评测及商务社交产业平台

标题: SignalR实战:在.NET Framework和.NET Core中如何使用SignalR? [打印本页]

作者: 张春    时间: 2023-8-31 04:16
标题: SignalR实战:在.NET Framework和.NET Core中如何使用SignalR?
官网文档:https://learn.microsoft.com/zh-cn/aspnet/core/tutorials/signalr?view=aspnetcore-6.0&tabs=visual-studio
SignalR开源代码:https://github.com/signalr

很多小伙伴问:在前后端分离项目中,后端是.NET Core前端是Vue如何使用SignalR?在前后端不分离项目中,.NET Framework MVC项目中又如何使用SignalR技术呢?那就来看看下面这篇文章吧!本文主要介绍SignalR在实际项目中的应用,以及.NET Framework和.NET Core中如何去使用SignalR。

一、SignalR介绍

1.1-SignalR介绍

ASP.NET Core SignalR是一个开放源代码库,可用于简化向应用添加实时Web功能,实时Web功能使服务器端代码能够将内容推送到客户端。

1.2-SignalR的应用


二、.NET Framework使用SignalR

参考文献:
Owin介绍:https://www.cnblogs.com/Pinapple/p/6721361.html
Owin相关案例:https://blog.csdn.net/bifan546/article/details/77098896/

2.1-服务端(.NET Framework MVC)

(1)选择要使用Signalr的项目,点击【管理NuGet程序包】。

(2)搜索【SignalR】包,找到这个后然后点击下载。

会自动安装四个,注意他们之间有依赖关系:

(3).NET Framework平台需要添加Owin相关的包。
OWIN 是一种定义 Web 服务器和应用程序组件之间的交互的规范 。这一规范的目的是发展一个广阔且充满活力的、基于 Microsoft .NET Framework 的 Web 服务器和应用程序组件生态系统。
Microsoft.Owin.Hosting
Microsoft.Owin.Cors
Microsoft.Owin.Host.HttpListener
(4)在Web项目(要使用的Signal)中创建一个【Startup.cs】文件。
  1. using JiCai.Admin.Hubs;
  2. using Microsoft.Owin;
  3. using Owin;
  4. [assembly: OwinStartup(typeof(JiCai.Admin.Startup))]
  5. namespace JiCai.Admin
  6. {
  7.     public class Startup
  8.     {
  9.         /// <summary>
  10.         /// 应用程序配置
  11.         /// </summary>
  12.         /// <param name="app"></param>
  13.         public void Configuration(IAppBuilder app)
  14.         {
  15.             //启用SignalR
  16.             app.MapSignalR();
  17.             //绑定多个Hub
  18.             app.MapSignalR<DemoHub>("/demoHub");
  19.         }
  20.     }
  21. }
复制代码
例如:

(5)可以创建一个【Hubs】文件夹,专门放置Hub相关文件。[西瓜程序猿]这边是创建一个Hub文件,名为【DemoHub.cs】。
  1. using Fenqibao.DTO;
  2. using JiCai.Admin.Hubs.ConectionOperate;
  3. using JiCai.Admin.Hubs.Models;
  4. using log4net;
  5. using Microsoft.AspNet.SignalR;
  6. using Microsoft.AspNet.SignalR.Json;
  7. using Newtonsoft.Json;
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Linq;
  11. using System.Threading.Tasks;
  12. namespace JiCai.Admin.Hubs
  13. {
  14.     /// <summary>
  15.     /// 报表管理-总表Hub
  16.     /// </summary>
  17.     public class SummaryTableHub :  PersistentConnection
  18.     {
  19.         public readonly BaseService _base = new BaseService();
  20.         private readonly ILog logger = LogManager.GetLogger(typeof(SummaryTableHub));
  21.         private ConnectionManagement summaryTableCon = new ConnectionManagement();
  22.         public CookieUserData LoginUserData
  23.         {
  24.             get
  25.             {
  26.                 IOperator oper = ContainerManager.Resolve<IOperator>();
  27.                 LoginUser loginUser = oper as LoginUser;
  28.                 if (loginUser != null && loginUser.UserData != null)
  29.                 {
  30.                     return loginUser.UserData;
  31.                 }
  32.                 return null;
  33.             }
  34.         }
  35.         /// <summary>
  36.         /// 连接成功后调用
  37.         /// </summary>
  38.         /// <param name="request"></param>
  39.         /// <param name="connectionId"></param>
  40.         /// <returns></returns>
  41.         protected override Task OnConnected(IRequest request, string connectionId)
  42.         {
  43.             //获得SignalR的连接id
  44.             var connid = connectionId;
  45.             //获得用户id
  46.             var userid = LoginUserData.Id.ToString();
  47.             Console.Write($"【{connid}】:已建立连接!");
  48.             //判断一下用户是不是已经链接了
  49.             var checkUserConn = summaryTableCon.IsConn(connid, userid);
  50.             if (!checkUserConn)
  51.             {
  52.                 //添加一个新的连接
  53.                 summaryTableCon.AddConnInfo(new SignalRConn()
  54.                 {
  55.                     UserId = userid,
  56.                     ConnectionId = connid
  57.                 });
  58.             }
  59.             //更新连接
  60.             else
  61.             {
  62.                 summaryTableCon.UpdateConnInfo(userid, connid);
  63.             }
  64.             return Connection.Send(connectionId, $"【用户:{connectionId}】真正连接成功!");
  65.             //return base.OnConnected(request, connectionId);
  66.         }
  67.         /// <summary>
  68.         /// 接收到请求的时候调用
  69.         /// </summary>
  70.         /// <param name="request"></param>
  71.         /// <param name="connectionId"></param>
  72.         /// <param name="data"></param>
  73.         /// <returns></returns>
  74.         protected override async Task OnReceived(IRequest request, string connectionId, string data)
  75.         {
  76.             //获得用户id
  77.             var userid = LoginUserData.Id.ToString();
  78.             await Task.Factory.StartNew(async () =>
  79.             {
  80.                 while (true)
  81.                 {
  82.                                         var list = GetSummaryTableList(userid);
  83.                     string json_jieshou_mes = "";
  84.                     if (list != null && list.Count > 0)
  85.                     {
  86.                                                 json_jieshou_mes = JsonConvert.SerializeObject(list);
  87.                     }
  88.                     await Connection.Send(connectionId, json_jieshou_mes);
  89.                     //每5秒同步一次
  90.                     await Task.Delay(5000);
  91.                 }
  92.             }, TaskCreationOptions.LongRunning);
  93.             //return base.OnReceived(request, connectionId, data);
  94.         }
  95.         /// <summary>
  96.         /// 连接中断的时候调用
  97.         /// </summary>
  98.         /// <param name="request"></param>
  99.         /// <param name="connectionId"></param>
  100.         /// <param name="stopCalled"></param>
  101.         /// <returns></returns>
  102.         protected override Task OnDisconnected(IRequest request, string connectionId, bool stopCalled)
  103.         {
  104.             Console.Write($"【{connectionId}】:已断开连接!");
  105.             //获得SignalR的连接id
  106.             var connid = connectionId;
  107.             //关闭连接
  108.             summaryTableCon.DelConnInfo(connid);
  109.             return base.OnDisconnected(request, connectionId, stopCalled);
  110.         }
  111.         /// <summary>
  112.         /// 连接超时重新连接的时候调用
  113.         /// </summary>
  114.         /// <param name="request"></param>
  115.         /// <param name="connectionId"></param>
  116.         /// <returns></returns>
  117.         protected override Task OnReconnected(IRequest request, string connectionId)
  118.         {
  119.             return base.OnReconnected(request, connectionId);
  120.         }
  121.                 /// <summary>
  122.                 /// 查询数据
  123.                 /// </summary>
  124.                 /// <param name="userId"></param>
  125.                 /// <returns></returns>
  126.                 private List<SummaryTableDataModel> GetSummaryTableList(string userId)
  127.         {
  128.             var result = _base.Query<SummaryTableDataModel>($@"
  129.                             select * from demo-data
  130.                                         ;
  131.             ").ToList();
  132.                         return result;
  133.         }
  134.     }
  135. }
复制代码
(6)在Hubs/ConectionOperate文件夹中,[西瓜程序猿]这边创建【ConnectionManagement】文件,用来管理所有连接。
  1. using System.Collections.Generic;
  2. using System.Linq;
  3. namespace JiCai.Admin.Hubs.ConectionOperate
  4. {
  5.     /// <summary>
  6.     /// 连接管理
  7.     /// </summary>
  8.     public class ConnectionManagement
  9.     {
  10.         /// <summary>
  11.         /// 用户连接集合
  12.         /// </summary>
  13.         public static  List<SignalRConn> SignalRConns { get; set; } = new List<SignalRConn>();
  14.         /// <summary>
  15.         /// 添加连接
  16.         /// </summary>
  17.         /// <param name="conn"></param>
  18.         public  void AddConnInfo(SignalRConn conn)
  19.         {
  20.             SignalRConns.Add(conn);
  21.         }
  22.         /// <summary>
  23.         /// 删除连接
  24.         /// </summary>
  25.         /// <param name="connid"></param>
  26.         public  void DelConnInfo(string connid)
  27.         {
  28.             var signalRConns = SignalRConns.FirstOrDefault(u => u.ConnectionId == connid);
  29.             if (signalRConns != null)
  30.             {
  31.                 SignalRConns.Remove(signalRConns);
  32.             }
  33.         }
  34.         /// <summary>
  35.         /// 更新链接(老的链接不起作用了)
  36.         /// 场景:客户端重连了,userid没变,但是connid变了
  37.         /// </summary>
  38.         /// <param name="userId">用户id</param>
  39.         /// <param name="newConnsId">新的链接id</param>
  40.         public  void UpdateConnInfo(string userId, string newConnsId)
  41.         {
  42.             var signalRConns = SignalRConns.FirstOrDefault(u => u.UserId.ToLower() == userId.ToLower());
  43.             if (signalRConns != null)
  44.             {
  45.                 signalRConns.ConnectionId = newConnsId;
  46.             }
  47.         }
  48.         /// <summary>
  49.         /// 判断用户是否已经链接
  50.         /// </summary>
  51.         /// <param name="connid">连接id</param>
  52.         /// <param name="userid">用户id</param>
  53.         /// <returns></returns>
  54.         public bool IsConn(string connid,string userid)
  55.         {
  56.             var userConn = SignalRConns.FirstOrDefault(u => u.ConnectionId.ToLower() == connid.ToLower() && u.UserId.ToLower() == userid.ToLower());
  57.             return userConn == null ? false : true;
  58.         }
  59.     }
  60. }
复制代码
(7)在Hubs/ConectionOperate文件夹中,创建【SignalRConn.cs】文件用来作为SignalR和系统用户的连接实体。
  1. namespace JiCai.Admin.Hubs.ConectionOperate
  2. {
  3.     /// <summary>
  4.     /// 连接
  5.     /// </summary>
  6.     public class SignalRConn
  7.     {
  8.         /// <summary>
  9.         /// 系统用户id
  10.         /// </summary>
  11.         public string UserId { get; set; }
  12.         /// <summary>
  13.         /// SignleR链接Id(每次链接SignalR都会分配一个id)
  14.         /// </summary>
  15.         public string ConnectionId { get; set; }
  16.     }
  17. }
复制代码
2.2-客户端(JS)

(1)下载相关jq/signalr相关包,可以自行找CDN直接访问。也可以自行新建这2个文件,然后复制一下内容。
【jquery.signalR-2.4.3.js】:
  1. /* jquery.signalR.core.js */
  2. /*global window:false */
  3. /*!
  4. * ASP.NET SignalR JavaScript Library 2.4.3
  5. * http://signalr.net/
  6. *
  7. * Copyright (c) .NET Foundation. All rights reserved.
  8. * Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  9. *
  10. */
  11. /// <reference path="Scripts/jquery-1.6.4.js" />
  12. /// <reference path="jquery.signalR.version.js" />
  13. (function ($, window, undefined) {
  14.     var resources = {
  15.         nojQuery: "jQuery was not found. Please ensure jQuery is referenced before the SignalR client JavaScript file.",
  16.         noTransportOnInit: "No transport could be initialized successfully. Try specifying a different transport or none at all for auto initialization.",
  17.         errorOnNegotiate: "Error during negotiation request.",
  18.         stoppedWhileLoading: "The connection was stopped during page load.",
  19.         stoppedWhileNegotiating: "The connection was stopped during the negotiate request.",
  20.         errorParsingNegotiateResponse: "Error parsing negotiate response.",
  21.         errorRedirectionExceedsLimit: "Negotiate redirection limit exceeded.",
  22.         errorDuringStartRequest: "Error during start request. Stopping the connection.",
  23.         errorFromServer: "Error message received from the server: '{0}'.",
  24.         stoppedDuringStartRequest: "The connection was stopped during the start request.",
  25.         errorParsingStartResponse: "Error parsing start response: '{0}'. Stopping the connection.",
  26.         invalidStartResponse: "Invalid start response: '{0}'. Stopping the connection.",
  27.         protocolIncompatible: "You are using a version of the client that isn't compatible with the server. Client version {0}, server version {1}.",
  28.         aspnetCoreSignalrServer: "Detected a connection attempt to an ASP.NET Core SignalR Server. This client only supports connecting to an ASP.NET SignalR Server. See https://aka.ms/signalr-core-differences for details.",
  29.         sendFailed: "Send failed.",
  30.         parseFailed: "Failed at parsing response: {0}",
  31.         longPollFailed: "Long polling request failed.",
  32.         eventSourceFailedToConnect: "EventSource failed to connect.",
  33.         eventSourceError: "Error raised by EventSource",
  34.         webSocketClosed: "WebSocket closed.",
  35.         pingServerFailedInvalidResponse: "Invalid ping response when pinging server: '{0}'.",
  36.         pingServerFailed: "Failed to ping server.",
  37.         pingServerFailedStatusCode: "Failed to ping server.  Server responded with status code {0}, stopping the connection.",
  38.         pingServerFailedParse: "Failed to parse ping server response, stopping the connection.",
  39.         noConnectionTransport: "Connection is in an invalid state, there is no transport active.",
  40.         webSocketsInvalidState: "The Web Socket transport is in an invalid state, transitioning into reconnecting.",
  41.         reconnectTimeout: "Couldn't reconnect within the configured timeout of {0} ms, disconnecting.",
  42.         reconnectWindowTimeout: "The client has been inactive since {0} and it has exceeded the inactivity timeout of {1} ms. Stopping the connection.",
  43.         jsonpNotSupportedWithAccessToken: "The JSONP protocol does not support connections that require a Bearer token to connect, such as the Azure SignalR Service."
  44.     };
  45.     if (typeof ($) !== "function") {
  46.         // no jQuery!
  47.         throw new Error(resources.nojQuery);
  48.     }
  49.     var signalR,
  50.         _connection,
  51.         _pageLoaded = (window.document.readyState === "complete"),
  52.         _pageWindow = $(window),
  53.         _negotiateAbortText = "__Negotiate Aborted__",
  54.         events = {
  55.             onStart: "onStart",
  56.             onStarting: "onStarting",
  57.             onReceived: "onReceived",
  58.             onError: "onError",
  59.             onConnectionSlow: "onConnectionSlow",
  60.             onReconnecting: "onReconnecting",
  61.             onReconnect: "onReconnect",
  62.             onStateChanged: "onStateChanged",
  63.             onDisconnect: "onDisconnect"
  64.         },
  65.         ajaxDefaults = {
  66.             processData: true,
  67.             timeout: null,
  68.             async: true,
  69.             global: false,
  70.             cache: false
  71.         },
  72.         log = function (msg, logging) {
  73.             if (logging === false) {
  74.                 return;
  75.             }
  76.             var m;
  77.             if (typeof (window.console) === "undefined") {
  78.                 return;
  79.             }
  80.             m = "[" + new Date().toTimeString() + "] SignalR: " + msg;
  81.             if (window.console.debug) {
  82.                 window.console.debug(m);
  83.             } else if (window.console.log) {
  84.                 window.console.log(m);
  85.             }
  86.         },
  87.         changeState = function (connection, expectedState, newState) {
  88.             if (expectedState === connection.state) {
  89.                 connection.state = newState;
  90.                 $(connection).triggerHandler(events.onStateChanged, [{ oldState: expectedState, newState: newState }]);
  91.                 return true;
  92.             }
  93.             return false;
  94.         },
  95.         isDisconnecting = function (connection) {
  96.             return connection.state === signalR.connectionState.disconnected;
  97.         },
  98.         supportsKeepAlive = function (connection) {
  99.             return connection._.keepAliveData.activated &&
  100.                 connection.transport.supportsKeepAlive(connection);
  101.         },
  102.         configureStopReconnectingTimeout = function (connection) {
  103.             var stopReconnectingTimeout,
  104.                 onReconnectTimeout;
  105.             // Check if this connection has already been configured to stop reconnecting after a specified timeout.
  106.             // Without this check if a connection is stopped then started events will be bound multiple times.
  107.             if (!connection._.configuredStopReconnectingTimeout) {
  108.                 onReconnectTimeout = function (connection) {
  109.                     var message = signalR._.format(signalR.resources.reconnectTimeout, connection.disconnectTimeout);
  110.                     connection.log(message);
  111.                     $(connection).triggerHandler(events.onError, [signalR._.error(message, /* source */ "TimeoutException")]);
  112.                     connection.stop(/* async */ false, /* notifyServer */ false);
  113.                 };
  114.                 connection.reconnecting(function () {
  115.                     var connection = this;
  116.                     // Guard against state changing in a previous user defined even handler
  117.                     if (connection.state === signalR.connectionState.reconnecting) {
  118.                         stopReconnectingTimeout = window.setTimeout(function () { onReconnectTimeout(connection); }, connection.disconnectTimeout);
  119.                     }
  120.                 });
  121.                 connection.stateChanged(function (data) {
  122.                     if (data.oldState === signalR.connectionState.reconnecting) {
  123.                         // Clear the pending reconnect timeout check
  124.                         window.clearTimeout(stopReconnectingTimeout);
  125.                     }
  126.                 });
  127.                 connection._.configuredStopReconnectingTimeout = true;
  128.             }
  129.         };
  130.     signalR = function (url, qs, logging) {
  131.         /// <summary>Creates a new SignalR connection for the given url</summary>
  132.         /// <param name="url" type="String">The URL of the long polling endpoint</param>
  133.         /// <param name="qs" type="Object">
  134.         ///     [Optional] Custom querystring parameters to add to the connection URL.
  135.         ///     If an object, every non-function member will be added to the querystring.
  136.         ///     If a string, it's added to the QS as specified.
  137.         /// </param>
  138.         /// <param name="logging" type="Boolean">
  139.         ///     [Optional] A flag indicating whether connection logging is enabled to the browser
  140.         ///     console/log. Defaults to false.
  141.         /// </param>
  142.         return new signalR.fn.init(url, qs, logging);
  143.     };
  144.     signalR._ = {
  145.         defaultContentType: "application/x-www-form-urlencoded; charset=UTF-8",
  146.         ieVersion: (function () {
  147.             var version,
  148.                 matches;
  149.             if (window.navigator.appName === 'Microsoft Internet Explorer') {
  150.                 // Check if the user agent has the pattern "MSIE (one or more numbers).(one or more numbers)";
  151.                 matches = /MSIE ([0-9]+\.[0-9]+)/.exec(window.navigator.userAgent);
  152.                 if (matches) {
  153.                     version = window.parseFloat(matches[1]);
  154.                 }
  155.             }
  156.             // undefined value means not IE
  157.             return version;
  158.         })(),
  159.         error: function (message, source, context) {
  160.             var e = new Error(message);
  161.             e.source = source;
  162.             if (typeof context !== "undefined") {
  163.                 e.context = context;
  164.             }
  165.             return e;
  166.         },
  167.         transportError: function (message, transport, source, context) {
  168.             var e = this.error(message, source, context);
  169.             e.transport = transport ? transport.name : undefined;
  170.             return e;
  171.         },
  172.         format: function () {
  173.             /// <summary>Usage: format("Hi {0}, you are {1}!", "Foo", 100) </summary>
  174.             var s = arguments[0];
  175.             for (var i = 0; i < arguments.length - 1; i++) {
  176.                 s = s.replace("{" + i + "}", arguments[i + 1]);
  177.             }
  178.             return s;
  179.         },
  180.         firefoxMajorVersion: function (userAgent) {
  181.             // Firefox user agents: http://useragentstring.com/pages/Firefox/
  182.             var matches = userAgent.match(/Firefox\/(\d+)/);
  183.             if (!matches || !matches.length || matches.length < 2) {
  184.                 return 0;
  185.             }
  186.             return parseInt(matches[1], 10 /* radix */);
  187.         },
  188.         configurePingInterval: function (connection) {
  189.             var config = connection._.config,
  190.                 onFail = function (error) {
  191.                     $(connection).triggerHandler(events.onError, [error]);
  192.                 };
  193.             if (config && !connection._.pingIntervalId && config.pingInterval) {
  194.                 connection._.pingIntervalId = window.setInterval(function () {
  195.                     signalR.transports._logic.pingServer(connection).fail(onFail);
  196.                 }, config.pingInterval);
  197.             }
  198.         }
  199.     };
  200.     signalR.events = events;
  201.     signalR.resources = resources;
  202.     signalR.ajaxDefaults = ajaxDefaults;
  203.     signalR.changeState = changeState;
  204.     signalR.isDisconnecting = isDisconnecting;
  205.     signalR.connectionState = {
  206.         connecting: 0,
  207.         connected: 1,
  208.         reconnecting: 2,
  209.         disconnected: 4
  210.     };
  211.     signalR.hub = {
  212.         start: function () {
  213.             // This will get replaced with the real hub connection start method when hubs is referenced correctly
  214.             throw new Error("SignalR: Error loading hubs. Ensure your hubs reference is correct, e.g. .");
  215.         }
  216.     };
  217.     // .on() was added in version 1.7.0, .load() was removed in version 3.0.0 so we fallback to .load() if .on() does
  218.     // not exist to not break existing applications
  219.     if (typeof _pageWindow.on === "function") {
  220.         _pageWindow.on("load", function () { _pageLoaded = true; });
  221.     }
  222.     else {
  223.         _pageWindow.load(function () { _pageLoaded = true; });
  224.     }
  225.     function validateTransport(requestedTransport, connection) {
  226.         /// <summary>Validates the requested transport by cross checking it with the pre-defined signalR.transports</summary>
  227.         /// <param name="requestedTransport" type="Object">The designated transports that the user has specified.</param>
  228.         /// <param name="connection" type="signalR">The connection that will be using the requested transports.  Used for logging purposes.</param>
  229.         /// <returns type="Object" />
  230.         if ($.isArray(requestedTransport)) {
  231.             // Go through transport array and remove an "invalid" tranports
  232.             for (var i = requestedTransport.length - 1; i >= 0; i--) {
  233.                 var transport = requestedTransport[i];
  234.                 if ($.type(transport) !== "string" || !signalR.transports[transport]) {
  235.                     connection.log("Invalid transport: " + transport + ", removing it from the transports list.");
  236.                     requestedTransport.splice(i, 1);
  237.                 }
  238.             }
  239.             // Verify we still have transports left, if we dont then we have invalid transports
  240.             if (requestedTransport.length === 0) {
  241.                 connection.log("No transports remain within the specified transport array.");
  242.                 requestedTransport = null;
  243.             }
  244.         } else if (!signalR.transports[requestedTransport] && requestedTransport !== "auto") {
  245.             connection.log("Invalid transport: " + requestedTransport.toString() + ".");
  246.             requestedTransport = null;
  247.         } else if (requestedTransport === "auto" && signalR._.ieVersion <= 8) {
  248.             // If we're doing an auto transport and we're IE8 then force longPolling, #1764
  249.             return ["longPolling"];
  250.         }
  251.         return requestedTransport;
  252.     }
  253.     function getDefaultPort(protocol) {
  254.         if (protocol === "http:") {
  255.             return 80;
  256.         } else if (protocol === "https:") {
  257.             return 443;
  258.         }
  259.     }
  260.     function addDefaultPort(protocol, url) {
  261.         // Remove ports  from url.  We have to check if there's a / or end of line
  262.         // following the port in order to avoid removing ports such as 8080.
  263.         if (url.match(/:\d+$/)) {
  264.             return url;
  265.         } else {
  266.             return url + ":" + getDefaultPort(protocol);
  267.         }
  268.     }
  269.     function ConnectingMessageBuffer(connection, drainCallback) {
  270.         var that = this,
  271.             buffer = [];
  272.         that.tryBuffer = function (message) {
  273.             if (connection.state === $.signalR.connectionState.connecting) {
  274.                 buffer.push(message);
  275.                 return true;
  276.             }
  277.             return false;
  278.         };
  279.         that.drain = function () {
  280.             // Ensure that the connection is connected when we drain (do not want to drain while a connection is not active)
  281.             if (connection.state === $.signalR.connectionState.connected) {
  282.                 while (buffer.length > 0) {
  283.                     drainCallback(buffer.shift());
  284.                 }
  285.             }
  286.         };
  287.         that.clear = function () {
  288.             buffer = [];
  289.         };
  290.     }
  291.     signalR.fn = signalR.prototype = {
  292.         init: function (url, qs, logging) {
  293.             var $connection = $(this);
  294.             this.url = url;
  295.             this.qs = qs;
  296.             this.lastError = null;
  297.             this._ = {
  298.                 keepAliveData: {},
  299.                 connectingMessageBuffer: new ConnectingMessageBuffer(this, function (message) {
  300.                     $connection.triggerHandler(events.onReceived, [message]);
  301.                 }),
  302.                 lastMessageAt: new Date().getTime(),
  303.                 lastActiveAt: new Date().getTime(),
  304.                 beatInterval: 5000, // Default value, will only be overridden if keep alive is enabled,
  305.                 beatHandle: null,
  306.                 totalTransportConnectTimeout: 0, // This will be the sum of the TransportConnectTimeout sent in response to negotiate and connection.transportConnectTimeout
  307.                 redirectQs: null
  308.             };
  309.             if (typeof (logging) === "boolean") {
  310.                 this.logging = logging;
  311.             }
  312.         },
  313.         _parseResponse: function (response) {
  314.             var that = this;
  315.             if (!response) {
  316.                 return response;
  317.             } else if (typeof response === "string") {
  318.                 return that.json.parse(response);
  319.             } else {
  320.                 return response;
  321.             }
  322.         },
  323.         _originalJson: window.JSON,
  324.         json: window.JSON,
  325.         isCrossDomain: function (url, against) {
  326.             /// <summary>Checks if url is cross domain</summary>
  327.             /// <param name="url" type="String">The base URL</param>
  328.             /// <param name="against" type="Object">
  329.             ///     An optional argument to compare the URL against, if not specified it will be set to window.location.
  330.             ///     If specified it must contain a protocol and a host property.
  331.             /// </param>
  332.             var link;
  333.             url = $.trim(url);
  334.             against = against || window.location;
  335.             if (url.indexOf("http") !== 0) {
  336.                 return false;
  337.             }
  338.             // Create an anchor tag.
  339.             link = window.document.createElement("a");
  340.             link.href = url;
  341.             // When checking for cross domain we have to special case port 80 because the window.location will remove the
  342.             return link.protocol + addDefaultPort(link.protocol, link.host) !== against.protocol + addDefaultPort(against.protocol, against.host);
  343.         },
  344.         ajaxDataType: "text",
  345.         contentType: "application/json; charset=UTF-8",
  346.         logging: false,
  347.         state: signalR.connectionState.disconnected,
  348.         clientProtocol: "2.1",
  349.         // We want to support older servers since the 2.0 change is to support redirection results, which isn't
  350.         // really breaking in the protocol. So if a user updates their client to 2.0 protocol version there's
  351.         // no reason they can't still connect to a 1.5 server. The 2.1 protocol is sent by the client so the SignalR
  352.         // service knows the client will use they query string returned via the RedirectUrl for subsequent requests.
  353.         // It doesn't matter whether the server reflects back 2.1 or continues using 2.0 as the protocol version.
  354.         supportedProtocols: ["1.5", "2.0", "2.1"],
  355.         negotiateRedirectSupportedProtocols: ["2.0", "2.1"],
  356.         reconnectDelay: 2000,
  357.         transportConnectTimeout: 0,
  358.         disconnectTimeout: 30000, // This should be set by the server in response to the negotiate request (30s default)
  359.         reconnectWindow: 30000, // This should be set by the server in response to the negotiate request
  360.         keepAliveWarnAt: 2 / 3, // Warn user of slow connection if we breach the X% mark of the keep alive timeout
  361.         start: function (options, callback) {
  362.             /// <summary>Starts the connection</summary>
  363.             /// <param name="options" type="Object">Options map</param>
  364.             /// <param name="callback" type="Function">A callback function to execute when the connection has started</param>
  365.             var connection = this,
  366.                 config = {
  367.                     pingInterval: 300000,
  368.                     waitForPageLoad: true,
  369.                     transport: "auto",
  370.                     jsonp: false
  371.                 },
  372.                 initialize,
  373.                 deferred = connection._deferral || $.Deferred(), // Check to see if there is a pre-existing deferral that's being built on, if so we want to keep using it
  374.                 parser = window.document.createElement("a"),
  375.                 setConnectionUrl = function (connection, url) {
  376.                     if (connection.url === url && connection.baseUrl) {
  377.                         // when the url related properties are already set
  378.                         return;
  379.                     }
  380.                     connection.url = url;
  381.                     // Resolve the full url
  382.                     parser.href = connection.url;
  383.                     if (!parser.protocol || parser.protocol === ":") {
  384.                         connection.protocol = window.document.location.protocol;
  385.                         connection.host = parser.host || window.document.location.host;
  386.                     } else {
  387.                         connection.protocol = parser.protocol;
  388.                         connection.host = parser.host;
  389.                     }
  390.                     connection.baseUrl = connection.protocol + "//" + connection.host;
  391.                     // Set the websocket protocol
  392.                     connection.wsProtocol = connection.protocol === "https:" ? "wss://" : "ws://";
  393.                     // If the url is protocol relative, prepend the current windows protocol to the url.
  394.                     if (connection.url.indexOf("//") === 0) {
  395.                         connection.url = window.location.protocol + connection.url;
  396.                         connection.log("Protocol relative URL detected, normalizing it to '" + connection.url + "'.");
  397.                     }
  398.                     if (connection.isCrossDomain(connection.url)) {
  399.                         connection.log("Auto detected cross domain url.");
  400.                         if (config.transport === "auto") {
  401.                             // Cross-domain does not support foreverFrame
  402.                             config.transport = ["webSockets", "serverSentEvents", "longPolling"];
  403.                         }
  404.                         if (typeof connection.withCredentials === "undefined") {
  405.                             connection.withCredentials = true;
  406.                         }
  407.                         // Determine if jsonp is the only choice for negotiation, ajaxSend and ajaxAbort.
  408.                         // i.e. if the browser doesn't supports CORS
  409.                         // If it is, ignore any preference to the contrary, and switch to jsonp.
  410.                         if (!$.support.cors) {
  411.                             connection.ajaxDataType = "jsonp";
  412.                             connection.log("Using jsonp because this browser doesn't support CORS.");
  413.                         }
  414.                         connection.contentType = signalR._.defaultContentType;
  415.                     }
  416.                 };
  417.             connection.lastError = null;
  418.             // Persist the deferral so that if start is called multiple times the same deferral is used.
  419.             connection._deferral = deferred;
  420.             if (!connection.json) {
  421.                 // no JSON!
  422.                 throw new Error("SignalR: No JSON parser found. Please ensure json2.js is referenced before the SignalR.js file if you need to support clients without native JSON parsing support, e.g. IE<8.");
  423.             }
  424.             if ($.type(options) === "function") {
  425.                 // Support calling with single callback parameter
  426.                 callback = options;
  427.             } else if ($.type(options) === "object") {
  428.                 $.extend(config, options);
  429.                 if ($.type(config.callback) === "function") {
  430.                     callback = config.callback;
  431.                 }
  432.             }
  433.             config.transport = validateTransport(config.transport, connection);
  434.             // If the transport is invalid throw an error and abort start
  435.             if (!config.transport) {
  436.                 throw new Error("SignalR: Invalid transport(s) specified, aborting start.");
  437.             }
  438.             connection._.config = config;
  439.             // Check to see if start is being called prior to page load
  440.             // If waitForPageLoad is true we then want to re-direct function call to the window load event
  441.             if (!_pageLoaded && config.waitForPageLoad === true) {
  442.                 connection._.deferredStartHandler = function () {
  443.                     connection.start(options, callback);
  444.                 };
  445.                 _pageWindow.bind("load", connection._.deferredStartHandler);
  446.                 return deferred.promise();
  447.             }
  448.             // If we're already connecting just return the same deferral as the original connection start
  449.             if (connection.state === signalR.connectionState.connecting) {
  450.                 return deferred.promise();
  451.             } else if (changeState(connection,
  452.                 signalR.connectionState.disconnected,
  453.                 signalR.connectionState.connecting) === false) {
  454.                 // We're not connecting so try and transition into connecting.
  455.                 // If we fail to transition then we're either in connected or reconnecting.
  456.                 deferred.resolve(connection);
  457.                 return deferred.promise();
  458.             }
  459.             configureStopReconnectingTimeout(connection);
  460.             // If jsonp with no/auto transport is specified, then set the transport to long polling
  461.             // since that is the only transport for which jsonp really makes sense.
  462.             // Some developers might actually choose to specify jsonp for same origin requests
  463.             // as demonstrated by Issue #623.
  464.             if (config.transport === "auto" && config.jsonp === true) {
  465.                 config.transport = "longPolling";
  466.             }
  467.             connection.withCredentials = config.withCredentials;
  468.             // Save the original url so that we can reset it when we stop and restart the connection
  469.             connection._originalUrl = connection.url;
  470.             connection.ajaxDataType = config.jsonp ? "jsonp" : "text";
  471.             setConnectionUrl(connection, connection.url);
  472.             $(connection).bind(events.onStart, function (e, data) {
  473.                 if ($.type(callback) === "function") {
  474.                     callback.call(connection);
  475.                 }
  476.                 deferred.resolve(connection);
  477.             });
  478.             connection._.initHandler = signalR.transports._logic.initHandler(connection);
  479.             initialize = function (transports, index) {
  480.                 var noTransportError = signalR._.error(resources.noTransportOnInit);
  481.                 index = index || 0;
  482.                 if (index >= transports.length) {
  483.                     if (index === 0) {
  484.                         connection.log("No transports supported by the server were selected.");
  485.                     } else if (index === 1) {
  486.                         connection.log("No fallback transports were selected.");
  487.                     } else {
  488.                         connection.log("Fallback transports exhausted.");
  489.                     }
  490.                     // No transport initialized successfully
  491.                     $(connection).triggerHandler(events.onError, [noTransportError]);
  492.                     deferred.reject(noTransportError);
  493.                     // Stop the connection if it has connected and move it into the disconnected state
  494.                     connection.stop();
  495.                     return;
  496.                 }
  497.                 // The connection was aborted
  498.                 if (connection.state === signalR.connectionState.disconnected) {
  499.                     return;
  500.                 }
  501.                 var transportName = transports[index],
  502.                     transport = signalR.transports[transportName],
  503.                     onFallback = function () {
  504.                         initialize(transports, index + 1);
  505.                     };
  506.                 connection.transport = transport;
  507.                 try {
  508.                     connection._.initHandler.start(transport, function () { // success
  509.                         // Firefox 11+ doesn't allow sync XHR withCredentials: https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest#withCredentials
  510.                         var isFirefox11OrGreater = signalR._.firefoxMajorVersion(window.navigator.userAgent) >= 11,
  511.                             asyncAbort = true;
  512.                         connection.log("The start request succeeded. Transitioning to the connected state.");
  513.                         if (supportsKeepAlive(connection)) {
  514.                             signalR.transports._logic.monitorKeepAlive(connection);
  515.                         }
  516.                         if (connection._.keepAliveData.activated) {
  517.                             signalR.transports._logic.startHeartbeat(connection);
  518.                         }
  519.                         // Used to ensure low activity clients maintain their authentication.
  520.                         // Must be configured once a transport has been decided to perform valid ping requests.
  521.                         signalR._.configurePingInterval(connection);
  522.                         if (!changeState(connection,
  523.                             signalR.connectionState.connecting,
  524.                             signalR.connectionState.connected)) {
  525.                             connection.log("WARNING! The connection was not in the connecting state.");
  526.                         }
  527.                         // Drain any incoming buffered messages (messages that came in prior to connect)
  528.                         connection._.connectingMessageBuffer.drain();
  529.                         $(connection).triggerHandler(events.onStart);
  530.                         // wire the stop handler for when the user leaves the page
  531.                         _pageWindow.bind("unload", function () {
  532.                             connection.log("Window unloading, stopping the connection.");
  533.                             connection.stop(asyncAbort);
  534.                         });
  535.                         if (isFirefox11OrGreater) {
  536.                             // Firefox does not fire cross-domain XHRs in the normal unload handler on tab close.
  537.                             // #2400
  538.                             _pageWindow.bind("beforeunload", function () {
  539.                                 // If connection.stop() runs runs in beforeunload and fails, it will also fail
  540.                                 // in unload unless connection.stop() runs after a timeout.
  541.                                 window.setTimeout(function () {
  542.                                     connection.stop(asyncAbort);
  543.                                 }, 0);
  544.                             });
  545.                         }
  546.                     }, onFallback);
  547.                 }
  548.                 catch (error) {
  549.                     connection.log(transport.name + " transport threw '" + error.message + "' when attempting to start.");
  550.                     onFallback();
  551.                 }
  552.             };
  553.             var url = connection.url + "/negotiate",
  554.                 onFailed = function (error, connection) {
  555.                     var err = signalR._.error(resources.errorOnNegotiate, error, connection._.negotiateRequest);
  556.                     $(connection).triggerHandler(events.onError, err);
  557.                     deferred.reject(err);
  558.                     // Stop the connection if negotiate failed
  559.                     connection.stop();
  560.                 };
  561.             $(connection).triggerHandler(events.onStarting);
  562.             url = signalR.transports._logic.prepareQueryString(connection, url);
  563.             connection.log("Negotiating with '" + url + "'.");
  564.             // Save the ajax negotiate request object so we can abort it if stop is called while the request is in flight.
  565.             connection._.negotiateRequest = function () {
  566.                 var res,
  567.                     redirects = 0,
  568.                     MAX_REDIRECTS = 100,
  569.                     keepAliveData,
  570.                     protocolError,
  571.                     transports = [],
  572.                     supportedTransports = [],
  573.                     negotiate = function (connection, onSuccess) {
  574.                         var url = signalR.transports._logic.prepareQueryString(connection, connection.url + "/negotiate");
  575.                         connection.log("Negotiating with '" + url + "'.");
  576.                         var options = {
  577.                             url: url,
  578.                             error: function (error, statusText) {
  579.                                 // We don't want to cause any errors if we're aborting our own negotiate request.
  580.                                 if (statusText !== _negotiateAbortText) {
  581.                                     onFailed(error, connection);
  582.                                 } else {
  583.                                     // This rejection will noop if the deferred has already been resolved or rejected.
  584.                                     deferred.reject(signalR._.error(resources.stoppedWhileNegotiating, null /* error */, connection._.negotiateRequest));
  585.                                 }
  586.                             },
  587.                             success: onSuccess
  588.                         };
  589.                         if (connection.accessToken) {
  590.                             options.headers = { "Authorization": "Bearer " + connection.accessToken };
  591.                         }
  592.                         return signalR.transports._logic.ajax(connection, options);
  593.                     },
  594.                     callback = function (result) {
  595.                         try {
  596.                             res = connection._parseResponse(result);
  597.                         } catch (error) {
  598.                             onFailed(signalR._.error(resources.errorParsingNegotiateResponse, error), connection);
  599.                             return;
  600.                         }
  601.                         // Check if the server is an ASP.NET Core app
  602.                         if (res.availableTransports) {
  603.                             protocolError = signalR._.error(resources.aspnetCoreSignalrServer);
  604.                             $(connection).triggerHandler(events.onError, [protocolError]);
  605.                             deferred.reject(protocolError);
  606.                             return;
  607.                         }
  608.                         if (!res.ProtocolVersion || (connection.supportedProtocols.indexOf(res.ProtocolVersion) === -1)) {
  609.                             protocolError = signalR._.error(signalR._.format(resources.protocolIncompatible, connection.clientProtocol, res.ProtocolVersion));
  610.                             $(connection).triggerHandler(events.onError, [protocolError]);
  611.                             deferred.reject(protocolError);
  612.                             return;
  613.                         }
  614.                         // Check for a redirect response (which must have a ProtocolVersion of 2.0 or greater)
  615.                         // ProtocolVersion 2.1 is the highest supported by the client, so we can just check for 2.0 or 2.1 for now
  616.                         // instead of trying to do proper version string comparison in JavaScript.
  617.                         if (connection.negotiateRedirectSupportedProtocols.indexOf(res.ProtocolVersion) !== -1) {
  618.                             if (res.Error) {
  619.                                 protocolError = signalR._.error(signalR._.format(resources.errorFromServer, res.Error));
  620.                                 $(connection).triggerHandler(events.onError, [protocolError]);
  621.                                 deferred.reject(protocolError);
  622.                                 return;
  623.                             }
  624.                             else if (res.RedirectUrl) {
  625.                                 if (redirects === MAX_REDIRECTS) {
  626.                                     onFailed(signalR._.error(resources.errorRedirectionExceedsLimit), connection);
  627.                                     return;
  628.                                 }
  629.                                 if (config.transport === "auto") {
  630.                                     // Redirected connections do not support foreverFrame
  631.                                     config.transport = ["webSockets", "serverSentEvents", "longPolling"];
  632.                                 }
  633.                                 connection.log("Received redirect to: " + res.RedirectUrl);
  634.                                 connection.accessToken = res.AccessToken;
  635.                                 var splitUrlAndQs = res.RedirectUrl.split("?", 2);
  636.                                 setConnectionUrl(connection, splitUrlAndQs[0]);
  637.                                 // Update redirectQs with query string from only the most recent RedirectUrl.
  638.                                 connection._.redirectQs = splitUrlAndQs.length === 2 ? splitUrlAndQs[1] : null;
  639.                                 if (connection.ajaxDataType === "jsonp" && connection.accessToken) {
  640.                                     onFailed(signalR._.error(resources.jsonpNotSupportedWithAccessToken), connection);
  641.                                     return;
  642.                                 }
  643.                                 redirects++;
  644.                                 negotiate(connection, callback);
  645.                                 return;
  646.                             }
  647.                         }
  648.                         keepAliveData = connection._.keepAliveData;
  649.                         connection.appRelativeUrl = res.Url;
  650.                         connection.id = res.ConnectionId;
  651.                         connection.token = res.ConnectionToken;
  652.                         connection.webSocketServerUrl = res.WebSocketServerUrl;
  653.                         // The long poll timeout is the ConnectionTimeout plus 10 seconds
  654.                         connection._.pollTimeout = res.ConnectionTimeout * 1000 + 10000; // in ms
  655.                         // Once the server has labeled the PersistentConnection as Disconnected, we should stop attempting to reconnect
  656.                         // after res.DisconnectTimeout seconds.
  657.                         connection.disconnectTimeout = res.DisconnectTimeout * 1000; // in ms
  658.                         // Add the TransportConnectTimeout from the response to the transportConnectTimeout from the client to calculate the total timeout
  659.                         connection._.totalTransportConnectTimeout = connection.transportConnectTimeout + res.TransportConnectTimeout * 1000;
  660.                         // If we have a keep alive
  661.                         if (res.KeepAliveTimeout) {
  662.                             // Register the keep alive data as activated
  663.                             keepAliveData.activated = true;
  664.                             // Timeout to designate when to force the connection into reconnecting converted to milliseconds
  665.                             keepAliveData.timeout = res.KeepAliveTimeout * 1000;
  666.                             // Timeout to designate when to warn the developer that the connection may be dead or is not responding.
  667.                             keepAliveData.timeoutWarning = keepAliveData.timeout * connection.keepAliveWarnAt;
  668.                             // Instantiate the frequency in which we check the keep alive.  It must be short in order to not miss/pick up any changes
  669.                             connection._.beatInterval = (keepAliveData.timeout - keepAliveData.timeoutWarning) / 3;
  670.                         } else {
  671.                             keepAliveData.activated = false;
  672.                         }
  673.                         connection.reconnectWindow = connection.disconnectTimeout + (keepAliveData.timeout || 0);
  674.                         $.each(signalR.transports, function (key) {
  675.                             if ((key.indexOf("_") === 0) || (key === "webSockets" && !res.TryWebSockets)) {
  676.                                 return true;
  677.                             }
  678.                             supportedTransports.push(key);
  679.                         });
  680.                         if ($.isArray(config.transport)) {
  681.                             $.each(config.transport, function (_, transport) {
  682.                                 if ($.inArray(transport, supportedTransports) >= 0) {
  683.                                     transports.push(transport);
  684.                                 }
  685.                             });
  686.                         } else if (config.transport === "auto") {
  687.                             transports = supportedTransports;
  688.                         } else if ($.inArray(config.transport, supportedTransports) >= 0) {
  689.                             transports.push(config.transport);
  690.                         }
  691.                         initialize(transports);
  692.                     };
  693.                 return negotiate(connection, callback);
  694.             }();
  695.             return deferred.promise();
  696.         },
  697.         starting: function (callback) {
  698.             /// <summary>Adds a callback that will be invoked before anything is sent over the connection</summary>
  699.             /// <param name="callback" type="Function">A callback function to execute before the connection is fully instantiated.</param>
  700.             /// <returns type="signalR" />
  701.             var connection = this;
  702.             $(connection).bind(events.onStarting, function (e, data) {
  703.                 callback.call(connection);
  704.             });
  705.             return connection;
  706.         },
  707.         send: function (data) {
  708.             /// <summary>Sends data over the connection</summary>
  709.             /// <param name="data" type="String">The data to send over the connection</param>
  710.             /// <returns type="signalR" />
  711.             var connection = this;
  712.             if (connection.state === signalR.connectionState.disconnected) {
  713.                 // Connection hasn't been started yet
  714.                 throw new Error("SignalR: Connection must be started before data can be sent. Call .start() before .send()");
  715.             }
  716.             if (connection.state === signalR.connectionState.connecting) {
  717.                 // Connection hasn't been started yet
  718.                 throw new Error("SignalR: Connection has not been fully initialized. Use .start().done() or .start().fail() to run logic after the connection has started.");
  719.             }
  720.             connection.transport.send(connection, data);
  721.             // REVIEW: Should we return deferred here?
  722.             return connection;
  723.         },
  724.         received: function (callback) {
  725.             /// <summary>Adds a callback that will be invoked after anything is received over the connection</summary>
  726.             /// <param name="callback" type="Function">A callback function to execute when any data is received on the connection</param>
  727.             /// <returns type="signalR" />
  728.             var connection = this;
  729.             $(connection).bind(events.onReceived, function (e, data) {
  730.                 callback.call(connection, data);
  731.             });
  732.             return connection;
  733.         },
  734.         stateChanged: function (callback) {
  735.             /// <summary>Adds a callback that will be invoked when the connection state changes</summary>
  736.             /// <param name="callback" type="Function">A callback function to execute when the connection state changes</param>
  737.             /// <returns type="signalR" />
  738.             var connection = this;
  739.             $(connection).bind(events.onStateChanged, function (e, data) {
  740.                 callback.call(connection, data);
  741.             });
  742.             return connection;
  743.         },
  744.         error: function (callback) {
  745.             /// <summary>Adds a callback that will be invoked after an error occurs with the connection</summary>
  746.             /// <param name="callback" type="Function">A callback function to execute when an error occurs on the connection</param>
  747.             /// <returns type="signalR" />
  748.             var connection = this;
  749.             $(connection).bind(events.onError, function (e, errorData, sendData) {
  750.                 connection.lastError = errorData;
  751.                 // In practice 'errorData' is the SignalR built error object.
  752.                 // In practice 'sendData' is undefined for all error events except those triggered by
  753.                 // 'ajaxSend' and 'webSockets.send'.'sendData' is the original send payload.
  754.                 callback.call(connection, errorData, sendData);
  755.             });
  756.             return connection;
  757.         },
  758.         disconnected: function (callback) {
  759.             /// <summary>Adds a callback that will be invoked when the client disconnects</summary>
  760.             /// <param name="callback" type="Function">A callback function to execute when the connection is broken</param>
  761.             /// <returns type="signalR" />
  762.             var connection = this;
  763.             $(connection).bind(events.onDisconnect, function (e, data) {
  764.                 callback.call(connection);
  765.             });
  766.             return connection;
  767.         },
  768.         connectionSlow: function (callback) {
  769.             /// <summary>Adds a callback that will be invoked when the client detects a slow connection</summary>
  770.             /// <param name="callback" type="Function">A callback function to execute when the connection is slow</param>
  771.             /// <returns type="signalR" />
  772.             var connection = this;
  773.             $(connection).bind(events.onConnectionSlow, function (e, data) {
  774.                 callback.call(connection);
  775.             });
  776.             return connection;
  777.         },
  778.         reconnecting: function (callback) {
  779.             /// <summary>Adds a callback that will be invoked when the underlying transport begins reconnecting</summary>
  780.             /// <param name="callback" type="Function">A callback function to execute when the connection enters a reconnecting state</param>
  781.             /// <returns type="signalR" />
  782.             var connection = this;
  783.             $(connection).bind(events.onReconnecting, function (e, data) {
  784.                 callback.call(connection);
  785.             });
  786.             return connection;
  787.         },
  788.         reconnected: function (callback) {
  789.             /// <summary>Adds a callback that will be invoked when the underlying transport reconnects</summary>
  790.             /// <param name="callback" type="Function">A callback function to execute when the connection is restored</param>
  791.             /// <returns type="signalR" />
  792.             var connection = this;
  793.             $(connection).bind(events.onReconnect, function (e, data) {
  794.                 callback.call(connection);
  795.             });
  796.             return connection;
  797.         },
  798.         stop: function (async, notifyServer) {
  799.             /// <summary>Stops listening</summary>
  800.             /// <param name="async" type="Boolean">Whether or not to asynchronously abort the connection</param>
  801.             /// <param name="notifyServer" type="Boolean">Whether we want to notify the server that we are aborting the connection</param>
  802.             /// <returns type="signalR" />
  803.             var connection = this,
  804.                 // Save deferral because this is always cleaned up
  805.                 deferral = connection._deferral;
  806.             // Verify that we've bound a load event.
  807.             if (connection._.deferredStartHandler) {
  808.                 // Unbind the event.
  809.                 _pageWindow.unbind("load", connection._.deferredStartHandler);
  810.             }
  811.             // Always clean up private non-timeout based state.
  812.             delete connection._.config;
  813.             delete connection._.deferredStartHandler;
  814.             // This needs to be checked despite the connection state because a connection start can be deferred until page load.
  815.             // If we've deferred the start due to a page load we need to unbind the "onLoad" -> start event.
  816.             if (!_pageLoaded && (!connection._.config || connection._.config.waitForPageLoad === true)) {
  817.                 connection.log("Stopping connection prior to negotiate.");
  818.                 // If we have a deferral we should reject it
  819.                 if (deferral) {
  820.                     deferral.reject(signalR._.error(resources.stoppedWhileLoading));
  821.                 }
  822.                 // Short-circuit because the start has not been fully started.
  823.                 return;
  824.             }
  825.             if (connection.state === signalR.connectionState.disconnected) {
  826.                 return;
  827.             }
  828.             connection.log("Stopping connection.");
  829.             // Clear this no matter what
  830.             window.clearTimeout(connection._.beatHandle);
  831.             window.clearInterval(connection._.pingIntervalId);
  832.             if (connection.transport) {
  833.                 connection.transport.stop(connection);
  834.                 if (notifyServer !== false) {
  835.                     connection.transport.abort(connection, async);
  836.                 }
  837.                 if (supportsKeepAlive(connection)) {
  838.                     signalR.transports._logic.stopMonitoringKeepAlive(connection);
  839.                 }
  840.                 connection.transport = null;
  841.             }
  842.             if (connection._.negotiateRequest) {
  843.                 // If the negotiation request has already completed this will noop.
  844.                 connection._.negotiateRequest.abort(_negotiateAbortText);
  845.                 delete connection._.negotiateRequest;
  846.             }
  847.             // Ensure that initHandler.stop() is called before connection._deferral is deleted
  848.             if (connection._.initHandler) {
  849.                 connection._.initHandler.stop();
  850.             }
  851.             delete connection._deferral;
  852.             delete connection.messageId;
  853.             delete connection.groupsToken;
  854.             delete connection.id;
  855.             delete connection._.pingIntervalId;
  856.             delete connection._.lastMessageAt;
  857.             delete connection._.lastActiveAt;
  858.             // Clear out our message buffer
  859.             connection._.connectingMessageBuffer.clear();
  860.             // Clean up this event
  861.             $(connection).unbind(events.onStart);
  862.             // Reset the URL and clear the access token
  863.             delete connection.accessToken;
  864.             delete connection.protocol;
  865.             delete connection.host;
  866.             delete connection.baseUrl;
  867.             delete connection.wsProtocol;
  868.             delete connection.contentType;
  869.             connection.url = connection._originalUrl;
  870.             connection._.redirectQs = null;
  871.             // Trigger the disconnect event
  872.             changeState(connection, connection.state, signalR.connectionState.disconnected);
  873.             $(connection).triggerHandler(events.onDisconnect);
  874.             return connection;
  875.         },
  876.         log: function (msg) {
  877.             log(msg, this.logging);
  878.         }
  879.     };
  880.     signalR.fn.init.prototype = signalR.fn;
  881.     signalR.noConflict = function () {
  882.         /// <summary>Reinstates the original value of $.connection and returns the signalR object for manual assignment</summary>
  883.         /// <returns type="signalR" />
  884.         if ($.connection === signalR) {
  885.             $.connection = _connection;
  886.         }
  887.         return signalR;
  888.     };
  889.     if ($.connection) {
  890.         _connection = $.connection;
  891.     }
  892.     $.connection = $.signalR = signalR;
  893. }(window.jQuery, window));
  894. /* jquery.signalR.transports.common.js */
  895. // Copyright (c) .NET Foundation. All rights reserved.
  896. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  897. /*global window:false */
  898. /// <reference path="jquery.signalR.core.js" />
  899. (function ($, window, undefined) {
  900.     var signalR = $.signalR,
  901.         events = $.signalR.events,
  902.         changeState = $.signalR.changeState,
  903.         startAbortText = "__Start Aborted__",
  904.         transportLogic;
  905.     signalR.transports = {};
  906.     function beat(connection) {
  907.         if (connection._.keepAliveData.monitoring) {
  908.             checkIfAlive(connection);
  909.         }
  910.         // Ensure that we successfully marked active before continuing the heartbeat.
  911.         if (transportLogic.markActive(connection)) {
  912.             connection._.beatHandle = window.setTimeout(function () {
  913.                 beat(connection);
  914.             }, connection._.beatInterval);
  915.         }
  916.     }
  917.     function checkIfAlive(connection) {
  918.         var keepAliveData = connection._.keepAliveData,
  919.             timeElapsed;
  920.         // Only check if we're connected
  921.         if (connection.state === signalR.connectionState.connected) {
  922.             timeElapsed = new Date().getTime() - connection._.lastMessageAt;
  923.             // Check if the keep alive has completely timed out
  924.             if (timeElapsed >= keepAliveData.timeout) {
  925.                 connection.log("Keep alive timed out.  Notifying transport that connection has been lost.");
  926.                 // Notify transport that the connection has been lost
  927.                 connection.transport.lostConnection(connection);
  928.             } else if (timeElapsed >= keepAliveData.timeoutWarning) {
  929.                 // This is to assure that the user only gets a single warning
  930.                 if (!keepAliveData.userNotified) {
  931.                     connection.log("Keep alive has been missed, connection may be dead/slow.");
  932.                     $(connection).triggerHandler(events.onConnectionSlow);
  933.                     keepAliveData.userNotified = true;
  934.                 }
  935.             } else {
  936.                 keepAliveData.userNotified = false;
  937.             }
  938.         }
  939.     }
  940.     function getAjaxUrl(connection, path) {
  941.         var url = connection.url + path;
  942.         if (connection.transport) {
  943.             url += "?transport=" + connection.transport.name;
  944.         }
  945.         return transportLogic.prepareQueryString(connection, url);
  946.     }
  947.     function InitHandler(connection) {
  948.         this.connection = connection;
  949.         this.startRequested = false;
  950.         this.startCompleted = false;
  951.         this.connectionStopped = false;
  952.     }
  953.     InitHandler.prototype = {
  954.         start: function (transport, onSuccess, onFallback) {
  955.             var that = this,
  956.                 connection = that.connection,
  957.                 failCalled = false;
  958.             if (that.startRequested || that.connectionStopped) {
  959.                 connection.log("WARNING! " + transport.name + " transport cannot be started. Initialization ongoing or completed.");
  960.                 return;
  961.             }
  962.             connection.log(transport.name + " transport starting.");
  963.             transport.start(connection, function () {
  964.                 if (!failCalled) {
  965.                     that.initReceived(transport, onSuccess);
  966.                 }
  967.             }, function (error) {
  968.                 // Don't allow the same transport to cause onFallback to be called twice
  969.                 if (!failCalled) {
  970.                     failCalled = true;
  971.                     that.transportFailed(transport, error, onFallback);
  972.                 }
  973.                 // Returns true if the transport should stop;
  974.                 // false if it should attempt to reconnect
  975.                 return !that.startCompleted || that.connectionStopped;
  976.             });
  977.             that.transportTimeoutHandle = window.setTimeout(function () {
  978.                 if (!failCalled) {
  979.                     failCalled = true;
  980.                     connection.log(transport.name + " transport timed out when trying to connect.");
  981.                     that.transportFailed(transport, undefined, onFallback);
  982.                 }
  983.             }, connection._.totalTransportConnectTimeout);
  984.         },
  985.         stop: function () {
  986.             this.connectionStopped = true;
  987.             window.clearTimeout(this.transportTimeoutHandle);
  988.             signalR.transports._logic.tryAbortStartRequest(this.connection);
  989.         },
  990.         initReceived: function (transport, onSuccess) {
  991.             var that = this,
  992.                 connection = that.connection;
  993.             if (that.startRequested) {
  994.                 connection.log("WARNING! The client received multiple init messages.");
  995.                 return;
  996.             }
  997.             if (that.connectionStopped) {
  998.                 return;
  999.             }
  1000.             that.startRequested = true;
  1001.             window.clearTimeout(that.transportTimeoutHandle);
  1002.             connection.log(transport.name + " transport connected. Initiating start request.");
  1003.             signalR.transports._logic.ajaxStart(connection, function () {
  1004.                 that.startCompleted = true;
  1005.                 onSuccess();
  1006.             });
  1007.         },
  1008.         transportFailed: function (transport, error, onFallback) {
  1009.             var connection = this.connection,
  1010.                 deferred = connection._deferral,
  1011.                 wrappedError;
  1012.             if (this.connectionStopped) {
  1013.                 return;
  1014.             }
  1015.             window.clearTimeout(this.transportTimeoutHandle);
  1016.             if (!this.startRequested) {
  1017.                 transport.stop(connection);
  1018.                 connection.log(transport.name + " transport failed to connect. Attempting to fall back.");
  1019.                 onFallback();
  1020.             } else if (!this.startCompleted) {
  1021.                 // Do not attempt to fall back if a start request is ongoing during a transport failure.
  1022.                 // Instead, trigger an error and stop the connection.
  1023.                 wrappedError = signalR._.error(signalR.resources.errorDuringStartRequest, error);
  1024.                 connection.log(transport.name + " transport failed during the start request. Stopping the connection.");
  1025.                 $(connection).triggerHandler(events.onError, [wrappedError]);
  1026.                 if (deferred) {
  1027.                     deferred.reject(wrappedError);
  1028.                 }
  1029.                 connection.stop();
  1030.             } else {
  1031.                 // The start request has completed, but the connection has not stopped.
  1032.                 // No need to do anything here. The transport should attempt its normal reconnect logic.
  1033.             }
  1034.         }
  1035.     };
  1036.     transportLogic = signalR.transports._logic = {
  1037.         ajax: function (connection, options) {
  1038.             return $.ajax(
  1039.                 $.extend(/*deep copy*/ true, {}, $.signalR.ajaxDefaults, {
  1040.                     type: "GET",
  1041.                     data: {},
  1042.                     xhrFields: { withCredentials: connection.withCredentials },
  1043.                     contentType: connection.contentType,
  1044.                     dataType: connection.ajaxDataType
  1045.                 }, options));
  1046.         },
  1047.         pingServer: function (connection) {
  1048.             /// <summary>Pings the server</summary>
  1049.             /// <param name="connection" type="signalr">Connection associated with the server ping</param>
  1050.             /// <returns type="signalR" />
  1051.             var url,
  1052.                 xhr,
  1053.                 deferral = $.Deferred();
  1054.             if (connection.transport) {
  1055.                 url = connection.url + "/ping";
  1056.                 url = transportLogic.addQs(url, connection.qs);
  1057.                 xhr = transportLogic.ajax(connection, {
  1058.                     url: url,
  1059.                     headers: connection.accessToken ? { "Authorization": "Bearer " + connection.accessToken } : {},
  1060.                     success: function (result) {
  1061.                         var data;
  1062.                         try {
  1063.                             data = connection._parseResponse(result);
  1064.                         }
  1065.                         catch (error) {
  1066.                             deferral.reject(
  1067.                                 signalR._.transportError(
  1068.                                     signalR.resources.pingServerFailedParse,
  1069.                                     connection.transport,
  1070.                                     error,
  1071.                                     xhr
  1072.                                 )
  1073.                             );
  1074.                             connection.stop();
  1075.                             return;
  1076.                         }
  1077.                         if (data.Response === "pong") {
  1078.                             deferral.resolve();
  1079.                         }
  1080.                         else {
  1081.                             deferral.reject(
  1082.                                 signalR._.transportError(
  1083.                                     signalR._.format(signalR.resources.pingServerFailedInvalidResponse, result),
  1084.                                     connection.transport,
  1085.                                     null /* error */,
  1086.                                     xhr
  1087.                                 )
  1088.                             );
  1089.                         }
  1090.                     },
  1091.                     error: function (error) {
  1092.                         if (error.status === 401 || error.status === 403) {
  1093.                             deferral.reject(
  1094.                                 signalR._.transportError(
  1095.                                     signalR._.format(signalR.resources.pingServerFailedStatusCode, error.status),
  1096.                                     connection.transport,
  1097.                                     error,
  1098.                                     xhr
  1099.                                 )
  1100.                             );
  1101.                             connection.stop();
  1102.                         }
  1103.                         else {
  1104.                             deferral.reject(
  1105.                                 signalR._.transportError(
  1106.                                     signalR.resources.pingServerFailed,
  1107.                                     connection.transport,
  1108.                                     error,
  1109.                                     xhr
  1110.                                 )
  1111.                             );
  1112.                         }
  1113.                     }
  1114.                 });
  1115.             }
  1116.             else {
  1117.                 deferral.reject(
  1118.                     signalR._.transportError(
  1119.                         signalR.resources.noConnectionTransport,
  1120.                         connection.transport
  1121.                     )
  1122.                 );
  1123.             }
  1124.             return deferral.promise();
  1125.         },
  1126.         prepareQueryString: function (connection, url) {
  1127.             var preparedUrl;
  1128.             // Use addQs to start since it handles the ?/& prefix for us
  1129.             preparedUrl = transportLogic.addQs(url, "clientProtocol=" + connection.clientProtocol);
  1130.             if (typeof (connection._.redirectQs) === "string") {
  1131.                 // Add the redirect-specified query string params if any
  1132.                 preparedUrl = transportLogic.addQs(preparedUrl, connection._.redirectQs);
  1133.             } else {
  1134.                 // Otherwise, add the user-specified query string params if any
  1135.                 preparedUrl = transportLogic.addQs(preparedUrl, connection.qs);
  1136.             }
  1137.             if (connection.token) {
  1138.                 preparedUrl += "&connectionToken=" + window.encodeURIComponent(connection.token);
  1139.             }
  1140.             if (connection.data) {
  1141.                 preparedUrl += "&connectionData=" + window.encodeURIComponent(connection.data);
  1142.             }
  1143.             return preparedUrl;
  1144.         },
  1145.         addQs: function (url, qs) {
  1146.             var appender = url.indexOf("?") !== -1 ? "&" : "?",
  1147.                 firstChar;
  1148.             if (!qs) {
  1149.                 return url;
  1150.             }
  1151.             if (typeof (qs) === "object") {
  1152.                 return url + appender + $.param(qs);
  1153.             }
  1154.             if (typeof (qs) === "string") {
  1155.                 firstChar = qs.charAt(0);
  1156.                 if (firstChar === "?" || firstChar === "&") {
  1157.                     appender = "";
  1158.                 }
  1159.                 return url + appender + qs;
  1160.             }
  1161.             throw new Error("Query string property must be either a string or object.");
  1162.         },
  1163.         // BUG #2953: The url needs to be same otherwise it will cause a memory leak
  1164.         getUrl: function (connection, transport, reconnecting, poll, ajaxPost) {
  1165.             /// <summary>Gets the url for making a GET based connect request</summary>
  1166.             var baseUrl = transport === "webSockets" ? "" : connection.baseUrl,
  1167.                 url = baseUrl + connection.appRelativeUrl,
  1168.                 qs = "transport=" + transport;
  1169.             if (!ajaxPost && connection.groupsToken) {
  1170.                 qs += "&groupsToken=" + window.encodeURIComponent(connection.groupsToken);
  1171.             }
  1172.             if (!reconnecting) {
  1173.                 url += "/connect";
  1174.             } else {
  1175.                 if (poll) {
  1176.                     // longPolling transport specific
  1177.                     url += "/poll";
  1178.                 } else {
  1179.                     url += "/reconnect";
  1180.                 }
  1181.                 if (!ajaxPost && connection.messageId) {
  1182.                     qs += "&messageId=" + window.encodeURIComponent(connection.messageId);
  1183.                 }
  1184.             }
  1185.             url += "?" + qs;
  1186.             url = transportLogic.prepareQueryString(connection, url);
  1187.             // With sse or ws, access_token in request header is not supported
  1188.             if (connection.transport && connection.accessToken) {
  1189.                 if (connection.transport.name === "serverSentEvents" || connection.transport.name === "webSockets") {
  1190.                     url += "&access_token=" + window.encodeURIComponent(connection.accessToken);
  1191.                 }
  1192.             }
  1193.             if (!ajaxPost) {
  1194.                 url += "&tid=" + Math.floor(Math.random() * 11);
  1195.             }
  1196.             return url;
  1197.         },
  1198.         maximizePersistentResponse: function (minPersistentResponse) {
  1199.             return {
  1200.                 MessageId: minPersistentResponse.C,
  1201.                 Messages: minPersistentResponse.M,
  1202.                 Initialized: typeof (minPersistentResponse.S) !== "undefined" ? true : false,
  1203.                 ShouldReconnect: typeof (minPersistentResponse.T) !== "undefined" ? true : false,
  1204.                 LongPollDelay: minPersistentResponse.L,
  1205.                 GroupsToken: minPersistentResponse.G,
  1206.                 Error: minPersistentResponse.E
  1207.             };
  1208.         },
  1209.         updateGroups: function (connection, groupsToken) {
  1210.             if (groupsToken) {
  1211.                 connection.groupsToken = groupsToken;
  1212.             }
  1213.         },
  1214.         stringifySend: function (connection, message) {
  1215.             if (typeof (message) === "string" || typeof (message) === "undefined" || message === null) {
  1216.                 return message;
  1217.             }
  1218.             return connection.json.stringify(message);
  1219.         },
  1220.         ajaxSend: function (connection, data) {
  1221.             var payload = transportLogic.stringifySend(connection, data),
  1222.                 url = getAjaxUrl(connection, "/send"),
  1223.                 xhr,
  1224.                 onFail = function (error, connection) {
  1225.                     $(connection).triggerHandler(events.onError, [signalR._.transportError(signalR.resources.sendFailed, connection.transport, error, xhr), data]);
  1226.                 };
  1227.             xhr = transportLogic.ajax(connection, {
  1228.                 url: url,
  1229.                 type: connection.ajaxDataType === "jsonp" ? "GET" : "POST",
  1230.                 contentType: signalR._.defaultContentType,
  1231.                 headers: connection.accessToken ? { "Authorization": "Bearer " + connection.accessToken } : {},
  1232.                 data: {
  1233.                     data: payload
  1234.                 },
  1235.                 success: function (result) {
  1236.                     var res;
  1237.                     if (result) {
  1238.                         try {
  1239.                             res = connection._parseResponse(result);
  1240.                         }
  1241.                         catch (error) {
  1242.                             onFail(error, connection);
  1243.                             connection.stop();
  1244.                             return;
  1245.                         }
  1246.                         transportLogic.triggerReceived(connection, res);
  1247.                     }
  1248.                 },
  1249.                 error: function (error, textStatus) {
  1250.                     if (textStatus === "abort" || textStatus === "parsererror") {
  1251.                         // The parsererror happens for sends that don't return any data, and hence
  1252.                         // don't write the jsonp callback to the response. This is harder to fix on the server
  1253.                         // so just hack around it on the client for now.
  1254.                         return;
  1255.                     }
  1256.                     onFail(error, connection);
  1257.                 }
  1258.             });
  1259.             return xhr;
  1260.         },
  1261.         ajaxAbort: function (connection, async) {
  1262.             if (typeof (connection.transport) === "undefined") {
  1263.                 return;
  1264.             }
  1265.             // Async by default unless explicitly overidden
  1266.             async = typeof async === "undefined" ? true : async;
  1267.             var url = getAjaxUrl(connection, "/abort");
  1268.             var requestHeaders = connection.accessToken ? { "Authorization": "Bearer " + connection.accessToken } : {};
  1269.             //option #1 - send "fetch" with keepalive
  1270.             if (window.fetch) {
  1271.                 // use the fetch API with keepalive
  1272.                 window.fetch(url, {
  1273.                     method: "POST",
  1274.                     keepalive: true,
  1275.                     headers: requestHeaders,
  1276.                     credentials: connection.withCredentials === true ? "include" : "same-origin"
  1277.                 });
  1278.             }
  1279.             else {
  1280.                 // fetch is not available - fallback to $.ajax
  1281.                 transportLogic.ajax(connection, {
  1282.                     url: url,
  1283.                     async: async,
  1284.                     timeout: 1000,
  1285.                     type: "POST",
  1286.                     headers: requestHeaders,
  1287.                     dataType: "text" // We don't want to use JSONP here even when JSONP is enabled
  1288.                 });
  1289.             }
  1290.             connection.log("Fired ajax abort async = " + async + ".");
  1291.         },
  1292.         ajaxStart: function (connection, onSuccess) {
  1293.             var rejectDeferred = function (error) {
  1294.                     var deferred = connection._deferral;
  1295.                     if (deferred) {
  1296.                         deferred.reject(error);
  1297.                     }
  1298.                 },
  1299.                 triggerStartError = function (error) {
  1300.                     connection.log("The start request failed. Stopping the connection.");
  1301.                     $(connection).triggerHandler(events.onError, [error]);
  1302.                     rejectDeferred(error);
  1303.                     connection.stop();
  1304.                 };
  1305.             connection._.startRequest = transportLogic.ajax(connection, {
  1306.                 url: getAjaxUrl(connection, "/start"),
  1307.                 headers: connection.accessToken ? { "Authorization": "Bearer " + connection.accessToken } : {},
  1308.                 success: function (result, statusText, xhr) {
  1309.                     var data;
  1310.                     try {
  1311.                         data = connection._parseResponse(result);
  1312.                     } catch (error) {
  1313.                         triggerStartError(signalR._.error(
  1314.                             signalR._.format(signalR.resources.errorParsingStartResponse, result),
  1315.                             error, xhr));
  1316.                         return;
  1317.                     }
  1318.                     if (data.Response === "started") {
  1319.                         onSuccess();
  1320.                     } else {
  1321.                         triggerStartError(signalR._.error(
  1322.                             signalR._.format(signalR.resources.invalidStartResponse, result),
  1323.                             null /* error */, xhr));
  1324.                     }
  1325.                 },
  1326.                 error: function (xhr, statusText, error) {
  1327.                     if (statusText !== startAbortText) {
  1328.                         triggerStartError(signalR._.error(
  1329.                             signalR.resources.errorDuringStartRequest,
  1330.                             error, xhr));
  1331.                     } else {
  1332.                         // Stop has been called, no need to trigger the error handler
  1333.                         // or stop the connection again with onStartError
  1334.                         connection.log("The start request aborted because connection.stop() was called.");
  1335.                         rejectDeferred(signalR._.error(
  1336.                             signalR.resources.stoppedDuringStartRequest,
  1337.                             null /* error */, xhr));
  1338.                     }
  1339.                 }
  1340.             });
  1341.         },
  1342.         tryAbortStartRequest: function (connection) {
  1343.             if (connection._.startRequest) {
  1344.                 // If the start request has already completed this will noop.
  1345.                 connection._.startRequest.abort(startAbortText);
  1346.                 delete connection._.startRequest;
  1347.             }
  1348.         },
  1349.         tryInitialize: function (connection, persistentResponse, onInitialized) {
  1350.             if (persistentResponse.Initialized && onInitialized) {
  1351.                 onInitialized();
  1352.             } else if (persistentResponse.Initialized) {
  1353.                 connection.log("WARNING! The client received an init message after reconnecting.");
  1354.             }
  1355.         },
  1356.         triggerReceived: function (connection, data) {
  1357.             if (!connection._.connectingMessageBuffer.tryBuffer(data)) {
  1358.                 $(connection).triggerHandler(events.onReceived, [data]);
  1359.             }
  1360.         },
  1361.         processMessages: function (connection, minData, onInitialized) {
  1362.             var data;
  1363.             if(minData && (typeof minData.I !== "undefined")) {
  1364.                 // This is a response to a message the client sent
  1365.                 transportLogic.triggerReceived(connection, minData);
  1366.                 return;
  1367.             }
  1368.             // Update the last message time stamp
  1369.             transportLogic.markLastMessage(connection);
  1370.             if (minData) {
  1371.                 // This is a message send directly to the client
  1372.                 data = transportLogic.maximizePersistentResponse(minData);
  1373.                 if (data.Error) {
  1374.                     // This is a global error, stop the connection.
  1375.                     connection.log("Received an error message from the server: " + minData.E);
  1376.                     $(connection).triggerHandler(signalR.events.onError, [signalR._.error(minData.E, /* source */ "ServerError")]);
  1377.                     connection.stop(/* async */ false, /* notifyServer */ false);
  1378.                     return;
  1379.                 }
  1380.                 transportLogic.updateGroups(connection, data.GroupsToken);
  1381.                 if (data.MessageId) {
  1382.                     connection.messageId = data.MessageId;
  1383.                 }
  1384.                 if (data.Messages) {
  1385.                     $.each(data.Messages, function (index, message) {
  1386.                         transportLogic.triggerReceived(connection, message);
  1387.                     });
  1388.                     transportLogic.tryInitialize(connection, data, onInitialized);
  1389.                 }
  1390.             }
  1391.         },
  1392.         monitorKeepAlive: function (connection) {
  1393.             var keepAliveData = connection._.keepAliveData;
  1394.             // If we haven't initiated the keep alive timeouts then we need to
  1395.             if (!keepAliveData.monitoring) {
  1396.                 keepAliveData.monitoring = true;
  1397.                 transportLogic.markLastMessage(connection);
  1398.                 // Save the function so we can unbind it on stop
  1399.                 connection._.keepAliveData.reconnectKeepAliveUpdate = function () {
  1400.                     // Mark a new message so that keep alive doesn't time out connections
  1401.                     transportLogic.markLastMessage(connection);
  1402.                 };
  1403.                 // Update Keep alive on reconnect
  1404.                 $(connection).bind(events.onReconnect, connection._.keepAliveData.reconnectKeepAliveUpdate);
  1405.                 connection.log("Now monitoring keep alive with a warning timeout of " + keepAliveData.timeoutWarning + ", keep alive timeout of " + keepAliveData.timeout + " and disconnecting timeout of " + connection.disconnectTimeout);
  1406.             } else {
  1407.                 connection.log("Tried to monitor keep alive but it's already being monitored.");
  1408.             }
  1409.         },
  1410.         stopMonitoringKeepAlive: function (connection) {
  1411.             var keepAliveData = connection._.keepAliveData;
  1412.             // Only attempt to stop the keep alive monitoring if its being monitored
  1413.             if (keepAliveData.monitoring) {
  1414.                 // Stop monitoring
  1415.                 keepAliveData.monitoring = false;
  1416.                 // Remove the updateKeepAlive function from the reconnect event
  1417.                 $(connection).unbind(events.onReconnect, connection._.keepAliveData.reconnectKeepAliveUpdate);
  1418.                 // Clear all the keep alive data
  1419.                 connection._.keepAliveData = {};
  1420.                 connection.log("Stopping the monitoring of the keep alive.");
  1421.             }
  1422.         },
  1423.         startHeartbeat: function (connection) {
  1424.             connection._.lastActiveAt = new Date().getTime();
  1425.             beat(connection);
  1426.         },
  1427.         markLastMessage: function (connection) {
  1428.             connection._.lastMessageAt = new Date().getTime();
  1429.             connection._.lastActiveAt = connection._.lastMessageAt;
  1430.         },
  1431.         markActive: function (connection) {
  1432.             if (transportLogic.verifyLastActive(connection)) {
  1433.                 connection._.lastActiveAt = new Date().getTime();
  1434.                 return true;
  1435.             }
  1436.             return false;
  1437.         },
  1438.         isConnectedOrReconnecting: function (connection) {
  1439.             return connection.state === signalR.connectionState.connected ||
  1440.                    connection.state === signalR.connectionState.reconnecting;
  1441.         },
  1442.         ensureReconnectingState: function (connection) {
  1443.             if (changeState(connection,
  1444.                         signalR.connectionState.connected,
  1445.                         signalR.connectionState.reconnecting) === true) {
  1446.                 $(connection).triggerHandler(events.onReconnecting);
  1447.             }
  1448.             return connection.state === signalR.connectionState.reconnecting;
  1449.         },
  1450.         clearReconnectTimeout: function (connection) {
  1451.             if (connection && connection._.reconnectTimeout) {
  1452.                 window.clearTimeout(connection._.reconnectTimeout);
  1453.                 delete connection._.reconnectTimeout;
  1454.             }
  1455.         },
  1456.         verifyLastActive: function (connection) {
  1457.             // If there is no keep alive configured, we cannot assume that timer callbacks will
  1458.             // run frequently enough to keep lastActiveAt updated.
  1459.             // https://github.com/SignalR/SignalR/issues/4536
  1460.             if (!connection._.keepAliveData.activated ||
  1461.                 new Date().getTime() - connection._.lastActiveAt < connection.reconnectWindow) {
  1462.                 return true;
  1463.             }
  1464.             var message = signalR._.format(signalR.resources.reconnectWindowTimeout, new Date(connection._.lastActiveAt), connection.reconnectWindow);
  1465.             connection.log(message);
  1466.             $(connection).triggerHandler(events.onError, [signalR._.error(message, /* source */ "TimeoutException")]);
  1467.             connection.stop(/* async */ false, /* notifyServer */ false);
  1468.             return false;
  1469.         },
  1470.         reconnect: function (connection, transportName) {
  1471.             var transport = signalR.transports[transportName];
  1472.             // We should only set a reconnectTimeout if we are currently connected
  1473.             // and a reconnectTimeout isn't already set.
  1474.             if (transportLogic.isConnectedOrReconnecting(connection) && !connection._.reconnectTimeout) {
  1475.                 // Need to verify before the setTimeout occurs because an application sleep could occur during the setTimeout duration.
  1476.                 if (!transportLogic.verifyLastActive(connection)) {
  1477.                     return;
  1478.                 }
  1479.                 connection._.reconnectTimeout = window.setTimeout(function () {
  1480.                     if (!transportLogic.verifyLastActive(connection)) {
  1481.                         return;
  1482.                     }
  1483.                     transport.stop(connection);
  1484.                     if (transportLogic.ensureReconnectingState(connection)) {
  1485.                         connection.log(transportName + " reconnecting.");
  1486.                         transport.start(connection);
  1487.                     }
  1488.                 }, connection.reconnectDelay);
  1489.             }
  1490.         },
  1491.         handleParseFailure: function (connection, result, error, onFailed, context) {
  1492.             var wrappedError = signalR._.transportError(
  1493.                 signalR._.format(signalR.resources.parseFailed, result),
  1494.                 connection.transport,
  1495.                 error,
  1496.                 context);
  1497.             // If we're in the initialization phase trigger onFailed, otherwise stop the connection.
  1498.             if (onFailed && onFailed(wrappedError)) {
  1499.                 connection.log("Failed to parse server response while attempting to connect.");
  1500.             } else {
  1501.                 $(connection).triggerHandler(events.onError, [wrappedError]);
  1502.                 connection.stop();
  1503.             }
  1504.         },
  1505.         initHandler: function (connection) {
  1506.             return new InitHandler(connection);
  1507.         },
  1508.         foreverFrame: {
  1509.             count: 0,
  1510.             connections: {}
  1511.         }
  1512.     };
  1513. }(window.jQuery, window));
  1514. /* jquery.signalR.transports.webSockets.js */
  1515. // Copyright (c) .NET Foundation. All rights reserved.
  1516. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  1517. /*global window:false */
  1518. /// <reference path="jquery.signalR.transports.common.js" />
  1519. (function ($, window, undefined) {
  1520.     var signalR = $.signalR,
  1521.         events = $.signalR.events,
  1522.         changeState = $.signalR.changeState,
  1523.         transportLogic = signalR.transports._logic;
  1524.     signalR.transports.webSockets = {
  1525.         name: "webSockets",
  1526.         supportsKeepAlive: function () {
  1527.             return true;
  1528.         },
  1529.         send: function (connection, data) {
  1530.             var payload = transportLogic.stringifySend(connection, data);
  1531.             try {
  1532.                 connection.socket.send(payload);
  1533.             } catch (ex) {
  1534.                 $(connection).triggerHandler(events.onError,
  1535.                     [signalR._.transportError(
  1536.                         signalR.resources.webSocketsInvalidState,
  1537.                         connection.transport,
  1538.                         ex,
  1539.                         connection.socket
  1540.                     ),
  1541.                     data]);
  1542.             }
  1543.         },
  1544.         start: function (connection, onSuccess, onFailed) {
  1545.             var url,
  1546.                 opened = false,
  1547.                 that = this,
  1548.                 reconnecting = !onSuccess,
  1549.                 $connection = $(connection);
  1550.             if (!window.WebSocket) {
  1551.                 onFailed();
  1552.                 return;
  1553.             }
  1554.             if (!connection.socket) {
  1555.                 if (connection.webSocketServerUrl) {
  1556.                     url = connection.webSocketServerUrl;
  1557.                 } else {
  1558.                     url = connection.wsProtocol + connection.host;
  1559.                 }
  1560.                 url += transportLogic.getUrl(connection, this.name, reconnecting);
  1561.                 connection.log("Connecting to websocket endpoint '" + url + "'.");
  1562.                 connection.socket = new window.WebSocket(url);
  1563.                 connection.socket.onopen = function () {
  1564.                     opened = true;
  1565.                     connection.log("Websocket opened.");
  1566.                     transportLogic.clearReconnectTimeout(connection);
  1567.                     if (changeState(connection,
  1568.                                     signalR.connectionState.reconnecting,
  1569.                                     signalR.connectionState.connected) === true) {
  1570.                         $connection.triggerHandler(events.onReconnect);
  1571.                     }
  1572.                 };
  1573.                 connection.socket.onclose = function (event) {
  1574.                     var error;
  1575.                     // Only handle a socket close if the close is from the current socket.
  1576.                     // Sometimes on disconnect the server will push down an onclose event
  1577.                     // to an expired socket.
  1578.                     if (this === connection.socket) {
  1579.                         if (opened && typeof event.wasClean !== "undefined" && event.wasClean === false) {
  1580.                             // Ideally this would use the websocket.onerror handler (rather than checking wasClean in onclose) but
  1581.                             // I found in some circumstances Chrome won't call onerror. This implementation seems to work on all browsers.
  1582.                             error = signalR._.transportError(
  1583.                                 signalR.resources.webSocketClosed,
  1584.                                 connection.transport,
  1585.                                 event);
  1586.                             connection.log("Unclean disconnect from websocket: " + (event.reason || "[no reason given]."));
  1587.                         } else {
  1588.                             connection.log("Websocket closed.");
  1589.                         }
  1590.                         if (!onFailed || !onFailed(error)) {
  1591.                             if (error) {
  1592.                                 $(connection).triggerHandler(events.onError, [error]);
  1593.                             }
  1594.                             that.reconnect(connection);
  1595.                         }
  1596.                     }
  1597.                 };
  1598.                 connection.socket.onmessage = function (event) {
  1599.                     var data;
  1600.                     try {
  1601.                         data = connection._parseResponse(event.data);
  1602.                     }
  1603.                     catch (error) {
  1604.                         transportLogic.handleParseFailure(connection, event.data, error, onFailed, event);
  1605.                         return;
  1606.                     }
  1607.                     if (data) {
  1608.                         transportLogic.processMessages(connection, data, onSuccess);
  1609.                     }
  1610.                 };
  1611.             }
  1612.         },
  1613.         reconnect: function (connection) {
  1614.             transportLogic.reconnect(connection, this.name);
  1615.         },
  1616.         lostConnection: function (connection) {
  1617.             this.reconnect(connection);
  1618.         },
  1619.         stop: function (connection) {
  1620.             // Don't trigger a reconnect after stopping
  1621.             transportLogic.clearReconnectTimeout(connection);
  1622.             if (connection.socket) {
  1623.                 connection.log("Closing the Websocket.");
  1624.                 connection.socket.close();
  1625.                 connection.socket = null;
  1626.             }
  1627.         },
  1628.         abort: function (connection, async) {
  1629.             transportLogic.ajaxAbort(connection, async);
  1630.         }
  1631.     };
  1632. }(window.jQuery, window));
  1633. /* jquery.signalR.transports.serverSentEvents.js */
  1634. // Copyright (c) .NET Foundation. All rights reserved.
  1635. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  1636. /*global window:false */
  1637. /// <reference path="jquery.signalR.transports.common.js" />
  1638. (function ($, window, undefined) {
  1639.     var signalR = $.signalR,
  1640.         events = $.signalR.events,
  1641.         changeState = $.signalR.changeState,
  1642.         transportLogic = signalR.transports._logic,
  1643.         clearReconnectAttemptTimeout = function (connection) {
  1644.             window.clearTimeout(connection._.reconnectAttemptTimeoutHandle);
  1645.             delete connection._.reconnectAttemptTimeoutHandle;
  1646.         };
  1647.     signalR.transports.serverSentEvents = {
  1648.         name: "serverSentEvents",
  1649.         supportsKeepAlive: function () {
  1650.             return true;
  1651.         },
  1652.         timeOut: 3000,
  1653.         start: function (connection, onSuccess, onFailed) {
  1654.             var that = this,
  1655.                 opened = false,
  1656.                 $connection = $(connection),
  1657.                 reconnecting = !onSuccess,
  1658.                 url;
  1659.             if (connection.eventSource) {
  1660.                 connection.log("The connection already has an event source. Stopping it.");
  1661.                 connection.stop();
  1662.             }
  1663.             if (!window.EventSource) {
  1664.                 if (onFailed) {
  1665.                     connection.log("This browser doesn't support SSE.");
  1666.                     onFailed();
  1667.                 }
  1668.                 return;
  1669.             }
  1670.             url = transportLogic.getUrl(connection, this.name, reconnecting);
  1671.             try {
  1672.                 connection.log("Attempting to connect to SSE endpoint '" + url + "'.");
  1673.                 connection.eventSource = new window.EventSource(url, { withCredentials: connection.withCredentials });
  1674.             }
  1675.             catch (e) {
  1676.                 connection.log("EventSource failed trying to connect with error " + e.Message + ".");
  1677.                 if (onFailed) {
  1678.                     // The connection failed, call the failed callback
  1679.                     onFailed();
  1680.                 } else {
  1681.                     $connection.triggerHandler(events.onError, [signalR._.transportError(signalR.resources.eventSourceFailedToConnect, connection.transport, e)]);
  1682.                     if (reconnecting) {
  1683.                         // If we were reconnecting, rather than doing initial connect, then try reconnect again
  1684.                         that.reconnect(connection);
  1685.                     }
  1686.                 }
  1687.                 return;
  1688.             }
  1689.             if (reconnecting) {
  1690.                 connection._.reconnectAttemptTimeoutHandle = window.setTimeout(function () {
  1691.                     if (opened === false) {
  1692.                         // If we're reconnecting and the event source is attempting to connect,
  1693.                         // don't keep retrying. This causes duplicate connections to spawn.
  1694.                         if (connection.eventSource.readyState !== window.EventSource.OPEN) {
  1695.                             // If we were reconnecting, rather than doing initial connect, then try reconnect again
  1696.                             that.reconnect(connection);
  1697.                         }
  1698.                     }
  1699.                 },
  1700.                 that.timeOut);
  1701.             }
  1702.             connection.eventSource.addEventListener("open", function (e) {
  1703.                 connection.log("EventSource connected.");
  1704.                 clearReconnectAttemptTimeout(connection);
  1705.                 transportLogic.clearReconnectTimeout(connection);
  1706.                 if (opened === false) {
  1707.                     opened = true;
  1708.                     if (changeState(connection,
  1709.                                          signalR.connectionState.reconnecting,
  1710.                                          signalR.connectionState.connected) === true) {
  1711.                         $connection.triggerHandler(events.onReconnect);
  1712.                     }
  1713.                 }
  1714.             }, false);
  1715.             connection.eventSource.addEventListener("message", function (e) {
  1716.                 var res;
  1717.                 // process messages
  1718.                 if (e.data === "initialized") {
  1719.                     return;
  1720.                 }
  1721.                 try {
  1722.                     res = connection._parseResponse(e.data);
  1723.                 }
  1724.                 catch (error) {
  1725.                     transportLogic.handleParseFailure(connection, e.data, error, onFailed, e);
  1726.                     return;
  1727.                 }
  1728.                 transportLogic.processMessages(connection, res, onSuccess);
  1729.             }, false);
  1730.             connection.eventSource.addEventListener("error", function (e) {
  1731.                 var error = signalR._.transportError(
  1732.                     signalR.resources.eventSourceError,
  1733.                     connection.transport,
  1734.                     e);
  1735.                 // Only handle an error if the error is from the current Event Source.
  1736.                 // Sometimes on disconnect the server will push down an error event
  1737.                 // to an expired Event Source.
  1738.                 if (this !== connection.eventSource) {
  1739.                     return;
  1740.                 }
  1741.                 if (onFailed && onFailed(error)) {
  1742.                     return;
  1743.                 }
  1744.                 connection.log("EventSource readyState: " + connection.eventSource.readyState + ".");
  1745.                 if (e.eventPhase === window.EventSource.CLOSED) {
  1746.                     // We don't use the EventSource's native reconnect function as it
  1747.                     // doesn't allow us to change the URL when reconnecting. We need
  1748.                     // to change the URL to not include the /connect suffix, and pass
  1749.                     // the last message id we received.
  1750.                     connection.log("EventSource reconnecting due to the server connection ending.");
  1751.                     that.reconnect(connection);
  1752.                 } else {
  1753.                     // connection error
  1754.                     connection.log("EventSource error.");
  1755.                     $connection.triggerHandler(events.onError, [error]);
  1756.                 }
  1757.             }, false);
  1758.         },
  1759.         reconnect: function (connection) {
  1760.             transportLogic.reconnect(connection, this.name);
  1761.         },
  1762.         lostConnection: function (connection) {
  1763.             this.reconnect(connection);
  1764.         },
  1765.         send: function (connection, data) {
  1766.             transportLogic.ajaxSend(connection, data);
  1767.         },
  1768.         stop: function (connection) {
  1769.             // Don't trigger a reconnect after stopping
  1770.             clearReconnectAttemptTimeout(connection);
  1771.             transportLogic.clearReconnectTimeout(connection);
  1772.             if (connection && connection.eventSource) {
  1773.                 connection.log("EventSource calling close().");
  1774.                 connection.eventSource.close();
  1775.                 connection.eventSource = null;
  1776.                 delete connection.eventSource;
  1777.             }
  1778.         },
  1779.         abort: function (connection, async) {
  1780.             transportLogic.ajaxAbort(connection, async);
  1781.         }
  1782.     };
  1783. }(window.jQuery, window));
  1784. /* jquery.signalR.transports.foreverFrame.js */
  1785. // Copyright (c) .NET Foundation. All rights reserved.
  1786. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  1787. /*global window:false */
  1788. /// <reference path="jquery.signalR.transports.common.js" />
  1789. (function ($, window, undefined) {
  1790.     var signalR = $.signalR,
  1791.         events = $.signalR.events,
  1792.         changeState = $.signalR.changeState,
  1793.         transportLogic = signalR.transports._logic,
  1794.         createFrame = function () {
  1795.             var frame = window.document.createElement("iframe");
  1796.             frame.setAttribute("style", "position:absolute;top:0;left:0;width:0;height:0;visibility:hidden;");
  1797.             return frame;
  1798.         },
  1799.         // Used to prevent infinite loading icon spins in older versions of ie
  1800.         // We build this object inside a closure so we don't pollute the rest of
  1801.         // the foreverFrame transport with unnecessary functions/utilities.
  1802.         loadPreventer = (function () {
  1803.             var loadingFixIntervalId = null,
  1804.                 loadingFixInterval = 1000,
  1805.                 attachedTo = 0;
  1806.             return {
  1807.                 prevent: function () {
  1808.                     // Prevent additional iframe removal procedures from newer browsers
  1809.                     if (signalR._.ieVersion <= 8) {
  1810.                         // We only ever want to set the interval one time, so on the first attachedTo
  1811.                         if (attachedTo === 0) {
  1812.                             // Create and destroy iframe every 3 seconds to prevent loading icon, super hacky
  1813.                             loadingFixIntervalId = window.setInterval(function () {
  1814.                                 var tempFrame = createFrame();
  1815.                                 window.document.body.appendChild(tempFrame);
  1816.                                 window.document.body.removeChild(tempFrame);
  1817.                                 tempFrame = null;
  1818.                             }, loadingFixInterval);
  1819.                         }
  1820.                         attachedTo++;
  1821.                     }
  1822.                 },
  1823.                 cancel: function () {
  1824.                     // Only clear the interval if there's only one more object that the loadPreventer is attachedTo
  1825.                     if (attachedTo === 1) {
  1826.                         window.clearInterval(loadingFixIntervalId);
  1827.                     }
  1828.                     if (attachedTo > 0) {
  1829.                         attachedTo--;
  1830.                     }
  1831.                 }
  1832.             };
  1833.         })();
  1834.     signalR.transports.foreverFrame = {
  1835.         name: "foreverFrame",
  1836.         supportsKeepAlive: function () {
  1837.             return true;
  1838.         },
  1839.         // Added as a value here so we can create tests to verify functionality
  1840.         iframeClearThreshold: 50,
  1841.         start: function (connection, onSuccess, onFailed) {
  1842.             if (connection.accessToken) {
  1843.                 if (onFailed) {
  1844.                     connection.log("Forever Frame does not support connections that require a Bearer token to connect, such as the Azure SignalR Service.");
  1845.                     onFailed();
  1846.                 }
  1847.                 return;
  1848.             }
  1849.             var that = this,
  1850.                 frameId = (transportLogic.foreverFrame.count += 1),
  1851.                 url,
  1852.                 frame = createFrame(),
  1853.                 frameLoadHandler = function () {
  1854.                     connection.log("Forever frame iframe finished loading and is no longer receiving messages.");
  1855.                     if (!onFailed || !onFailed()) {
  1856.                         that.reconnect(connection);
  1857.                     }
  1858.                 };
  1859.             if (window.EventSource) {
  1860.                 // If the browser supports SSE, don't use Forever Frame
  1861.                 if (onFailed) {
  1862.                     connection.log("Forever Frame is not supported by SignalR on browsers with SSE support.");
  1863.                     onFailed();
  1864.                 }
  1865.                 return;
  1866.             }
  1867.             frame.setAttribute("data-signalr-connection-id", connection.id);
  1868.             // Start preventing loading icon
  1869.             // This will only perform work if the loadPreventer is not attached to another connection.
  1870.             loadPreventer.prevent();
  1871.             // Build the url
  1872.             url = transportLogic.getUrl(connection, this.name);
  1873.             url += "&frameId=" + frameId;
  1874.             // add frame to the document prior to setting URL to avoid caching issues.
  1875.             window.document.documentElement.appendChild(frame);
  1876.             connection.log("Binding to iframe's load event.");
  1877.             if (frame.addEventListener) {
  1878.                 frame.addEventListener("load", frameLoadHandler, false);
  1879.             } else if (frame.attachEvent) {
  1880.                 frame.attachEvent("onload", frameLoadHandler);
  1881.             }
  1882.             frame.src = url;
  1883.             transportLogic.foreverFrame.connections[frameId] = connection;
  1884.             connection.frame = frame;
  1885.             connection.frameId = frameId;
  1886.             if (onSuccess) {
  1887.                 connection.onSuccess = function () {
  1888.                     connection.log("Iframe transport started.");
  1889.                     onSuccess();
  1890.                 };
  1891.             }
  1892.         },
  1893.         reconnect: function (connection) {
  1894.             var that = this;
  1895.             // Need to verify connection state and verify before the setTimeout occurs because an application sleep could occur during the setTimeout duration.
  1896.             if (transportLogic.isConnectedOrReconnecting(connection) && transportLogic.verifyLastActive(connection)) {
  1897.                 window.setTimeout(function () {
  1898.                     // Verify that we're ok to reconnect.
  1899.                     if (!transportLogic.verifyLastActive(connection)) {
  1900.                         return;
  1901.                     }
  1902.                     if (connection.frame && transportLogic.ensureReconnectingState(connection)) {
  1903.                         var frame = connection.frame,
  1904.                             src = transportLogic.getUrl(connection, that.name, true) + "&frameId=" + connection.frameId;
  1905.                         connection.log("Updating iframe src to '" + src + "'.");
  1906.                         frame.src = src;
  1907.                     }
  1908.                 }, connection.reconnectDelay);
  1909.             }
  1910.         },
  1911.         lostConnection: function (connection) {
  1912.             this.reconnect(connection);
  1913.         },
  1914.         send: function (connection, data) {
  1915.             transportLogic.ajaxSend(connection, data);
  1916.         },
  1917.         receive: function (connection, data) {
  1918.             var cw,
  1919.                 body,
  1920.                 response;
  1921.             if (connection.json !== connection._originalJson) {
  1922.                 // If there's a custom JSON parser configured then serialize the object
  1923.                 // using the original (browser) JSON parser and then deserialize it using
  1924.                 // the custom parser (connection._parseResponse does that). This is so we
  1925.                 // can easily send the response from the server as "raw" JSON but still
  1926.                 // support custom JSON deserialization in the browser.
  1927.                 data = connection._originalJson.stringify(data);
  1928.             }
  1929.             response = connection._parseResponse(data);
  1930.             transportLogic.processMessages(connection, response, connection.onSuccess);
  1931.             // Protect against connection stopping from a callback trigger within the processMessages above.
  1932.             if (connection.state === $.signalR.connectionState.connected) {
  1933.                 // Delete the script & div elements
  1934.                 connection.frameMessageCount = (connection.frameMessageCount || 0) + 1;
  1935.                 if (connection.frameMessageCount > signalR.transports.foreverFrame.iframeClearThreshold) {
  1936.                     connection.frameMessageCount = 0;
  1937.                     cw = connection.frame.contentWindow || connection.frame.contentDocument;
  1938.                     if (cw && cw.document && cw.document.body) {
  1939.                         body = cw.document.body;
  1940.                         // Remove all the child elements from the iframe's body to conserver memory
  1941.                         while (body.firstChild) {
  1942.                             body.removeChild(body.firstChild);
  1943.                         }
  1944.                     }
  1945.                 }
  1946.             }
  1947.         },
  1948.         stop: function (connection) {
  1949.             var cw = null;
  1950.             // Stop attempting to prevent loading icon
  1951.             loadPreventer.cancel();
  1952.             if (connection.frame) {
  1953.                 if (connection.frame.stop) {
  1954.                     connection.frame.stop();
  1955.                 } else {
  1956.                     try {
  1957.                         cw = connection.frame.contentWindow || connection.frame.contentDocument;
  1958.                         if (cw.document && cw.document.execCommand) {
  1959.                             cw.document.execCommand("Stop");
  1960.                         }
  1961.                     }
  1962.                     catch (e) {
  1963.                         connection.log("Error occurred when stopping foreverFrame transport. Message = " + e.message + ".");
  1964.                     }
  1965.                 }
  1966.                 // Ensure the iframe is where we left it
  1967.                 if (connection.frame.parentNode === window.document.documentElement) {
  1968.                     window.document.documentElement.removeChild(connection.frame);
  1969.                 }
  1970.                 delete transportLogic.foreverFrame.connections[connection.frameId];
  1971.                 connection.frame = null;
  1972.                 connection.frameId = null;
  1973.                 delete connection.frame;
  1974.                 delete connection.frameId;
  1975.                 delete connection.onSuccess;
  1976.                 delete connection.frameMessageCount;
  1977.                 connection.log("Stopping forever frame.");
  1978.             }
  1979.         },
  1980.         abort: function (connection, async) {
  1981.             transportLogic.ajaxAbort(connection, async);
  1982.         },
  1983.         getConnection: function (id) {
  1984.             return transportLogic.foreverFrame.connections[id];
  1985.         },
  1986.         started: function (connection) {
  1987.             if (changeState(connection,
  1988.                 signalR.connectionState.reconnecting,
  1989.                 signalR.connectionState.connected) === true) {
  1990.                 $(connection).triggerHandler(events.onReconnect);
  1991.             }
  1992.         }
  1993.     };
  1994. }(window.jQuery, window));
  1995. /* jquery.signalR.transports.longPolling.js */
  1996. // Copyright (c) .NET Foundation. All rights reserved.
  1997. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  1998. /*global window:false */
  1999. /// <reference path="jquery.signalR.transports.common.js" />
  2000. (function ($, window, undefined) {
  2001.     var signalR = $.signalR,
  2002.         events = $.signalR.events,
  2003.         changeState = $.signalR.changeState,
  2004.         isDisconnecting = $.signalR.isDisconnecting,
  2005.         transportLogic = signalR.transports._logic;
  2006.     signalR.transports.longPolling = {
  2007.         name: "longPolling",
  2008.         supportsKeepAlive: function () {
  2009.             return false;
  2010.         },
  2011.         reconnectDelay: 3000,
  2012.         start: function (connection, onSuccess, onFailed) {
  2013.             /// <summary>Starts the long polling connection</summary>
  2014.             /// <param name="connection" type="signalR">The SignalR connection to start</param>
  2015.             var that = this,
  2016.                 fireConnect = function () {
  2017.                     fireConnect = $.noop;
  2018.                     connection.log("LongPolling connected.");
  2019.                     if (onSuccess) {
  2020.                         onSuccess();
  2021.                     } else {
  2022.                         connection.log("WARNING! The client received an init message after reconnecting.");
  2023.                     }
  2024.                 },
  2025.                 tryFailConnect = function (error) {
  2026.                     if (onFailed(error)) {
  2027.                         connection.log("LongPolling failed to connect.");
  2028.                         return true;
  2029.                     }
  2030.                     return false;
  2031.                 },
  2032.                 privateData = connection._,
  2033.                 reconnectErrors = 0,
  2034.                 fireReconnected = function (instance) {
  2035.                     window.clearTimeout(privateData.reconnectTimeoutId);
  2036.                     privateData.reconnectTimeoutId = null;
  2037.                     if (changeState(instance,
  2038.                                     signalR.connectionState.reconnecting,
  2039.                                     signalR.connectionState.connected) === true) {
  2040.                         // Successfully reconnected!
  2041.                         instance.log("Raising the reconnect event");
  2042.                         $(instance).triggerHandler(events.onReconnect);
  2043.                     }
  2044.                 },
  2045.                 // 1 hour
  2046.                 maxFireReconnectedTimeout = 3600000;
  2047.             if (connection.pollXhr) {
  2048.                 connection.log("Polling xhr requests already exists, aborting.");
  2049.                 connection.stop();
  2050.             }
  2051.             connection.messageId = null;
  2052.             privateData.reconnectTimeoutId = null;
  2053.             privateData.pollTimeoutId = window.setTimeout(function () {
  2054.                 (function poll(instance, raiseReconnect) {
  2055.                     var messageId = instance.messageId,
  2056.                         connect = (messageId === null),
  2057.                         reconnecting = !connect,
  2058.                         polling = !raiseReconnect,
  2059.                         url = transportLogic.getUrl(instance, that.name, reconnecting, polling, true /* use Post for longPolling */),
  2060.                         postData = {};
  2061.                     if (instance.messageId) {
  2062.                         postData.messageId = instance.messageId;
  2063.                     }
  2064.                     if (instance.groupsToken) {
  2065.                         postData.groupsToken = instance.groupsToken;
  2066.                     }
  2067.                     // If we've disconnected during the time we've tried to re-instantiate the poll then stop.
  2068.                     if (isDisconnecting(instance) === true) {
  2069.                         return;
  2070.                     }
  2071.                     connection.log("Opening long polling request to '" + url + "'.");
  2072.                     instance.pollXhr = transportLogic.ajax(connection, {
  2073.                         xhrFields: {
  2074.                             onprogress: function () {
  2075.                                 transportLogic.markLastMessage(connection);
  2076.                             }
  2077.                         },
  2078.                         url: url,
  2079.                         type: "POST",
  2080.                         contentType: signalR._.defaultContentType,
  2081.                         data: postData,
  2082.                         timeout: connection._.pollTimeout,
  2083.                         headers: connection.accessToken ? { "Authorization": "Bearer " + connection.accessToken } : {},
  2084.                         success: function (result) {
  2085.                             var minData,
  2086.                                 delay = 0,
  2087.                                 data,
  2088.                                 shouldReconnect;
  2089.                             connection.log("Long poll complete.");
  2090.                             // Reset our reconnect errors so if we transition into a reconnecting state again we trigger
  2091.                             // reconnected quickly
  2092.                             reconnectErrors = 0;
  2093.                             try {
  2094.                                 // Remove any keep-alives from the beginning of the result
  2095.                                 minData = connection._parseResponse(result);
  2096.                             }
  2097.                             catch (error) {
  2098.                                 transportLogic.handleParseFailure(instance, result, error, tryFailConnect, instance.pollXhr);
  2099.                                 return;
  2100.                             }
  2101.                             // If there's currently a timeout to trigger reconnect, fire it now before processing messages
  2102.                             if (privateData.reconnectTimeoutId !== null) {
  2103.                                 fireReconnected(instance);
  2104.                             }
  2105.                             if (minData) {
  2106.                                 data = transportLogic.maximizePersistentResponse(minData);
  2107.                             }
  2108.                             transportLogic.processMessages(instance, minData, fireConnect);
  2109.                             if (data &&
  2110.                                 $.type(data.LongPollDelay) === "number") {
  2111.                                 delay = data.LongPollDelay;
  2112.                             }
  2113.                             if (isDisconnecting(instance) === true) {
  2114.                                 return;
  2115.                             }
  2116.                             shouldReconnect = data && data.ShouldReconnect;
  2117.                             if (shouldReconnect) {
  2118.                                 // Transition into the reconnecting state
  2119.                                 // If this fails then that means that the user transitioned the connection into a invalid state in processMessages.
  2120.                                 if (!transportLogic.ensureReconnectingState(instance)) {
  2121.                                     return;
  2122.                                 }
  2123.                             }
  2124.                             // We never want to pass a raiseReconnect flag after a successful poll.  This is handled via the error function
  2125.                             if (delay > 0) {
  2126.                                 privateData.pollTimeoutId = window.setTimeout(function () {
  2127.                                     poll(instance, shouldReconnect);
  2128.                                 }, delay);
  2129.                             } else {
  2130.                                 poll(instance, shouldReconnect);
  2131.                             }
  2132.                         },
  2133.                         error: function (data, textStatus) {
  2134.                             var error = signalR._.transportError(signalR.resources.longPollFailed, connection.transport, data, instance.pollXhr);
  2135.                             // Stop trying to trigger reconnect, connection is in an error state
  2136.                             // If we're not in the reconnect state this will noop
  2137.                             window.clearTimeout(privateData.reconnectTimeoutId);
  2138.                             privateData.reconnectTimeoutId = null;
  2139.                             if (textStatus === "abort") {
  2140.                                 connection.log("Aborted xhr request.");
  2141.                                 return;
  2142.                             }
  2143.                             if (!tryFailConnect(error)) {
  2144.                                 // Increment our reconnect errors, we assume all errors to be reconnect errors
  2145.                                 // In the case that it's our first error this will cause Reconnect to be fired
  2146.                                 // after 1 second due to reconnectErrors being = 1.
  2147.                                 reconnectErrors++;
  2148.                                 if (connection.state !== signalR.connectionState.reconnecting) {
  2149.                                     connection.log("An error occurred using longPolling. Status = " + textStatus + ".  Response = " + data.responseText + ".");
  2150.                                     $(instance).triggerHandler(events.onError, [error]);
  2151.                                 }
  2152.                                 // We check the state here to verify that we're not in an invalid state prior to verifying Reconnect.
  2153.                                 // If we're not in connected or reconnecting then the next ensureReconnectingState check will fail and will return.
  2154.                                 // Therefore we don't want to change that failure code path.
  2155.                                 if ((connection.state === signalR.connectionState.connected ||
  2156.                                     connection.state === signalR.connectionState.reconnecting) &&
  2157.                                     !transportLogic.verifyLastActive(connection)) {
  2158.                                     return;
  2159.                                 }
  2160.                                 // Transition into the reconnecting state
  2161.                                 // If this fails then that means that the user transitioned the connection into the disconnected or connecting state within the above error handler trigger.
  2162.                                 if (!transportLogic.ensureReconnectingState(instance)) {
  2163.                                     return;
  2164.                                 }
  2165.                                 // Call poll with the raiseReconnect flag as true after the reconnect delay
  2166.                                 privateData.pollTimeoutId = window.setTimeout(function () {
  2167.                                     poll(instance, true);
  2168.                                 }, that.reconnectDelay);
  2169.                             }
  2170.                         }
  2171.                     });
  2172.                     // This will only ever pass after an error has occurred via the poll ajax procedure.
  2173.                     if (reconnecting && raiseReconnect === true) {
  2174.                         // We wait to reconnect depending on how many times we've failed to reconnect.
  2175.                         // This is essentially a heuristic that will exponentially increase in wait time before
  2176.                         // triggering reconnected.  This depends on the "error" handler of Poll to cancel this
  2177.                         // timeout if it triggers before the Reconnected event fires.
  2178.                         // The Math.min at the end is to ensure that the reconnect timeout does not overflow.
  2179.                         privateData.reconnectTimeoutId = window.setTimeout(function () { fireReconnected(instance); }, Math.min(1000 * (Math.pow(2, reconnectErrors) - 1), maxFireReconnectedTimeout));
  2180.                     }
  2181.                 }(connection));
  2182.             }, 250); // Have to delay initial poll so Chrome doesn't show loader spinner in tab
  2183.         },
  2184.         lostConnection: function (connection) {
  2185.             if (connection.pollXhr) {
  2186.                 connection.pollXhr.abort("lostConnection");
  2187.             }
  2188.         },
  2189.         send: function (connection, data) {
  2190.             transportLogic.ajaxSend(connection, data);
  2191.         },
  2192.         stop: function (connection) {
  2193.             /// <summary>Stops the long polling connection</summary>
  2194.             /// <param name="connection" type="signalR">The SignalR connection to stop</param>
  2195.             window.clearTimeout(connection._.pollTimeoutId);
  2196.             window.clearTimeout(connection._.reconnectTimeoutId);
  2197.             delete connection._.pollTimeoutId;
  2198.             delete connection._.reconnectTimeoutId;
  2199.             if (connection.pollXhr) {
  2200.                 connection.pollXhr.abort();
  2201.                 connection.pollXhr = null;
  2202.                 delete connection.pollXhr;
  2203.             }
  2204.         },
  2205.         abort: function (connection, async) {
  2206.             transportLogic.ajaxAbort(connection, async);
  2207.         }
  2208.     };
  2209. }(window.jQuery, window));
  2210. /* jquery.signalR.hubs.js */
  2211. // Copyright (c) .NET Foundation. All rights reserved.
  2212. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  2213. /*global window:false */
  2214. /// <reference path="jquery.signalR.core.js" />
  2215. (function ($, window, undefined) {
  2216.     var nextGuid = 0;
  2217.     var eventNamespace = ".hubProxy",
  2218.         signalR = $.signalR;
  2219.     function makeEventName(event) {
  2220.         return event + eventNamespace;
  2221.     }
  2222.     // Equivalent to Array.prototype.map
  2223.     function map(arr, fun, thisp) {
  2224.         var i,
  2225.             length = arr.length,
  2226.             result = [];
  2227.         for (i = 0; i < length; i += 1) {
  2228.             if (arr.hasOwnProperty(i)) {
  2229.                 result[i] = fun.call(thisp, arr[i], i, arr);
  2230.             }
  2231.         }
  2232.         return result;
  2233.     }
  2234.     function getArgValue(a) {
  2235.         return $.isFunction(a) ? null : ($.type(a) === "undefined" ? null : a);
  2236.     }
  2237.     function hasMembers(obj) {
  2238.         for (var key in obj) {
  2239.             // If we have any properties in our callback map then we have callbacks and can exit the loop via return
  2240.             if (obj.hasOwnProperty(key)) {
  2241.                 return true;
  2242.             }
  2243.         }
  2244.         return false;
  2245.     }
  2246.     function clearInvocationCallbacks(connection, error) {
  2247.         /// <param name="connection" type="hubConnection" />
  2248.         var callbacks = connection._.invocationCallbacks,
  2249.             callback;
  2250.         if (hasMembers(callbacks)) {
  2251.             connection.log("Clearing hub invocation callbacks with error: " + error + ".");
  2252.         }
  2253.         // Reset the callback cache now as we have a local var referencing it
  2254.         connection._.invocationCallbackId = 0;
  2255.         delete connection._.invocationCallbacks;
  2256.         connection._.invocationCallbacks = {};
  2257.         // Loop over the callbacks and invoke them.
  2258.         // We do this using a local var reference and *after* we've cleared the cache
  2259.         // so that if a fail callback itself tries to invoke another method we don't
  2260.         // end up with its callback in the list we're looping over.
  2261.         for (var callbackId in callbacks) {
  2262.             callback = callbacks[callbackId];
  2263.             callback.method.call(callback.scope, { E: error });
  2264.         }
  2265.     }
  2266.     function isCallbackFromGeneratedHubProxy(callback) {
  2267.         // https://github.com/SignalR/SignalR/issues/4310
  2268.         // The stringified callback from the old generated hub proxy is 137 characters in Edge, Firefox and Chrome.
  2269.         // We slice to avoid wasting too many cycles searching through the text of a long large function.
  2270.         return $.isFunction(callback) && callback.toString().slice(0, 256).indexOf("// Call the client hub method") >= 0;
  2271.     }
  2272.     // hubProxy
  2273.     function hubProxy(hubConnection, hubName) {
  2274.         /// <summary>
  2275.         ///     Creates a new proxy object for the given hub connection that can be used to invoke
  2276.         ///     methods on server hubs and handle client method invocation requests from the server.
  2277.         /// </summary>
  2278.         return new hubProxy.fn.init(hubConnection, hubName);
  2279.     }
  2280.     hubProxy.fn = hubProxy.prototype = {
  2281.         init: function (connection, hubName) {
  2282.             this.state = {};
  2283.             this.connection = connection;
  2284.             this.hubName = hubName;
  2285.             this._ = {
  2286.                 callbackMap: {}
  2287.             };
  2288.         },
  2289.         constructor: hubProxy,
  2290.         hasSubscriptions: function () {
  2291.             return hasMembers(this._.callbackMap);
  2292.         },
  2293.         on: function (eventName, callback, callbackIdentity) {
  2294.             /// <summary>Wires up a callback to be invoked when a invocation request is received from the server hub.</summary>
  2295.             /// <param name="eventName" type="String">The name of the hub event to register the callback for.</param>
  2296.             /// <param name="callback" type="Function">The callback to be invoked.</param>
  2297.             /// <param name="callbackIdentity" type="Object">An optional object to use as the "identity" for the callback when checking if the handler has already been registered. Defaults to the value of 'callback' if not provided.</param>
  2298.             var that = this,
  2299.                 callbackMap = that._.callbackMap,
  2300.                 isFromOldGeneratedHubProxy = !callbackIdentity && isCallbackFromGeneratedHubProxy(callback);
  2301.             // We need the third "identity" argument because the registerHubProxies call made by signalr/js wraps the user-provided callback in a custom wrapper which breaks the identity comparison.
  2302.             // callbackIdentity allows the caller of `on` to provide a separate object to use as the "identity". `registerHubProxies` uses the original user callback as this identity object.
  2303.             callbackIdentity = callbackIdentity || callback;
  2304.             // Assign a global ID to the identity object. This tags the object so we can detect the same object when it comes back.
  2305.             if(!callbackIdentity._signalRGuid) {
  2306.                 callbackIdentity._signalRGuid = nextGuid++;
  2307.             }
  2308.             // Normalize the event name to lowercase
  2309.             eventName = eventName.toLowerCase();
  2310.             // If there is not an event registered for this callback yet we want to create its event space in the callback map.
  2311.             var callbackSpace = callbackMap[eventName];
  2312.             if (!callbackSpace) {
  2313.                 callbackSpace = [];
  2314.                 callbackMap[eventName] = callbackSpace;
  2315.             }
  2316.             // Check if there's already a registration
  2317.             var registration;
  2318.             for (var i = 0; i < callbackSpace.length; i++) {
  2319.                 if (callbackSpace[i].guid === callbackIdentity._signalRGuid || (isFromOldGeneratedHubProxy && callbackSpace[i].isFromOldGeneratedHubProxy)) {
  2320.                     registration = callbackSpace[i];
  2321.                 }
  2322.             }
  2323.             // Create a registration if there isn't one already
  2324.             if (!registration) {
  2325.                 registration = {
  2326.                     guid: callbackIdentity._signalRGuid,
  2327.                     eventHandlers: [],
  2328.                     isFromOldGeneratedHubProxy: isFromOldGeneratedHubProxy
  2329.                 };
  2330.                 callbackMap[eventName].push(registration);
  2331.             }
  2332.             var handler = function (e, data) {
  2333.                 callback.apply(that, data);
  2334.             };
  2335.             registration.eventHandlers.push(handler);
  2336.             $(that).bind(makeEventName(eventName), handler);
  2337.             return that;
  2338.         },
  2339.         off: function (eventName, callback, callbackIdentity) {
  2340.             /// <summary>Removes the callback invocation request from the server hub for the given event name.</summary>
  2341.             /// <param name="eventName" type="String">The name of the hub event to unregister the callback for.</param>
  2342.             /// <param name="callback" type="Function">The callback to be removed.</param>
  2343.             /// <param name="callbackIdentity" type="Object">An optional object to use as the "identity" when looking up the callback. Corresponds to the same parameter provided to 'on'. Defaults to the value of 'callback' if not provided.</param>
  2344.             var that = this,
  2345.                 callbackMap = that._.callbackMap,
  2346.                 callbackSpace,
  2347.                 isFromOldGeneratedHubProxy = !callbackIdentity && isCallbackFromGeneratedHubProxy(callback);
  2348.             callbackIdentity = callbackIdentity || callback;
  2349.             // Normalize the event name to lowercase
  2350.             eventName = eventName.toLowerCase();
  2351.             callbackSpace = callbackMap[eventName];
  2352.             // Verify that there is an event space to unbind
  2353.             if (callbackSpace) {
  2354.                 if (callback) {
  2355.                     // Find the callback registration
  2356.                     var callbackRegistration;
  2357.                     var callbackIndex;
  2358.                     for (var i = 0; i < callbackSpace.length; i++) {
  2359.                         if (callbackSpace[i].guid === callbackIdentity._signalRGuid || (isFromOldGeneratedHubProxy && callbackSpace[i].isFromOldGeneratedHubProxy)) {
  2360.                             callbackIndex = i;
  2361.                             callbackRegistration = callbackSpace[i];
  2362.                         }
  2363.                     }
  2364.                     // Only unbind if there's an event bound with eventName and a callback with the specified callback
  2365.                     if (callbackRegistration) {
  2366.                         // Unbind all event handlers associated with the registration.
  2367.                         for (var j = 0; j < callbackRegistration.eventHandlers.length; j++) {
  2368.                             $(that).unbind(makeEventName(eventName), callbackRegistration.eventHandlers[j]);
  2369.                         }
  2370.                         // Remove the registration from the list
  2371.                         callbackSpace.splice(i, 1);
  2372.                         // Check if there are any registrations left, if not we need to destroy it.
  2373.                         if (callbackSpace.length === 0) {
  2374.                             delete callbackMap[eventName];
  2375.                         }
  2376.                     }
  2377.                 } else if (!callback) { // Check if we're removing the whole event and we didn't error because of an invalid callback
  2378.                     $(that).unbind(makeEventName(eventName));
  2379.                     delete callbackMap[eventName];
  2380.                 }
  2381.             }
  2382.             return that;
  2383.         },
  2384.         invoke: function (methodName) {
  2385.             /// <summary>Invokes a server hub method with the given arguments.</summary>
  2386.             /// <param name="methodName" type="String">The name of the server hub method.</param>
  2387.             var that = this,
  2388.                 connection = that.connection,
  2389.                 args = $.makeArray(arguments).slice(1),
  2390.                 argValues = map(args, getArgValue),
  2391.                 data = { H: that.hubName, M: methodName, A: argValues, I: connection._.invocationCallbackId },
  2392.                 d = $.Deferred(),
  2393.                 callback = function (minResult) {
  2394.                     var result = that._maximizeHubResponse(minResult),
  2395.                         source,
  2396.                         error;
  2397.                     // Update the hub state
  2398.                     $.extend(that.state, result.State);
  2399.                     if (result.Progress) {
  2400.                         if (d.notifyWith) {
  2401.                             // Progress is only supported in jQuery 1.7+
  2402.                             d.notifyWith(that, [result.Progress.Data]);
  2403.                         } else if (!connection._.progressjQueryVersionLogged) {
  2404.                             connection.log("A hub method invocation progress update was received but the version of jQuery in use (" + $.prototype.jquery + ") does not support progress updates. Upgrade to jQuery 1.7+ to receive progress notifications.");
  2405.                             connection._.progressjQueryVersionLogged = true;
  2406.                         }
  2407.                     } else if (result.Error) {
  2408.                         // Server hub method threw an exception, log it & reject the deferred
  2409.                         if (result.StackTrace) {
  2410.                             connection.log(result.Error + "\n" + result.StackTrace + ".");
  2411.                         }
  2412.                         // result.ErrorData is only set if a HubException was thrown
  2413.                         source = result.IsHubException ? "HubException" : "Exception";
  2414.                         error = signalR._.error(result.Error, source);
  2415.                         error.data = result.ErrorData;
  2416.                         connection.log(that.hubName + "." + methodName + " failed to execute. Error: " + error.message);
  2417.                         d.rejectWith(that, [error]);
  2418.                     } else {
  2419.                         // Server invocation succeeded, resolve the deferred
  2420.                         connection.log("Invoked " + that.hubName + "." + methodName);
  2421.                         d.resolveWith(that, [result.Result]);
  2422.                     }
  2423.                 };
  2424.             connection._.invocationCallbacks[connection._.invocationCallbackId.toString()] = { scope: that, method: callback };
  2425.             connection._.invocationCallbackId += 1;
  2426.             if (!$.isEmptyObject(that.state)) {
  2427.                 data.S = that.state;
  2428.             }
  2429.             connection.log("Invoking " + that.hubName + "." + methodName);
  2430.             connection.send(data);
  2431.             return d.promise();
  2432.         },
  2433.         _maximizeHubResponse: function (minHubResponse) {
  2434.             return {
  2435.                 State: minHubResponse.S,
  2436.                 Result: minHubResponse.R,
  2437.                 Progress: minHubResponse.P ? {
  2438.                     Id: minHubResponse.P.I,
  2439.                     Data: minHubResponse.P.D
  2440.                 } : null,
  2441.                 Id: minHubResponse.I,
  2442.                 IsHubException: minHubResponse.H,
  2443.                 Error: minHubResponse.E,
  2444.                 StackTrace: minHubResponse.T,
  2445.                 ErrorData: minHubResponse.D
  2446.             };
  2447.         }
  2448.     };
  2449.     hubProxy.fn.init.prototype = hubProxy.fn;
  2450.     // hubConnection
  2451.     function hubConnection(url, options) {
  2452.         /// <summary>Creates a new hub connection.</summary>
  2453.         /// <param name="url" type="String">[Optional] The hub route url, defaults to "/signalr".</param>
  2454.         /// <param name="options" type="Object">[Optional] Settings to use when creating the hubConnection.</param>
  2455.         var settings = {
  2456.             qs: null,
  2457.             logging: false,
  2458.             useDefaultPath: true
  2459.         };
  2460.         $.extend(settings, options);
  2461.         if (!url || settings.useDefaultPath) {
  2462.             url = (url || "") + "/signalr";
  2463.         }
  2464.         return new hubConnection.fn.init(url, settings);
  2465.     }
  2466.     hubConnection.fn = hubConnection.prototype = $.connection();
  2467.     hubConnection.fn.init = function (url, options) {
  2468.         var settings = {
  2469.             qs: null,
  2470.             logging: false,
  2471.             useDefaultPath: true
  2472.         },
  2473.             connection = this;
  2474.         $.extend(settings, options);
  2475.         // Call the base constructor
  2476.         $.signalR.fn.init.call(connection, url, settings.qs, settings.logging);
  2477.         // Object to store hub proxies for this connection
  2478.         connection.proxies = {};
  2479.         connection._.invocationCallbackId = 0;
  2480.         connection._.invocationCallbacks = {};
  2481.         // Wire up the received handler
  2482.         connection.received(function (minData) {
  2483.             var data, proxy, dataCallbackId, callback, hubName, eventName;
  2484.             if (!minData) {
  2485.                 return;
  2486.             }
  2487.             // We have to handle progress updates first in order to ensure old clients that receive
  2488.             // progress updates enter the return value branch and then no-op when they can't find
  2489.             // the callback in the map (because the minData.I value will not be a valid callback ID)
  2490.             // Process progress notification
  2491.             if (typeof (minData.P) !== "undefined") {
  2492.                 dataCallbackId = minData.P.I.toString();
  2493.                 callback = connection._.invocationCallbacks[dataCallbackId];
  2494.                 if (callback) {
  2495.                     callback.method.call(callback.scope, minData);
  2496.                 }
  2497.             } else if (typeof (minData.I) !== "undefined") {
  2498.                 // We received the return value from a server method invocation, look up callback by id and call it
  2499.                 dataCallbackId = minData.I.toString();
  2500.                 callback = connection._.invocationCallbacks[dataCallbackId];
  2501.                 if (callback) {
  2502.                     // Delete the callback from the proxy
  2503.                     connection._.invocationCallbacks[dataCallbackId] = null;
  2504.                     delete connection._.invocationCallbacks[dataCallbackId];
  2505.                     // Invoke the callback
  2506.                     callback.method.call(callback.scope, minData);
  2507.                 }
  2508.             } else {
  2509.                 data = this._maximizeClientHubInvocation(minData);
  2510.                 // We received a client invocation request, i.e. broadcast from server hub
  2511.                 connection.log("Triggering client hub event '" + data.Method + "' on hub '" + data.Hub + "'.");
  2512.                 // Normalize the names to lowercase
  2513.                 hubName = data.Hub.toLowerCase();
  2514.                 eventName = data.Method.toLowerCase();
  2515.                 // Trigger the local invocation event
  2516.                 proxy = this.proxies[hubName];
  2517.                 // Update the hub state
  2518.                 $.extend(proxy.state, data.State);
  2519.                 $(proxy).triggerHandler(makeEventName(eventName), [data.Args]);
  2520.             }
  2521.         });
  2522.         connection.error(function (errData, origData) {
  2523.             var callbackId, callback;
  2524.             if (!origData) {
  2525.                 // No original data passed so this is not a send error
  2526.                 return;
  2527.             }
  2528.             callbackId = origData.I;
  2529.             callback = connection._.invocationCallbacks[callbackId];
  2530.             // Verify that there is a callback bound (could have been cleared)
  2531.             if (callback) {
  2532.                 // Delete the callback
  2533.                 connection._.invocationCallbacks[callbackId] = null;
  2534.                 delete connection._.invocationCallbacks[callbackId];
  2535.                 // Invoke the callback with an error to reject the promise
  2536.                 callback.method.call(callback.scope, { E: errData });
  2537.             }
  2538.         });
  2539.         connection.reconnecting(function () {
  2540.             if (connection.transport && connection.transport.name === "webSockets") {
  2541.                 clearInvocationCallbacks(connection, "Connection started reconnecting before invocation result was received.");
  2542.             }
  2543.         });
  2544.         connection.disconnected(function () {
  2545.             clearInvocationCallbacks(connection, "Connection was disconnected before invocation result was received.");
  2546.         });
  2547.     };
  2548.     hubConnection.fn._maximizeClientHubInvocation = function (minClientHubInvocation) {
  2549.         return {
  2550.             Hub: minClientHubInvocation.H,
  2551.             Method: minClientHubInvocation.M,
  2552.             Args: minClientHubInvocation.A,
  2553.             State: minClientHubInvocation.S
  2554.         };
  2555.     };
  2556.     hubConnection.fn._registerSubscribedHubs = function () {
  2557.         /// <summary>
  2558.         ///     Sets the starting event to loop through the known hubs and register any new hubs
  2559.         ///     that have been added to the proxy.
  2560.         /// </summary>
  2561.         var connection = this;
  2562.         if (!connection._subscribedToHubs) {
  2563.             connection._subscribedToHubs = true;
  2564.             connection.starting(function () {
  2565.                 // Set the connection's data object with all the hub proxies with active subscriptions.
  2566.                 // These proxies will receive notifications from the server.
  2567.                 var subscribedHubs = [];
  2568.                 $.each(connection.proxies, function (key) {
  2569.                     if (this.hasSubscriptions()) {
  2570.                         subscribedHubs.push({ name: key });
  2571.                         connection.log("Client subscribed to hub '" + key + "'.");
  2572.                     }
  2573.                 });
  2574.                 if (subscribedHubs.length === 0) {
  2575.                     connection.log("No hubs have been subscribed to.  The client will not receive data from hubs.  To fix, declare at least one client side function prior to connection start for each hub you wish to subscribe to.");
  2576.                 }
  2577.                 connection.data = connection.json.stringify(subscribedHubs);
  2578.             });
  2579.         }
  2580.     };
  2581.     hubConnection.fn.createHubProxy = function (hubName) {
  2582.         /// <summary>
  2583.         ///     Creates a new proxy object for the given hub connection that can be used to invoke
  2584.         ///     methods on server hubs and handle client method invocation requests from the server.
  2585.         /// </summary>
  2586.         /// <param name="hubName" type="String">
  2587.         ///     The name of the hub on the server to create the proxy for.
  2588.         /// </param>
  2589.         // Normalize the name to lowercase
  2590.         hubName = hubName.toLowerCase();
  2591.         var proxy = this.proxies[hubName];
  2592.         if (!proxy) {
  2593.             proxy = hubProxy(this, hubName);
  2594.             this.proxies[hubName] = proxy;
  2595.         }
  2596.         this._registerSubscribedHubs();
  2597.         return proxy;
  2598.     };
  2599.     hubConnection.fn.init.prototype = hubConnection.fn;
  2600.     $.hubConnection = hubConnection;
  2601. }(window.jQuery, window));
  2602. /* jquery.signalR.version.js */
  2603. // Copyright (c) .NET Foundation. All rights reserved.
  2604. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
  2605. /*global window:false */
  2606. /// <reference path="jquery.signalR.core.js" />
  2607. (function ($, undefined) {
  2608.     // This will be modified by the build script
  2609.     $.signalR.version = "2.4.3";
  2610. }(window.jQuery));
复制代码
(4)效果展示:


三、.NET Core WebAPI使用SignalR

场景:项目中需要服务端主动向客户端发送通知消息,后端是使用.NETCore实现的,前端是使用Vue3全家桶项目,前后端分离模式。本次主要讲使用SignalR如何来实现的。

3.1-服务端(.NET Core WebAPI)

1、右击项目,点击【管理NuGet程序包】安装SignalR。

2、搜索【SignalR】,点击【安装】。


3、如果是.NET6以前的版本,在【Startup.cs】中配置:

如果是.NET6以后得版本,在【Program.cs】配置:
  1. /*! jQuery v1.6.4 http://jquery.com/ | http://jquery.org/license */
  2. (function(a,b){function cu(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cr(a){if(!cg[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){ch||(ch=c.createElement("iframe"),ch.frameBorder=ch.width=ch.height=0),b.appendChild(ch);if(!ci||!ch.createElement)ci=(ch.contentWindow||ch.contentDocument).document,ci.write((c.compatMode==="CSS1Compat"?"<!doctype html>":"")+"<html><body>"),ci.close();d=ci.createElement(a),ci.body.appendChild(d),e=f.css(d,"display"),b.removeChild(ch)}cg[a]=e}return cg[a]}function cq(a,b){var c={};f.each(cm.concat.apply([],cm.slice(0,b)),function(){c[this]=a});return c}function cp(){cn=b}function co(){setTimeout(cp,0);return cn=f.now()}function cf(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ce(){try{return new a.XMLHttpRequest}catch(b){}}function b$(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g<i;g++){if(g===1)for(h in a.converters)typeof h=="string"&&(e[h.toLowerCase()]=a.converters[h]);l=k,k=d[g];if(k==="*")k=l;else if(l!=="*"&&l!==k){m=l+" "+k,n=e[m]||e["* "+k];if(!n){p=b;for(o in e){j=o.split(" ");if(j[0]===l||j[0]==="*"){p=e[j[1]+" "+k];if(p){o=e[o],o===!0?n=p:p===!0&&(n=o);break}}}}!n&&!p&&f.error("No conversion from "+m.replace(" "," to ")),n!==!0&&(c=n?n(c):p(o(c)))}}return c}function bZ(a,c,d){var e=a.contents,f=a.dataTypes,g=a.responseFields,h,i,j,k;for(i in g)i in d&&(c[g[i]]=d[i]);while(f[0]==="*")f.shift(),h===b&&(h=a.mimeType||c.getResponseHeader("content-type"));if(h)for(i in e)if(e[i]&&e[i].test(h)){f.unshift(i);break}if(f[0]in d)j=f[0];else{for(i in d){if(!f[0]||a.converters[i+" "+f[0]]){j=i;break}k||(k=i)}j=j||k}if(j){j!==f[0]&&f.unshift(j);return d[j]}}function bY(a,b,c,d){if(f.isArray(b))f.each(b,function(b,e){c||bA.test(a)?d(a,e):bY(a+"["+(typeof e=="object"||f.isArray(e)?b:"")+"]",e,c,d)});else if(!c&&b!=null&&typeof b=="object")for(var e in b)bY(a+"["+e+"]",b[e],c,d);else d(a,b)}function bX(a,c){var d,e,g=f.ajaxSettings.flatOptions||{};for(d in c)c[d]!==b&&((g[d]?a:e||(e={}))[d]=c[d]);e&&f.extend(!0,a,e)}function bW(a,c,d,e,f,g){f=f||c.dataTypes[0],g=g||{},g[f]=!0;var h=a[f],i=0,j=h?h.length:0,k=a===bP,l;for(;i<j&&(k||!l);i++)l=h[i](c,d,e),typeof l=="string"&&(!k||g[l]?l=b:(c.dataTypes.unshift(l),l=bW(a,c,d,e,l,g)));(k||!l)&&!g["*"]&&(l=bW(a,c,d,e,"*",g));return l}function bV(a){return function(b,c){typeof b!="string"&&(c=b,b="*");if(f.isFunction(c)){var d=b.toLowerCase().split(bL),e=0,g=d.length,h,i,j;for(;e<g;e++)h=d[e],j=/^\+/.test(h),j&&(h=h.substr(1)||"*"),i=a[h]=a[h]||[],i[j?"unshift":"push"](c)}}}function by(a,b,c){var d=b==="width"?a.offsetWidth:a.offsetHeight,e=b==="width"?bt:bu;if(d>0){c!=="border"&&f.each(e,function(){c||(d-=parseFloat(f.css(a,"padding"+this))||0),c==="margin"?d+=parseFloat(f.css(a,c+this))||0:d-=parseFloat(f.css(a,"border"+this+"Width"))||0});return d+"px"}d=bv(a,b,b);if(d<0||d==null)d=a.style[b]||0;d=parseFloat(d)||0,c&&f.each(e,function(){d+=parseFloat(f.css(a,"padding"+this))||0,c!=="padding"&&(d+=parseFloat(f.css(a,"border"+this+"Width"))||0),c==="margin"&&(d+=parseFloat(f.css(a,c+this))||0)});return d+"px"}function bl(a,b){b.src?f.ajax({url:b.src,async:!1,dataType:"script"}):f.globalEval((b.text||b.textContent||b.innerHTML||"").replace(bd,"/*$0*/")),b.parentNode&&b.parentNode.removeChild(b)}function bk(a){f.nodeName(a,"input")?bj(a):"getElementsByTagName"in a&&f.grep(a.getElementsByTagName("input"),bj)}function bj(a){if(a.type==="checkbox"||a.type==="radio")a.defaultChecked=a.checked}function bi(a){return"getElementsByTagName"in a?a.getElementsByTagName("*"):"querySelectorAll"in a?a.querySelectorAll("*"):[]}function bh(a,b){var c;if(b.nodeType===1){b.clearAttributes&&b.clearAttributes(),b.mergeAttributes&&b.mergeAttributes(a),c=b.nodeName.toLowerCase();if(c==="object")b.outerHTML=a.outerHTML;else if(c!=="input"||a.type!=="checkbox"&&a.type!=="radio"){if(c==="option")b.selected=a.defaultSelected;else if(c==="input"||c==="textarea")b.defaultValue=a.defaultValue}else a.checked&&(b.defaultChecked=b.checked=a.checked),b.value!==a.value&&(b.value=a.value);b.removeAttribute(f.expando)}}function bg(a,b){if(b.nodeType===1&&!!f.hasData(a)){var c=f.expando,d=f.data(a),e=f.data(b,d);if(d=d[c]){var g=d.events;e=e[c]=f.extend({},d);if(g){delete e.handle,e.events={};for(var h in g)for(var i=0,j=g[h].length;i<j;i++)f.event.add(b,h+(g[h][i].namespace?".":"")+g[h][i].namespace,g[h][i],g[h][i].data)}}}}function bf(a,b){return f.nodeName(a,"table")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function V(a,b,c){b=b||0;if(f.isFunction(b))return f.grep(a,function(a,d){var e=!!b.call(a,d,a);return e===c});if(b.nodeType)return f.grep(a,function(a,d){return a===b===c});if(typeof b=="string"){var d=f.grep(a,function(a){return a.nodeType===1});if(Q.test(b))return f.filter(b,d,!c);b=f.filter(b,d)}return f.grep(a,function(a,d){return f.inArray(a,b)>=0===c})}function U(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function M(a,b){return(a&&a!=="*"?a+".":"")+b.replace(y,"`").replace(z,"&")}function L(a){var b,c,d,e,g,h,i,j,k,l,m,n,o,p=[],q=[],r=f._data(this,"events");if(!(a.liveFired===this||!r||!r.live||a.target.disabled||a.button&&a.type==="click")){a.namespace&&(n=new RegExp("(^|\\.)"+a.namespace.split(".").join("\\.(?:.*\\.)?")+"(\\.|$)")),a.liveFired=this;var s=r.live.slice(0);for(i=0;i<s.length;i++)g=s[i],g.origType.replace(w,"")===a.type?q.push(g.selector):s.splice(i--,1);e=f(a.target).closest(q,a.currentTarget);for(j=0,k=e.length;j<k;j++){m=e[j];for(i=0;i<s.length;i++){g=s[i];if(m.selector===g.selector&&(!n||n.test(g.namespace))&&!m.elem.disabled){h=m.elem,d=null;if(g.preType==="mouseenter"||g.preType==="mouseleave")a.type=g.preType,d=f(a.relatedTarget).closest(g.selector)[0],d&&f.contains(h,d)&&(d=h);(!d||d!==h)&&p.push({elem:h,handleObj:g,level:m.level})}}}for(j=0,k=p.length;j<k;j++){e=p[j];if(c&&e.level>c)break;a.currentTarget=e.elem,a.data=e.handleObj.data,a.handleObj=e.handleObj,o=e.handleObj.origHandler.apply(e.elem,arguments);if(o===!1||a.isPropagationStopped()){c=e.level,o===!1&&(b=!1);if(a.isImmediatePropagationStopped())break}}return b}}function J(a,c,d){var e=f.extend({},d[0]);e.type=a,e.originalEvent={},e.liveFired=b,f.event.handle.call(c,e),e.isDefaultPrevented()&&d[0].preventDefault()}function D(){return!0}function C(){return!1}function m(a,c,d){var e=c+"defer",g=c+"queue",h=c+"mark",i=f.data(a,e,b,!0);i&&(d==="queue"||!f.data(a,g,b,!0))&&(d==="mark"||!f.data(a,h,b,!0))&&setTimeout(function(){!f.data(a,g,b,!0)&&!f.data(a,h,b,!0)&&(f.removeData(a,e,!0),i.resolve())},0)}function l(a){for(var b in a)if(b!=="toJSON")return!1;return!0}function k(a,c,d){if(d===b&&a.nodeType===1){var e="data-"+c.replace(j,"-$1").toLowerCase();d=a.getAttribute(e);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNaN(d)?i.test(d)?f.parseJSON(d):d:parseFloat(d)}catch(g){}f.data(a,c,d)}else d=b}return d}var c=a.document,d=a.navigator,e=a.location,f=function(){function K(){if(!e.isReady){try{c.documentElement.doScroll("left")}catch(a){setTimeout(K,1);return}e.ready()}}var e=function(a,b){return new e.fn.init(a,b,h)},f=a.jQuery,g=a.$,h,i=/^(?:[^#<]*(<[\w\W]+>)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/\d/,n=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,o=/^[\],:{}\s]*$/,p=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,q=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,r=/(?:^|:|,)(?:\s*\[)+/g,s=/(webkit)[ \/]([\w.]+)/,t=/(opera)(?:.*version)?[ \/]([\w.]+)/,u=/(msie) ([\w.]+)/,v=/(mozilla)(?:.*? rv:([\w.]+))?/,w=/-([a-z]|[0-9])/ig,x=/^-ms-/,y=function(a,b){return(b+"").toUpperCase()},z=d.userAgent,A,B,C,D=Object.prototype.toString,E=Object.prototype.hasOwnProperty,F=Array.prototype.push,G=Array.prototype.slice,H=String.prototype.trim,I=Array.prototype.indexOf,J={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)!=="<"||a.charAt(a.length-1)!==">"||a.length<3?g=i.exec(a):g=[null,a,null];if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=n.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.6.4",length:0,size:function(){return this.length},toArray:function(){return G.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?F.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),B.done(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(G.apply(this,arguments),"slice",G.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:F,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j<k;j++)if((a=arguments[j])!=null)for(c in a){d=i[c],f=a[c];if(i===f)continue;l&&f&&(e.isPlainObject(f)||(g=e.isArray(f)))?(g?(g=!1,h=d&&e.isArray(d)?d:[]):h=d&&e.isPlainObject(d)?d:{},i[c]=e.extend(l,h,f)):f!==b&&(i[c]=f)}return i},e.extend({noConflict:function(b){a.$===e&&(a.$=g),b&&a.jQuery===e&&(a.jQuery=f);return e},isReady:!1,readyWait:1,holdReady:function(a){a?e.readyWait++:e.ready(!0)},ready:function(a){if(a===!0&&!--e.readyWait||a!==!0&&!e.isReady){if(!c.body)return setTimeout(e.ready,1);e.isReady=!0;if(a!==!0&&--e.readyWait>0)return;B.resolveWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").unbind("ready")}},bindReady:function(){if(!B){B=e._Deferred();if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",C,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",C),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&K()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNaN:function(a){return a==null||!m.test(a)||isNaN(a)},type:function(a){return a==null?String(a):J[D.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;try{if(a.constructor&&!E.call(a,"constructor")&&!E.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}var d;for(d in a);return d===b||E.call(a,d)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw a},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(o.test(b.replace(p,"@").replace(q,"]").replace(r,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(c){var d,f;try{a.DOMParser?(f=new DOMParser,d=f.parseFromString(c,"text/xml")):(d=new ActiveXObject("Microsoft.XMLDOM"),d.async="false",d.loadXML(c))}catch(g){d=b}(!d||!d.documentElement||d.getElementsByTagName("parsererror").length)&&e.error("Invalid XML: "+c);return d},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(x,"ms-").replace(w,y)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g<h;)if(c.apply(a[g++],d)===!1)break}else if(i){for(f in a)if(c.call(a[f],f,a[f])===!1)break}else for(;g<h;)if(c.call(a[g],g,a[g++])===!1)break;return a},trim:H?function(a){return a==null?"":H.call(a)}:function(a){return a==null?"":(a+"").replace(k,"").replace(l,"")},makeArray:function(a,b){var c=b||[];if(a!=null){var d=e.type(a);a.length==null||d==="string"||d==="function"||d==="regexp"||e.isWindow(a)?F.call(c,a):e.merge(c,a)}return c},inArray:function(a,b){if(!b)return-1;if(I)return I.call(b,a);for(var c=0,d=b.length;c<d;c++)if(b[c]===a)return c;return-1},merge:function(a,c){var d=a.length,e=0;if(typeof c.length=="number")for(var f=c.length;e<f;e++)a[d++]=c[e];else while(c[e]!==b)a[d++]=c[e++];a.length=d;return a},grep:function(a,b,c){var d=[],e;c=!!c;for(var f=0,g=a.length;f<g;f++)e=!!b(a[f],f),c!==e&&d.push(a[f]);return d},map:function(a,c,d){var f,g,h=[],i=0,j=a.length,k=a instanceof e||j!==b&&typeof j=="number"&&(j>0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i<j;i++)f=c(a[i],i,d),f!=null&&(h[h.length]=f);else for(g in a)f=c(a[g],g,d),f!=null&&(h[h.length]=f);return h.concat.apply([],h)},guid:1,proxy:function(a,c){if(typeof c=="string"){var d=a[c];c=a,a=d}if(!e.isFunction(a))return b;var f=G.call(arguments,2),g=function(){return a.apply(c,f.concat(G.call(arguments)))};g.guid=a.guid=a.guid||g.guid||e.guid++;return g},access:function(a,c,d,f,g,h){var i=a.length;if(typeof c=="object"){for(var j in c)e.access(a,j,c[j],f,g,d);return a}if(d!==b){f=!h&&f&&e.isFunction(d);for(var k=0;k<i;k++)g(a[k],c,f?d.call(a[k],k,g(a[k],c)):d,h);return a}return i?g(a[0],c):b},now:function(){return(new Date).getTime()},uaMatch:function(a){a=a.toLowerCase();var b=s.exec(a)||t.exec(a)||u.exec(a)||a.indexOf("compatible")<0&&v.exec(a)||[];return{browser:b[1]||"",version:b[2]||"0"}},sub:function(){function a(b,c){return new a.fn.init(b,c)}e.extend(!0,a,this),a.superclass=this,a.fn=a.prototype=this(),a.fn.constructor=a,a.sub=this.sub,a.fn.init=function(d,f){f&&f instanceof e&&!(f instanceof a)&&(f=a(f));return e.fn.init.call(this,d,f,b)},a.fn.init.prototype=a.fn;var b=a(c);return a},browser:{}}),e.each("Boolean Number String Function Array Date RegExp Object".split(" "),function(a,b){J["[object "+b+"]"]=b.toLowerCase()}),A=e.uaMatch(z),A.browser&&(e.browser[A.browser]=!0,e.browser.version=A.version),e.browser.webkit&&(e.browser.safari=!0),j.test(" ")&&(k=/^[\s\xA0]+/,l=/[\s\xA0]+$/),h=e(c),c.addEventListener?C=function(){c.removeEventListener("DOMContentLoaded",C,!1),e.ready()}:c.attachEvent&&(C=function(){c.readyState==="complete"&&(c.detachEvent("onreadystatechange",C),e.ready())});return e}(),g="done fail isResolved isRejected promise then always pipe".split(" "),h=[].slice;f.extend({_Deferred:function(){var a=[],b,c,d,e={done:function(){if(!d){var c=arguments,g,h,i,j,k;b&&(k=b,b=0);for(g=0,h=c.length;g<h;g++)i=c[g],j=f.type(i),j==="array"?e.done.apply(e,i):j==="function"&&a.push(i);k&&e.resolveWith(k[0],k[1])}return this},resolveWith:function(e,f){if(!d&&!b&&!c){f=f||[],c=1;try{while(a[0])a.shift().apply(e,f)}finally{b=[e,f],c=0}}return this},resolve:function(){e.resolveWith(this,arguments);return this},isResolved:function(){return!!c||!!b},cancel:function(){d=1,a=[];return this}};return e},Deferred:function(a){var b=f._Deferred(),c=f._Deferred(),d;f.extend(b,{then:function(a,c){b.done(a).fail(c);return this},always:function(){return b.done.apply(b,arguments).fail.apply(this,arguments)},fail:c.done,rejectWith:c.resolveWith,reject:c.resolve,isRejected:c.isResolved,pipe:function(a,c){return f.Deferred(function(d){f.each({done:[a,"resolve"],fail:[c,"reject"]},function(a,c){var e=c[0],g=c[1],h;f.isFunction(e)?b[a](function(){h=e.apply(this,arguments),h&&f.isFunction(h.promise)?h.promise().then(d.resolve,d.reject):d[g+"With"](this===b?d:this,[h])}):b[a](d[g])})}).promise()},promise:function(a){if(a==null){if(d)return d;d=a={}}var c=g.length;while(c--)a[g[c]]=b[g[c]];return a}}),b.done(c.cancel).fail(b.cancel),delete b.cancel,a&&a.call(b,b);return b},when:function(a){function i(a){return function(c){b[a]=arguments.length>1?h.call(arguments,0):c,--e||g.resolveWith(g,h.call(b,0))}}var b=arguments,c=0,d=b.length,e=d,g=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred();if(d>1){for(;c<d;c++)b[c]&&f.isFunction(b[c].promise)?b[c].promise().then(i(c),g.reject):--e;e||g.resolveWith(g,b)}else g!==a&&g.resolveWith(g,d?[a]:[]);return g.promise()}}),f.support=function(){var a=c.createElement("div"),b=c.documentElement,d,e,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u;a.setAttribute("className","t"),a.innerHTML="   <link/><table></table><a target="_blank" href='https://www.cnblogs.com/a' style='top:1px;float:left;opacity:.55;'>a</a><input type='checkbox'/>",d=a.getElementsByTagName("*"),e=a.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};g=c.createElement("select"),h=g.appendChild(c.createElement("option")),i=a.getElementsByTagName("input")[0],k={leadingWhitespace:a.firstChild.nodeType===3,tbody:!a.getElementsByTagName("tbody").length,htmlSerialize:!!a.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55$/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:i.value==="on",optSelected:h.selected,getSetAttribute:a.className!=="t",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},i.checked=!0,k.noCloneChecked=i.cloneNode(!0).checked,g.disabled=!0,k.optDisabled=!h.disabled;try{delete a.test}catch(v){k.deleteExpando=!1}!a.addEventListener&&a.attachEvent&&a.fireEvent&&(a.attachEvent("onclick",function(){k.noCloneEvent=!1}),a.cloneNode(!0).fireEvent("onclick")),i=c.createElement("input"),i.value="t",i.setAttribute("type","radio"),k.radioValue=i.value==="t",i.setAttribute("checked","checked"),a.appendChild(i),l=c.createDocumentFragment(),l.appendChild(a.firstChild),k.checkClone=l.cloneNode(!0).cloneNode(!0).lastChild.checked,a.innerHTML="",a.style.width=a.style.paddingLeft="1px",m=c.getElementsByTagName("body")[0],o=c.createElement(m?"div":"body"),p={visibility:"hidden",width:0,height:0,border:0,margin:0,background:"none"},m&&f.extend(p,{position:"absolute",left:"-1000px",top:"-1000px"});for(t in p)o.style[t]=p[t];o.appendChild(a),n=m||b,n.insertBefore(o,n.firstChild),k.appendChecked=i.checked,k.boxModel=a.offsetWidth===2,"zoom"in a.style&&(a.style.display="inline",a.style.zoom=1,k.inlineBlockNeedsLayout=a.offsetWidth===2,a.style.display="",a.innerHTML="",k.shrinkWrapBlocks=a.offsetWidth!==2),a.innerHTML="<table><tr><td style='padding:0;border:0;display:none'></td><td>t</td></tr></table>",q=a.getElementsByTagName("td"),u=q[0].offsetHeight===0,q[0].style.display="",q[1].style.display="none",k.reliableHiddenOffsets=u&&q[0].offsetHeight===0,a.innerHTML="",c.defaultView&&c.defaultView.getComputedStyle&&(j=c.createElement("div"),j.style.width="0",j.style.marginRight="0",a.appendChild(j),k.reliableMarginRight=(parseInt((c.defaultView.getComputedStyle(j,null)||{marginRight:0}).marginRight,10)||0)===0),o.innerHTML="",n.removeChild(o);if(a.attachEvent)for(t in{submit:1,change:1,focusin:1})s="on"+t,u=s in a,u||(a.setAttribute(s,"return;"),u=typeof a[s]=="function"),k[t+"Bubbles"]=u;o=l=g=h=m=j=a=i=null;return k}(),f.boxModel=f.support.boxModel;var i=/^(?:\{.*\}|\[.*\])$/,j=/([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!l(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g,h,i=f.expando,j=typeof c=="string",k=a.nodeType,l=k?f.cache:a,m=k?a[f.expando]:a[f.expando]&&f.expando;if((!m||e&&m&&l[m]&&!l[m][i])&&j&&d===b)return;m||(k?a[f.expando]=m=++f.uuid:m=f.expando),l[m]||(l[m]={},k||(l[m].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?l[m][i]=f.extend(l[m][i],c):l[m]=f.extend(l[m],c);g=l[m],e&&(g[i]||(g[i]={}),g=g[i]),d!==b&&(g[f.camelCase(c)]=d);if(c==="events"&&!g[c])return g[i]&&g[i].events;j?(h=g[c],h==null&&(h=g[f.camelCase(c)])):h=g;return h}},removeData:function(a,b,c){if(!!f.acceptData(a)){var d,e=f.expando,g=a.nodeType,h=g?f.cache:a,i=g?a[f.expando]:f.expando;if(!h[i])return;if(b){d=c?h[i][e]:h[i];if(d){d[b]||(b=f.camelCase(b)),delete d[b];if(!l(d))return}}if(c){delete h[i][e];if(!l(h[i]))return}var j=h[i][e];f.support.deleteExpando||!h.setInterval?delete h[i]:h[i]=null,j?(h[i]={},g||(h[i].toJSON=f.noop),h[i][e]=j):g&&(f.support.deleteExpando?delete a[f.expando]:a.removeAttribute?a.removeAttribute(f.expando):a[f.expando]=null)}},_data:function(a,b,c){return f.data(a,b,c,!0)},acceptData:function(a){if(a.nodeName){var b=f.noData[a.nodeName.toLowerCase()];if(b)return b!==!0&&a.getAttribute("classid")===b}return!0}}),f.fn.extend({data:function(a,c){var d=null;if(typeof a=="undefined"){if(this.length){d=f.data(this[0]);if(this[0].nodeType===1){var e=this[0].attributes,g;for(var h=0,i=e.length;h<i;h++)g=e[h].name,g.indexOf("data-")===0&&(g=f.camelCase(g.substring(5)),k(this[0],g,d[g]))}}return d}if(typeof a=="object")return this.each(function(){f.data(this,a)});var j=a.split(".");j[1]=j[1]?"."+j[1]:"";if(c===b){d=this.triggerHandler("getData"+j[1]+"!",[j[0]]),d===b&&this.length&&(d=f.data(this[0],a),d=k(this[0],a,d));return d===b&&j[1]?this.data(j[0]):d}return this.each(function(){var b=f(this),d=[j[0],c];b.triggerHandler("setData"+j[1]+"!",d),f.data(this,a,c),b.triggerHandler("changeData"+j[1]+"!",d)})},removeData:function(a){return this.each(function(){f.removeData(this,a)})}}),f.extend({_mark:function(a,c){a&&(c=(c||"fx")+"mark",f.data(a,c,(f.data(a,c,b,!0)||0)+1,!0))},_unmark:function(a,c,d){a!==!0&&(d=c,c=a,a=!1);if(c){d=d||"fx";var e=d+"mark",g=a?0:(f.data(c,e,b,!0)||1)-1;g?f.data(c,e,g,!0):(f.removeData(c,e,!0),m(c,d,"mark"))}},queue:function(a,c,d){if(a){c=(c||"fx")+"queue";var e=f.data(a,c,b,!0);d&&(!e||f.isArray(d)?e=f.data(a,c,f.makeArray(d),!0):e.push(d));return e||[]}},dequeue:function(a,b){b=b||"fx";var c=f.queue(a,b),d=c.shift(),e;d==="inprogress"&&(d=c.shift()),d&&(b==="fx"&&c.unshift("inprogress"),d.call(a,function(){f.dequeue(a,b)})),c.length||(f.removeData(a,b+"queue",!0),m(a,b,"queue"))}}),f.fn.extend({queue:function(a,c){typeof a!="string"&&(c=a,a="fx");if(c===b)return f.queue(this[0],a);return this.each(function(){var b=f.queue(this,a,c);a==="fx"&&b[0]!=="inprogress"&&f.dequeue(this,a)})},dequeue:function(a){return this.each(function(){f.dequeue(this,a)})},delay:function(a,b){a=f.fx?f.fx.speeds[a]||a:a,b=b||"fx";return this.queue(b,function(){var c=this;setTimeout(function(){f.dequeue(c,b)},a)})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,c){function m(){--h||d.resolveWith(e,[e])}typeof a!="string"&&(c=a,a=b),a=a||"fx";var d=f.Deferred(),e=this,g=e.length,h=1,i=a+"defer",j=a+"queue",k=a+"mark",l;while(g--)if(l=f.data(e[g],i,b,!0)||(f.data(e[g],j,b,!0)||f.data(e[g],k,b,!0))&&f.data(e[g],i,f._Deferred(),!0))h++,l.done(m);m();return d.promise()}});var n=/[\n\t\r]/g,o=/\s+/,p=/\r/g,q=/^(?:button|input)$/i,r=/^(?:button|input|object|select|textarea)$/i,s=/^a(?:rea)?$/i,t=/^(?:autofocus|autoplay|async|checked|controls|defer|disabled|hidden|loop|multiple|open|readonly|required|scoped|selected)$/i,u,v;f.fn.extend({attr:function(a,b){return f.access(this,a,b,!0,f.attr)},removeAttr:function(a){return this.each(function(){f.removeAttr(this,a)})},prop:function(a,b){return f.access(this,a,b,!0,f.prop)},removeProp:function(a){a=f.propFix[a]||a;return this.each(function(){try{this[a]=b,delete this[a]}catch(c){}})},addClass:function(a){var b,c,d,e,g,h,i;if(f.isFunction(a))return this.each(function(b){f(this).addClass(a.call(this,b,this.className))});if(a&&typeof a=="string"){b=a.split(o);for(c=0,d=this.length;c<d;c++){e=this[c];if(e.nodeType===1)if(!e.className&&b.length===1)e.className=a;else{g=" "+e.className+" ";for(h=0,i=b.length;h<i;h++)~g.indexOf(" "+b[h]+" ")||(g+=b[h]+" ");e.className=f.trim(g)}}}return this},removeClass:function(a){var c,d,e,g,h,i,j;if(f.isFunction(a))return this.each(function(b){f(this).removeClass(a.call(this,b,this.className))});if(a&&typeof a=="string"||a===b){c=(a||"").split(o);for(d=0,e=this.length;d<e;d++){g=this[d];if(g.nodeType===1&&g.className)if(a){h=(" "+g.className+" ").replace(n," ");for(i=0,j=c.length;i<j;i++)h=h.replace(" "+c[i]+" "," ");g.className=f.trim(h)}else g.className=""}}return this},toggleClass:function(a,b){var c=typeof a,d=typeof b=="boolean";if(f.isFunction(a))return this.each(function(c){f(this).toggleClass(a.call(this,c,this.className,b),b)});return this.each(function(){if(c==="string"){var e,g=0,h=f(this),i=b,j=a.split(o);while(e=j[g++])i=d?i:!h.hasClass(e),h[i?"addClass":"removeClass"](e)}else if(c==="undefined"||c==="boolean")this.className&&f._data(this,"__className__",this.className),this.className=this.className||a===!1?"":f._data(this,"__className__")||""})},hasClass:function(a){var b=" "+a+" ";for(var c=0,d=this.length;c<d;c++)if(this[c].nodeType===1&&(" "+this[c].className+" ").replace(n," ").indexOf(b)>-1)return!0;return!1},val:function(a){var c,d,e=this[0];if(!arguments.length){if(e){c=f.valHooks[e.nodeName.toLowerCase()]||f.valHooks[e.type];if(c&&"get"in c&&(d=c.get(e,"value"))!==b)return d;d=e.value;return typeof d=="string"?d.replace(p,""):d==null?"":d}return b}var g=f.isFunction(a);return this.each(function(d){var e=f(this),h;if(this.nodeType===1){g?h=a.call(this,d,e.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||!("set"in c)||c.set(this,h,"value")===b)this.value=h}})}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c=a.selectedIndex,d=[],e=a.options,g=a.type==="select-one";if(c<0)return null;for(var h=g?c:0,i=g?c+1:e.length;h<i;h++){var j=e[h];if(j.selected&&(f.support.optDisabled?!j.disabled:j.getAttribute("disabled")===null)&&(!j.parentNode.disabled||!f.nodeName(j.parentNode,"optgroup"))){b=f(j).val();if(g)return b;d.push(b)}}if(g&&!d.length&&e.length)return f(e[c]).val();return d},set:function(a,b){var c=f.makeArray(b);f(a).find("option").each(function(){this.selected=f.inArray(f(this).val(),c)>=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attrFix:{tabindex:"tabIndex"},attr:function(a,c,d,e){var g=a.nodeType;if(!a||g===3||g===8||g===2)return b;if(e&&c in f.attrFn)return f(a)[c](d);if(!("getAttribute"in a))return f.prop(a,c,d);var h,i,j=g!==1||!f.isXMLDoc(a);j&&(c=f.attrFix[c]||c,i=f.attrHooks[c],i||(t.test(c)?i=v:u&&(i=u)));if(d!==b){if(d===null){f.removeAttr(a,c);return b}if(i&&"set"in i&&j&&(h=i.set(a,d,c))!==b)return h;a.setAttribute(c,""+d);return d}if(i&&"get"in i&&j&&(h=i.get(a,c))!==null)return h;h=a.getAttribute(c);return h===null?b:h},removeAttr:function(a,b){var c;a.nodeType===1&&(b=f.attrFix[b]||b,f.attr(a,b,""),a.removeAttribute(b),t.test(b)&&(c=f.propFix[b]||b)in a&&(a[c]=!1))},attrHooks:{type:{set:function(a,b){if(q.test(a.nodeName)&&a.parentNode)f.error("type property can't be changed");else if(!f.support.radioValue&&b==="radio"&&f.nodeName(a,"input")){var c=a.value;a.setAttribute("type",b),c&&(a.value=c);return b}}},value:{get:function(a,b){if(u&&f.nodeName(a,"button"))return u.get(a,b);return b in a?a.value:null},set:function(a,b,c){if(u&&f.nodeName(a,"button"))return u.set(a,b,c);a.value=b}}},propFix:{tabindex:"tabIndex",readonly:"readOnly","for":"htmlFor","class":"className",maxlength:"maxLength",cellspacing:"cellSpacing",cellpadding:"cellPadding",rowspan:"rowSpan",colspan:"colSpan",usemap:"useMap",frameborder:"frameBorder",contenteditable:"contentEditable"},prop:function(a,c,d){var e=a.nodeType;if(!a||e===3||e===8||e===2)return b;var g,h,i=e!==1||!f.isXMLDoc(a);i&&(c=f.propFix[c]||c,h=f.propHooks[c]);return d!==b?h&&"set"in h&&(g=h.set(a,d,c))!==b?g:a[c]=d:h&&"get"in h&&(g=h.get(a,c))!==null?g:a[c]},propHooks:{tabIndex:{get:function(a){var c=a.getAttributeNode("tabindex");return c&&c.specified?parseInt(c.value,10):r.test(a.nodeName)||s.test(a.nodeName)&&a.href?0:b}}}}),f.attrHooks.tabIndex=f.propHooks.tabIndex,v={get:function(a,c){var d;return f.prop(a,c)===!0||(d=a.getAttributeNode(c))&&d.nodeValue!==!1?c.toLowerCase():b},set:function(a,b,c){var d;b===!1?f.removeAttr(a,c):(d=f.propFix[c]||c,d in a&&(a[d]=!0),a.setAttribute(c,c.toLowerCase()));return c}},f.support.getSetAttribute||(u=f.valHooks.button={get:function(a,c){var d;d=a.getAttributeNode(c);return d&&d.nodeValue!==""?d.nodeValue:b},set:function(a,b,d){var e=a.getAttributeNode(d);e||(e=c.createAttribute(d),a.setAttributeNode(e));return e.nodeValue=b+""}},f.each(["width","height"],function(a,b){f.attrHooks[b]=f.extend(f.attrHooks[b],{set:function(a,c){if(c===""){a.setAttribute(b,"auto");return c}}})})),f.support.hrefNormalized||f.each(["href","src","width","height"],function(a,c){f.attrHooks[c]=f.extend(f.attrHooks[c],{get:function(a){var d=a.getAttribute(c,2);return d===null?b:d}})}),f.support.style||(f.attrHooks.style={get:function(a){return a.style.cssText.toLowerCase()||b},set:function(a,b){return a.style.cssText=""+b}}),f.support.optSelected||(f.propHooks.selected=f.extend(f.propHooks.selected,{get:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex);return null}})),f.support.checkOn||f.each(["radio","checkbox"],function(){f.valHooks[this]={get:function(a){return a.getAttribute("value")===null?"on":a.value}}}),f.each(["radio","checkbox"],function(){f.valHooks[this]=f.extend(f.valHooks[this],{set:function(a,b){if(f.isArray(b))return a.checked=f.inArray(f(a).val(),b)>=0}})});var w=/\.(.*)$/,x=/^(?:textarea|input|select)$/i,y=/\./g,z=/ /g,A=/[^\w\s.|`]/g,B=function(a){return a.replace(A,"\\$&")};f.event={add:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){if(d===!1)d=C;else if(!d)return;var g,h;d.handler&&(g=d,d=g.handler),d.guid||(d.guid=f.guid++);var i=f._data(a);if(!i)return;var j=i.events,k=i.handle;j||(i.events=j={}),k||(i.handle=k=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.handle.apply(k.elem,arguments):b}),k.elem=a,c=c.split(" ");var l,m=0,n;while(l=c[m++]){h=g?f.extend({},g):{handler:d,data:e},l.indexOf(".")>-1?(n=l.split("."),l=n.shift(),h.namespace=n.slice(0).sort().join(".")):(n=[],h.namespace=""),h.type=l,h.guid||(h.guid=d.guid);var o=j[l],p=f.event.special[l]||{};if(!o){o=j[l]=[];if(!p.setup||p.setup.call(a,e,n,k)===!1)a.addEventListener?a.addEventListener(l,k,!1):a.attachEvent&&a.attachEvent("on"+l,k)}p.add&&(p.add.call(a,h),h.handler.guid||(h.handler.guid=d.guid)),o.push(h),f.event.global[l]=!0}a=null}},global:{},remove:function(a,c,d,e){if(a.nodeType!==3&&a.nodeType!==8){d===!1&&(d=C);var g,h,i,j,k=0,l,m,n,o,p,q,r,s=f.hasData(a)&&f._data(a),t=s&&s.events;if(!s||!t)return;c&&c.type&&(d=c.handler,c=c.type);if(!c||typeof c=="string"&&c.charAt(0)==="."){c=c||"";for(h in t)f.event.remove(a,h+c);return}c=c.split(" ");while(h=c[k++]){r=h,q=null,l=h.indexOf(".")<0,m=[],l||(m=h.split("."),h=m.shift(),n=new RegExp("(^|\\.)"+f.map(m.slice(0).sort(),B).join("\\.(?:.*\\.)?")+"(\\.|$)")),p=t[h];if(!p)continue;if(!d){for(j=0;j<p.length;j++){q=p[j];if(l||n.test(q.namespace))f.event.remove(a,r,q.handler,j),p.splice(j--,1)}continue}o=f.event.special[h]||{};for(j=e||0;j<p.length;j++){q=p[j];if(d.guid===q.guid){if(l||n.test(q.namespace))e==null&&p.splice(j--,1),o.remove&&o.remove.call(a,q);if(e!=null)break}}if(p.length===0||e!=null&&p.length===1)(!o.teardown||o.teardown.call(a,m)===!1)&&f.removeEvent(a,h,s.handle),g=null,delete
  3. t[h]}if(f.isEmptyObject(t)){var u=s.handle;u&&(u.elem=null),delete s.events,delete s.handle,f.isEmptyObject(s)&&f.removeData(a,b,!0)}}},customEvent:{getData:!0,setData:!0,changeData:!0},trigger:function(c,d,e,g){var h=c.type||c,i=[],j;h.indexOf("!")>=0&&(h=h.slice(0,-1),j=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.shift(),i.sort());if(!!e&&!f.event.customEvent[h]||!!f.event.global[h]){c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.exclusive=j,c.namespace=i.join("."),c.namespace_re=new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)");if(g||!e)c.preventDefault(),c.stopPropagation();if(!e){f.each(f.cache,function(){var a=f.expando,b=this[a];b&&b.events&&b.events[h]&&f.event.trigger(c,d,b.handle.elem)});return}if(e.nodeType===3||e.nodeType===8)return;c.result=b,c.target=e,d=d!=null?f.makeArray(d):[],d.unshift(c);var k=e,l=h.indexOf(":")<0?"on"+h:"";do{var m=f._data(k,"handle");c.currentTarget=k,m&&m.apply(k,d),l&&f.acceptData(k)&&k[l]&&k[l].apply(k,d)===!1&&(c.result=!1,c.preventDefault()),k=k.parentNode||k.ownerDocument||k===c.target.ownerDocument&&a}while(k&&!c.isPropagationStopped());if(!c.isDefaultPrevented()){var n,o=f.event.special[h]||{};if((!o._default||o._default.call(e.ownerDocument,c)===!1)&&(h!=="click"||!f.nodeName(e,"a"))&&f.acceptData(e)){try{l&&e[h]&&(n=e[l],n&&(e[l]=null),f.event.triggered=h,e[h]())}catch(p){}n&&(e[l]=n),f.event.triggered=b}}return c.result}},handle:function(c){c=f.event.fix(c||a.event);var d=((f._data(this,"events")||{})[c.type]||[]).slice(0),e=!c.exclusive&&!c.namespace,g=Array.prototype.slice.call(arguments,0);g[0]=c,c.currentTarget=this;for(var h=0,i=d.length;h<i;h++){var j=d[h];if(e||c.namespace_re.test(j.namespace)){c.handler=j.handler,c.data=j.data,c.handleObj=j;var k=j.handler.apply(this,g);k!==b&&(c.result=k,k===!1&&(c.preventDefault(),c.stopPropagation()));if(c.isImmediatePropagationStopped())break}}return c.result},props:"altKey attrChange attrName bubbles button cancelable charCode clientX clientY ctrlKey currentTarget data detail eventPhase fromElement handler keyCode layerX layerY metaKey newValue offsetX offsetY pageX pageY prevValue relatedNode relatedTarget screenX screenY shiftKey srcElement target toElement view wheelDelta which".split(" "),fix:function(a){if(a[f.expando])return a;var d=a;a=f.Event(d);for(var e=this.props.length,g;e;)g=this.props[--e],a[g]=d[g];a.target||(a.target=a.srcElement||c),a.target.nodeType===3&&(a.target=a.target.parentNode),!a.relatedTarget&&a.fromElement&&(a.relatedTarget=a.fromElement===a.target?a.toElement:a.fromElement);if(a.pageX==null&&a.clientX!=null){var h=a.target.ownerDocument||c,i=h.documentElement,j=h.body;a.pageX=a.clientX+(i&&i.scrollLeft||j&&j.scrollLeft||0)-(i&&i.clientLeft||j&&j.clientLeft||0),a.pageY=a.clientY+(i&&i.scrollTop||j&&j.scrollTop||0)-(i&&i.clientTop||j&&j.clientTop||0)}a.which==null&&(a.charCode!=null||a.keyCode!=null)&&(a.which=a.charCode!=null?a.charCode:a.keyCode),!a.metaKey&&a.ctrlKey&&(a.metaKey=a.ctrlKey),!a.which&&a.button!==b&&(a.which=a.button&1?1:a.button&2?3:a.button&4?2:0);return a},guid:1e8,proxy:f.proxy,special:{ready:{setup:f.bindReady,teardown:f.noop},live:{add:function(a){f.event.add(this,M(a.origType,a.selector),f.extend({},a,{handler:L,guid:a.handler.guid}))},remove:function(a){f.event.remove(this,M(a.origType,a.selector),a)}},beforeunload:{setup:function(a,b,c){f.isWindow(this)&&(this.onbeforeunload=c)},teardown:function(a,b){this.onbeforeunload===b&&(this.onbeforeunload=null)}}}},f.removeEvent=c.removeEventListener?function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c,!1)}:function(a,b,c){a.detachEvent&&a.detachEvent("on"+b,c)},f.Event=function(a,b){if(!this.preventDefault)return new f.Event(a,b);a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||a.returnValue===!1||a.getPreventDefault&&a.getPreventDefault()?D:C):this.type=a,b&&f.extend(this,b),this.timeStamp=f.now(),this[f.expando]=!0},f.Event.prototype={preventDefault:function(){this.isDefaultPrevented=D;var a=this.originalEvent;!a||(a.preventDefault?a.preventDefault():a.returnValue=!1)},stopPropagation:function(){this.isPropagationStopped=D;var a=this.originalEvent;!a||(a.stopPropagation&&a.stopPropagation(),a.cancelBubble=!0)},stopImmediatePropagation:function(){this.isImmediatePropagationStopped=D,this.stopPropagation()},isDefaultPrevented:C,isPropagationStopped:C,isImmediatePropagationStopped:C};var E=function(a){var b=a.relatedTarget,c=!1,d=a.type;a.type=a.data,b!==this&&(b&&(c=f.contains(this,b)),c||(f.event.handle.apply(this,arguments),a.type=d))},F=function(a){a.type=a.data,f.event.handle.apply(this,arguments)};f.each({mouseenter:"mouseover",mouseleave:"mouseout"},function(a,b){f.event.special[a]={setup:function(c){f.event.add(this,b,c&&c.selector?F:E,a)},teardown:function(a){f.event.remove(this,b,a&&a.selector?F:E)}}}),f.support.submitBubbles||(f.event.special.submit={setup:function(a,b){if(!f.nodeName(this,"form"))f.event.add(this,"click.specialSubmit",function(a){var b=a.target,c=f.nodeName(b,"input")||f.nodeName(b,"button")?b.type:"";(c==="submit"||c==="image")&&f(b).closest("form").length&&J("submit",this,arguments)}),f.event.add(this,"keypress.specialSubmit",function(a){var b=a.target,c=f.nodeName(b,"input")||f.nodeName(b,"button")?b.type:"";(c==="text"||c==="password")&&f(b).closest("form").length&&a.keyCode===13&&J("submit",this,arguments)});else return!1},teardown:function(a){f.event.remove(this,".specialSubmit")}});if(!f.support.changeBubbles){var G,H=function(a){var b=f.nodeName(a,"input")?a.type:"",c=a.value;b==="radio"||b==="checkbox"?c=a.checked:b==="select-multiple"?c=a.selectedIndex>-1?f.map(a.options,function(a){return a.selected}).join("-"):"":f.nodeName(a,"select")&&(c=a.selectedIndex);return c},I=function(c){var d=c.target,e,g;if(!!x.test(d.nodeName)&&!d.readOnly){e=f._data(d,"_change_data"),g=H(d),(c.type!=="focusout"||d.type!=="radio")&&f._data(d,"_change_data",g);if(e===b||g===e)return;if(e!=null||g)c.type="change",c.liveFired=b,f.event.trigger(c,arguments[1],d)}};f.event.special.change={filters:{focusout:I,beforedeactivate:I,click:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(c==="radio"||c==="checkbox"||f.nodeName(b,"select"))&&I.call(this,a)},keydown:function(a){var b=a.target,c=f.nodeName(b,"input")?b.type:"";(a.keyCode===13&&!f.nodeName(b,"textarea")||a.keyCode===32&&(c==="checkbox"||c==="radio")||c==="select-multiple")&&I.call(this,a)},beforeactivate:function(a){var b=a.target;f._data(b,"_change_data",H(b))}},setup:function(a,b){if(this.type==="file")return!1;for(var c in G)f.event.add(this,c+".specialChange",G[c]);return x.test(this.nodeName)},teardown:function(a){f.event.remove(this,".specialChange");return x.test(this.nodeName)}},G=f.event.special.change.filters,G.focus=G.beforeactivate}f.support.focusinBubbles||f.each({focus:"focusin",blur:"focusout"},function(a,b){function e(a){var c=f.event.fix(a);c.type=b,c.originalEvent={},f.event.trigger(c,null,c.target),c.isDefaultPrevented()&&a.preventDefault()}var d=0;f.event.special[b]={setup:function(){d++===0&&c.addEventListener(a,e,!0)},teardown:function(){--d===0&&c.removeEventListener(a,e,!0)}}}),f.each(["bind","one"],function(a,c){f.fn[c]=function(a,d,e){var g;if(typeof a=="object"){for(var h in a)this[c](h,d,a[h],e);return this}if(arguments.length===2||d===!1)e=d,d=b;c==="one"?(g=function(a){f(this).unbind(a,g);return e.apply(this,arguments)},g.guid=e.guid||f.guid++):g=e;if(a==="unload"&&c!=="one")this.one(a,d,e);else for(var i=0,j=this.length;i<j;i++)f.event.add(this[i],a,g,d);return this}}),f.fn.extend({unbind:function(a,b){if(typeof a=="object"&&!a.preventDefault)for(var c in a)this.unbind(c,a[c]);else for(var d=0,e=this.length;d<e;d++)f.event.remove(this[d],a,b);return this},delegate:function(a,b,c,d){return this.live(b,c,d,a)},undelegate:function(a,b,c){return arguments.length===0?this.unbind("live"):this.die(b,null,c,a)},trigger:function(a,b){return this.each(function(){f.event.trigger(a,b,this)})},triggerHandler:function(a,b){if(this[0])return f.event.trigger(a,b,this[0],!0)},toggle:function(a){var b=arguments,c=a.guid||f.guid++,d=0,e=function(c){var e=(f.data(this,"lastToggle"+a.guid)||0)%d;f.data(this,"lastToggle"+a.guid,e+1),c.preventDefault();return b[e].apply(this,arguments)||!1};e.guid=c;while(d<b.length)b[d++].guid=c;return this.click(e)},hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}});var K={focus:"focusin",blur:"focusout",mouseenter:"mouseover",mouseleave:"mouseout"};f.each(["live","die"],function(a,c){f.fn[c]=function(a,d,e,g){var h,i=0,j,k,l,m=g||this.selector,n=g?this:f(this.context);if(typeof a=="object"&&!a.preventDefault){for(var o in a)n[c](o,d,a[o],m);return this}if(c==="die"&&!a&&g&&g.charAt(0)==="."){n.unbind(g);return this}if(d===!1||f.isFunction(d))e=d||C,d=b;a=(a||"").split(" ");while((h=a[i++])!=null){j=w.exec(h),k="",j&&(k=j[0],h=h.replace(w,""));if(h==="hover"){a.push("mouseenter"+k,"mouseleave"+k);continue}l=h,K[h]?(a.push(K[h]+k),h=h+k):h=(K[h]||h)+k;if(c==="live")for(var p=0,q=n.length;p<q;p++)f.event.add(n[p],"live."+M(h,m),{data:d,selector:m,handler:e,origType:h,origHandler:e,preType:l});else n.unbind("live."+M(h,m),e)}return this}}),f.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error".split(" "),function(a,b){f.fn[b]=function(a,c){c==null&&(c=a,a=null);return arguments.length>0?this.bind(b,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0)}),function(){function u(a,b,c,d,e,f){for(var g=0,h=d.length;g<h;g++){var i=d[g];if(i){var j=!1;i=i[a];while(i){if(i.sizcache===c){j=d[i.sizset];break}if(i.nodeType===1){f||(i.sizcache=c,i.sizset=g);if(typeof b!="string"){if(i===b){j=!0;break}}else if(k.filter(b,[i]).length>0){j=i;break}}i=i[a]}d[g]=j}}}function t(a,b,c,d,e,f){for(var g=0,h=d.length;g<h;g++){var i=d[g];if(i){var j=!1;i=i[a];while(i){if(i.sizcache===c){j=d[i.sizset];break}i.nodeType===1&&!f&&(i.sizcache=c,i.sizset=g);if(i.nodeName.toLowerCase()===b){j=i;break}i=i[a]}d[g]=j}}}var a=/((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^\[\]]*\]|['"][^'"]*['"]|[^\[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d=0,e=Object.prototype.toString,g=!1,h=!0,i=/\\/g,j=/\W/;[0,0].sort(function(){h=!1;return 0});var k=function(b,d,f,g){f=f||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return f;var i,j,n,o,q,r,s,t,u=!0,w=k.isXML(d),x=[],y=b;do{a.exec(""),i=a.exec(y);if(i){y=i[3],x.push(i[1]);if(i[2]){o=i[3];break}}}while(i);if(x.length>1&&m.exec(b))if(x.length===2&&l.relative[x[0]])j=v(x[0]+x[1],d);else{j=l.relative[x[0]]?[d]:k(x.shift(),d);while(x.length)b=x.shift(),l.relative[b]&&(b+=x.shift()),j=v(b,j)}else{!g&&x.length>1&&d.nodeType===9&&!w&&l.match.ID.test(x[0])&&!l.match.ID.test(x[x.length-1])&&(q=k.find(x.shift(),d,w),d=q.expr?k.filter(q.expr,q.set)[0]:q.set[0]);if(d){q=g?{expr:x.pop(),set:p(g)}:k.find(x.pop(),x.length===1&&(x[0]==="~"||x[0]==="+")&&d.parentNode?d.parentNode:d,w),j=q.expr?k.filter(q.expr,q.set):q.set,x.length>0?n=p(j):u=!1;while(x.length)r=x.pop(),s=r,l.relative[r]?s=x.pop():r="",s==null&&(s=d),l.relative[r](n,s,w)}else n=x=[]}n||(n=j),n||k.error(r||b);if(e.call(n)==="[object Array]")if(!u)f.push.apply(f,n);else if(d&&d.nodeType===1)for(t=0;n[t]!=null;t++)n[t]&&(n[t]===!0||n[t].nodeType===1&&k.contains(d,n[t]))&&f.push(j[t]);else for(t=0;n[t]!=null;t++)n[t]&&n[t].nodeType===1&&f.push(j[t]);else p(n,f);o&&(k(o,h,f,g),k.uniqueSort(f));return f};k.uniqueSort=function(a){if(r){g=h,a.sort(r);if(g)for(var b=1;b<a.length;b++)a[b]===a[b-1]&&a.splice(b--,1)}return a},k.matches=function(a,b){return k(a,null,null,b)},k.matchesSelector=function(a,b){return k(b,null,null,[a]).length>0},k.find=function(a,b,c){var d;if(!a)return[];for(var e=0,f=l.order.length;e<f;e++){var g,h=l.order[e];if(g=l.leftMatch[h].exec(a)){var j=g[1];g.splice(1,1);if(j.substr(j.length-1)!=="\"){g[1]=(g[1]||"").replace(i,""),d=l.find[h](g,b,c);if(d!=null){a=a.replace(l.match[h],"");break}}}}d||(d=typeof b.getElementsByTagName!="undefined"?b.getElementsByTagName("*"):[]);return{set:d,expr:a}},k.filter=function(a,c,d,e){var f,g,h=a,i=[],j=c,m=c&&c[0]&&k.isXML(c[0]);while(a&&c.length){for(var n in l.filter)if((f=l.leftMatch[n].exec(a))!=null&&f[2]){var o,p,q=l.filter[n],r=f[1];g=!1,f.splice(1,1);if(r.substr(r.length-1)==="\")continue;j===i&&(i=[]);if(l.preFilter[n]){f=l.preFilter[n](f,j,d,i,e,m);if(!f)g=o=!0;else if(f===!0)continue}if(f)for(var s=0;(p=j[s])!=null;s++)if(p){o=q(p,f,s,j);var t=e^!!o;d&&o!=null?t?g=!0:j[s]=!1:t&&(i.push(p),g=!0)}if(o!==b){d||(j=i),a=a.replace(l.match[n],"");if(!g)return[];break}}if(a===h)if(g==null)k.error(a);else break;h=a}return j},k.error=function(a){throw"Syntax error, unrecognized expression: "+a};var l=k.selectors={order:["ID","NAME","TAG"],match:{ID:/#((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,CLASS:/\.((?:[\w\u00c0-\uFFFF\-]|\\.)+)/,NAME:/\[name=['"]*((?:[\w\u00c0-\uFFFF\-]|\\.)+)['"]*\]/,ATTR:/\[\s*((?:[\w\u00c0-\uFFFF\-]|\\.)+)\s*(?:(\S?=)\s*(?:(['"])(.*?)\3|(#?(?:[\w\u00c0-\uFFFF\-]|\\.)*)|)|)\s*\]/,TAG:/^((?:[\w\u00c0-\uFFFF\*\-]|\\.)+)/,CHILD:/:(only|nth|last|first)-child(?:\(\s*(even|odd|(?:[+\-]?\d+|(?:[+\-]?\d*)?n\s*(?:[+\-]\s*\d+)?))\s*\))?/,POS:/:(nth|eq|gt|lt|first|last|even|odd)(?:\((\d*)\))?(?=[^\-]|$)/,PSEUDO:/:((?:[\w\u00c0-\uFFFF\-]|\\.)+)(?:\((['"]?)((?:\([^\)]+\)|[^\(\)]*)+)\2\))?/},leftMatch:{},attrMap:{"class":"className","for":"htmlFor"},attrHandle:{href:function(a){return a.getAttribute("href")},type:function(a){return a.getAttribute("type")}},relative:{"+":function(a,b){var c=typeof b=="string",d=c&&!j.test(b),e=c&&!d;d&&(b=b.toLowerCase());for(var f=0,g=a.length,h;f<g;f++)if(h=a[f]){while((h=h.previousSibling)&&h.nodeType!==1);a[f]=e||h&&h.nodeName.toLowerCase()===b?h||!1:h===b}e&&k.filter(b,a,!0)},">":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!j.test(b)){b=b.toLowerCase();for(;e<f;e++){c=a[e];if(c){var g=c.parentNode;a[e]=g.nodeName.toLowerCase()===b?g:!1}}}else{for(;e<f;e++)c=a[e],c&&(a[e]=d?c.parentNode:c.parentNode===b);d&&k.filter(b,a,!0)}},"":function(a,b,c){var e,f=d++,g=u;typeof b=="string"&&!j.test(b)&&(b=b.toLowerCase(),e=b,g=t),g("parentNode",b,f,a,e,c)},"~":function(a,b,c){var e,f=d++,g=u;typeof b=="string"&&!j.test(b)&&(b=b.toLowerCase(),e=b,g=t),g("previousSibling",b,f,a,e,c)}},find:{ID:function(a,b,c){if(typeof b.getElementById!="undefined"&&!c){var d=b.getElementById(a[1]);return d&&d.parentNode?[d]:[]}},NAME:function(a,b){if(typeof b.getElementsByName!="undefined"){var c=[],d=b.getElementsByName(a[1]);for(var e=0,f=d.length;e<f;e++)d[e].getAttribute("name")===a[1]&&c.push(d[e]);return c.length===0?null:c}},TAG:function(a,b){if(typeof b.getElementsByTagName!="undefined")return b.getElementsByTagName(a[1])}},preFilter:{CLASS:function(a,b,c,d,e,f){a=" "+a[1].replace(i,"")+" ";if(f)return a;for(var g=0,h;(h=b[g])!=null;g++)h&&(e^(h.className&&(" "+h.className+" ").replace(/[\t\n\r]/g," ").indexOf(a)>=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(i,"")},TAG:function(a,b){return a[1].replace(i,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||k.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&k.error(a[0]);a[0]=d++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(i,"");!f&&l.attrMap[g]&&(a[1]=l.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(i,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=k(b[3],null,null,c);else{var g=k.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(l.match.POS.test(b[0])||l.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!k(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return b<c[3]-0},gt:function(a,b,c){return b>c[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=l.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||k.getText([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h<i;h++)if(g[h]===a)return!1;return!0}k.error(e)},CHILD:function(a,b){var c=b[1],d=a;switch(c){case"only":case"first":while(d=d.previousSibling)if(d.nodeType===1)return!1;if(c==="first")return!0;d=a;case"last":while(d=d.nextSibling)if(d.nodeType===1)return!1;return!0;case"nth":var e=b[2],f=b[3];if(e===1&&f===0)return!0;var g=b[0],h=a.parentNode;if(h&&(h.sizcache!==g||!a.nodeIndex)){var i=0;for(d=h.firstChild;d;d=d.nextSibling)d.nodeType===1&&(d.nodeIndex=++i);h.sizcache=g}var j=a.nodeIndex-f;return e===0?j===0:j%e===0&&j/e>=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=l.attrHandle[c]?l.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=l.setFilters[e];if(f)return f(a,c,b,d)}}},m=l.match.POS,n=function(a,b){return"\"+(b-0+1)};for(var o in l.match)l.match[o]=new RegExp(l.match[o].source+/(?![^\[]*\])(?![^\(]*\))/.source),l.leftMatch[o]=new RegExp(/(^(?:.|\r|\n)*?)/.source+l.match[o].source.replace(/\\(\d+)/g,n));var p=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(q){p=function(a,b){var c=0,d=b||[];if(e.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var f=a.length;c<f;c++)d.push(a[c]);else for(;a[c];c++)d.push(a[c]);return d}}var r,s;c.documentElement.compareDocumentPosition?r=function(a,b){if(a===b){g=!0;return 0}if(!a.compareDocumentPosition||!b.compareDocumentPosition)return a.compareDocumentPosition?-1:1;return a.compareDocumentPosition(b)&4?-1:1}:(r=function(a,b){if(a===b){g=!0;return 0}if(a.sourceIndex&&b.sourceIndex)return a.sourceIndex-b.sourceIndex;var c,d,e=[],f=[],h=a.parentNode,i=b.parentNode,j=h;if(h===i)return s(a,b);if(!h)return-1;if(!i)return 1;while(j)e.unshift(j),j=j.parentNode;j=i;while(j)f.unshift(j),j=j.parentNode;c=e.length,d=f.length;for(var k=0;k<c&&k<d;k++)if(e[k]!==f[k])return s(e[k],f[k]);return k===c?s(a,f[k],-1):s(e[k],b,1)},s=function(a,b,c){if(a===b)return c;var d=a.nextSibling;while(d){if(d===b)return-1;d=d.nextSibling}return 1}),k.getText=function(a){var b="",c;for(var d=0;a[d];d++)c=a[d],c.nodeType===3||c.nodeType===4?b+=c.nodeValue:c.nodeType!==8&&(b+=k.getText(c.childNodes));return b},function(){var a=c.createElement("div"),d="script"+(new Date).getTime(),e=c.documentElement;a.innerHTML="<a name='"+d+"'/>",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(l.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},l.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(l.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="<a target="_blank" href='https://www.cnblogs.com/#'></a>",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(l.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=k,b=c.createElement("div"),d="__sizzle__";b.innerHTML="<p class='TEST'></p>";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){k=function(b,e,f,g){e=e||c;if(!g&&!k.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return p(e.getElementsByTagName(b),f);if(h[2]&&l.find.CLASS&&e.getElementsByClassName)return p(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return p([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return p([],f);if(i.id===h[3])return p([i],f)}try{return p(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var m=e,n=e.getAttribute("id"),o=n||d,q=e.parentNode,r=/^\s*[+~]/.test(b);n?o=o.replace(/'/g,"\\$&"):e.setAttribute("id",o),r&&q&&(e=e.parentNode);try{if(!r||q)return p(e.querySelectorAll("[id='"+o+"'] "+b),f)}catch(s){}finally{n||m.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)k[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}k.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!k.isXML(a))try{if(e||!l.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return k(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;l.order.splice(1,0,"CLASS"),l.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?k.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?k.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:k.contains=function(){return!1},k.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var v=function(a,b){var c,d=[],e="",f=b.nodeType?[b]:b;while(c=l.match.PSEUDO.exec(a))e+=c[0],a=a.replace(l.match.PSEUDO,"");a=l.relative[a]?a+"*":a;for(var g=0,h=f.length;g<h;g++)k(a,f[g],d);return k.filter(e,d)};f.find=k,f.expr=k.selectors,f.expr[":"]=f.expr.filters,f.unique=k.uniqueSort,f.text=k.getText,f.isXMLDoc=k.isXML,f.contains=k.contains}();var N=/Until$/,O=/^(?:parents|prevUntil|prevAll)/,P=/,/,Q=/^.[^:#\[\.,]*$/,R=Array.prototype.slice,S=f.expr.match.POS,T={children:!0,contents:!0,next:!0,prev:!0};f.fn.extend({find:function(a){var b=this,c,d;if(typeof a!="string")return f(a).filter(function(){for(c=0,d=b.length;c<d;c++)if(f.contains(b[c],this))return!0});var e=this.pushStack("","find",a),g,h,i;for(c=0,d=this.length;c<d;c++){g=e.length,f.find(a,this[c],e);if(c>0)for(h=g;h<e.length;h++)for(i=0;i<g;i++)if(e[i]===e[h]){e.splice(h--,1);break}}return e},has:function(a){var b=f(a);return this.filter(function(){for(var a=0,c=b.length;a<c;a++)if(f.contains(this,b[a]))return!0})},not:function(a){return this.pushStack(V(this,a,!1),"not",a)},filter:function(a){return this.pushStack(V(this,a,!0),"filter",a)},is:function(a){return!!a&&(typeof a=="string"?f.filter(a,this).length>0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h,i,j={},k=1;if(g&&a.length){for(d=0,e=a.length;d<e;d++)i=a[d],j[i]||(j[i]=S.test(i)?f(i,b||this.context):i);while(g&&g.ownerDocument&&g!==b){for(i in j)h=j[i],(h.jquery?h.index(g)>-1:f(g).is(h))&&c.push({selector:i,elem:g,level:k});g=g.parentNode,k++}}return c}var l=S.test(a)||typeof a!="string"?f(a,b||this.context):0;for(d=0,e=this.length;d<e;d++){g=this[d];while(g){if(l?l.index(g)>-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a)return this[0]&&this[0].parentNode?this.prevAll().length:-1;if(typeof a=="string")return f.inArray(this[0],f(a));return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(U(c[0])||U(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c),g=R.call(arguments);N.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!T[a]?f.unique(e):e,(this.length>1||P.test(d))&&O.test(a)&&(e=e.reverse());return this.pushStack(e,a,g.join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var W=/ jQuery\d+="(?:\d+|null)"/g,X=/^\s+/,Y=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,Z=/<([\w:]+)/,$=/<tbody/i,_=/<|&#?\w+;/,ba=/<(?:script|object|embed|option|style)/i,bb=/checked\s*(?:[^=]|=\s*.checked.)/i,bc=/\/(java|ecma)script/i,bd=/^\s*<!(?:\[CDATA\[|\-\-)/,be={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],area:[1,"<map>","</map>"],_default:[0,"",""]};be.optgroup=be.option,be.tbody=be.tfoot=be.colgroup=be.caption=be.thead,be.th=be.td,f.support.htmlSerialize||(be._default=[1,"div",""]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){f(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f(arguments[0]).toArray());return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(W,""):null;if(typeof a=="string"&&!ba.test(a)&&(f.support.leadingWhitespace||!X.test(a))&&!be[(Z.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Y,"<$1></$2>");try{for(var c=0,d=this.length;c<d;c++)this[c].nodeType===1&&(f.cleanData(this[c].getElementsByTagName("*")),this[c].innerHTML=a)}catch(e){this.empty().append(a)}}else f.isFunction(a)?this.each(function(b){var c=f(this);c.html(a.call(this,b,c.html()))}):this.empty().append(a);return this},replaceWith:function(a){if(this[0]&&this[0].parentNode){if(f.isFunction(a))return this.each(function(b){var c=f(this),d=c.html();c.replaceWith(a.call(this,b,d))});typeof a!="string"&&(a=f(a).detach());return this.each(function(){var b=this.nextSibling,c=this.parentNode;f(this).remove(),b?f(b).before(a):f(c).append(a)})}return this.length?this.pushStack(f(f.isFunction(a)?a():a),"replaceWith",a):this},detach:function(a){return this.remove(a,!0)},domManip:function(a,c,d){var e,g,h,i,j=a[0],k=[];if(!f.support.checkClone&&arguments.length===3&&typeof j=="string"&&bb.test(j))return this.each(function(){f(this).domManip(a,c,d,!0)});if(f.isFunction(j))return this.each(function(e){var g=f(this);a[0]=j.call(this,e,c?g.html():b),g.domManip(a,c,d)});if(this[0]){i=j&&j.parentNode,f.support.parentNode&&i&&i.nodeType===11&&i.childNodes.length===this.length?e={fragment:i}:e=f.buildFragment(a,this,k),h=e.fragment,h.childNodes.length===1?g=h=h.firstChild:g=h.firstChild;if(g){c=c&&f.nodeName(g,"tr");for(var l=0,m=this.length,n=m-1;l<m;l++)d.call(c?bf(this[l],g):this[l],e.cacheable||m>1&&l<n?f.clone(h,!0,!0):h)}k.length&&f.each(k,bl)}return this}}),f.buildFragment=function(a,b,d){var e,g,h,i;b&&b[0]&&(i=b[0].ownerDocument||b[0]),i.createDocumentFragment||(i=c),a.length===1&&typeof a[0]=="string"&&a[0].length<512&&i===c&&a[0].charAt(0)==="<"&&!ba.test(a[0])&&(f.support.checkClone||!bb.test(a[0]))&&(g=!0,h=f.fragments[a[0]],h&&h!==1&&(e=h)),e||(e=i.createDocumentFragment(),f.clean
  4. (a,i,e,d)),g&&(f.fragments[a[0]]=h?e:1);return{fragment:e,cacheable:g}},f.fragments={},f.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){f.fn[a]=function(c){var d=[],e=f(c),g=this.length===1&&this[0].parentNode;if(g&&g.nodeType===11&&g.childNodes.length===1&&e.length===1){e[b](this[0]);return this}for(var h=0,i=e.length;h<i;h++){var j=(h>0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d=a.cloneNode(!0),e,g,h;if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bh(a,d),e=bi(a),g=bi(d);for(h=0;e[h];++h)g[h]&&bh(e[h],g[h])}if(b){bg(a,d);if(c){e=bi(a),g=bi(d);for(h=0;e[h];++h)bg(e[h],g[h])}}e=g=null;return d},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!_.test(k))k=b.createTextNode(k);else{k=k.replace(Y,"<$1></$2>");var l=(Z.exec(k)||["",""])[1].toLowerCase(),m=be[l]||be._default,n=m[0],o=b.createElement("div");o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=$.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]==="<table>"&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&X.test(k)&&o.insertBefore(b.createTextNode(X.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i<r;i++)bk(k[i]);else bk(k);k.nodeType?h.push(k):h=f.merge(h,k)}if(d){g=function(a){return!a.type||bc.test(a.type)};for(j=0;h[j];j++)if(e&&f.nodeName(h[j],"script")&&(!h[j].type||h[j].type.toLowerCase()==="text/javascript"))e.push(h[j].parentNode?h[j].parentNode.removeChild(h[j]):h[j]);else{if(h[j].nodeType===1){var s=f.grep(h[j].getElementsByTagName("script"),g);h.splice.apply(h,[j+1,0].concat(s))}d.appendChild(h[j])}}return h},cleanData:function(a){var b,c,d=f.cache,e=f.expando,g=f.event.special,h=f.support.deleteExpando;for(var i=0,j;(j=a[i])!=null;i++){if(j.nodeName&&f.noData[j.nodeName.toLowerCase()])continue;c=j[f.expando];if(c){b=d[c]&&d[c][e];if(b&&b.events){for(var k in b.events)g[k]?f.event.remove(j,k):f.removeEvent(j,k,b.handle);b.handle&&(b.handle.elem=null)}h?delete j[f.expando]:j.removeAttribute&&j.removeAttribute(f.expando),delete d[c]}}}});var bm=/alpha\([^)]*\)/i,bn=/opacity=([^)]*)/,bo=/([A-Z]|^ms)/g,bp=/^-?\d+(?:px)?$/i,bq=/^-?\d/,br=/^([\-+])=([\-+.\de]+)/,bs={position:"absolute",visibility:"hidden",display:"block"},bt=["Left","Right"],bu=["Top","Bottom"],bv,bw,bx;f.fn.css=function(a,c){if(arguments.length===2&&c===b)return this;return f.access(this,a,c,!0,function(a,c,d){return d!==b?f.style(a,c,d):f.css(a,c)})},f.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=bv(a,"opacity","opacity");return c===""?"1":c}return a.style.opacity}}},cssNumber:{fillOpacity:!0,fontWeight:!0,lineHeight:!0,opacity:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":f.support.cssFloat?"cssFloat":"styleFloat"},style:function(a,c,d,e){if(!!a&&a.nodeType!==3&&a.nodeType!==8&&!!a.style){var g,h,i=f.camelCase(c),j=a.style,k=f.cssHooks[i];c=f.cssProps[i]||i;if(d===b){if(k&&"get"in k&&(g=k.get(a,!1,e))!==b)return g;return j[c]}h=typeof d,h==="string"&&(g=br.exec(d))&&(d=+(g[1]+1)*+g[2]+parseFloat(f.css(a,c)),h="number");if(d==null||h==="number"&&isNaN(d))return;h==="number"&&!f.cssNumber[i]&&(d+="px");if(!k||!("set"in k)||(d=k.set(a,d))!==b)try{j[c]=d}catch(l){}}},css:function(a,c,d){var e,g;c=f.camelCase(c),g=f.cssHooks[c],c=f.cssProps[c]||c,c==="cssFloat"&&(c="float");if(g&&"get"in g&&(e=g.get(a,!0,d))!==b)return e;if(bv)return bv(a,c)},swap:function(a,b,c){var d={};for(var e in b)d[e]=a.style[e],a.style[e]=b[e];c.call(a);for(e in b)a.style[e]=d[e]}}),f.curCSS=f.css,f.each(["height","width"],function(a,b){f.cssHooks[b]={get:function(a,c,d){var e;if(c){if(a.offsetWidth!==0)return by(a,b,d);f.swap(a,bs,function(){e=by(a,b,d)});return e}},set:function(a,b){if(!bp.test(b))return b;b=parseFloat(b);if(b>=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return bn.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=f.isNaN(b)?"":"alpha(opacity="+b*100+")",g=d&&d.filter||c.filter||"";c.zoom=1;if(b>=1&&f.trim(g.replace(bm,""))===""){c.removeAttribute("filter");if(d&&!d.filter)return}c.filter=bm.test(g)?g.replace(bm,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bv(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(bw=function(a,c){var d,e,g;c=c.replace(bo,"-$1").toLowerCase();if(!(e=a.ownerDocument.defaultView))return b;if(g=e.getComputedStyle(a,null))d=g.getPropertyValue(c),d===""&&!f.contains(a.ownerDocument.documentElement,a)&&(d=f.style(a,c));return d}),c.documentElement.currentStyle&&(bx=function(a,b){var c,d=a.currentStyle&&a.currentStyle[b],e=a.runtimeStyle&&a.runtimeStyle[b],f=a.style;!bp.test(d)&&bq.test(d)&&(c=f.left,e&&(a.runtimeStyle.left=a.currentStyle.left),f.left=b==="fontSize"?"1em":d||0,d=f.pixelLeft+"px",f.left=c,e&&(a.runtimeStyle.left=e));return d===""?"auto":d}),bv=bw||bx,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bz=/%20/g,bA=/\[\]$/,bB=/\r?\n/g,bC=/#.*$/,bD=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bE=/^(?:color|date|datetime|datetime-local|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bF=/^(?:about|app|app\-storage|.+\-extension|file|res|widget):$/,bG=/^(?:GET|HEAD)$/,bH=/^\/\//,bI=/\?/,bJ=/
  5.    
  6.    
  7. }
  8.    
  9.    
  10.    
  11.         
  12.             报表最新同步时间:
  13.         
  14.         
  15.             <table >
  16.                 <colgroup>
  17.                     <col>
  18.                     <col>
  19.                     <col>
  20.                     <col>
  21.                     <col>
  22.                     <col>
  23.                     <col>
  24.                     <col>
  25.                     <col>
  26.                     <col>
  27.                     <col>
  28.                     <col>
  29.                     <col>
  30.                 </colgroup>
  31.                 <thead>
  32.                     <tr>
  33.                         <th colspan="2" >项目信息</th>
  34.                         <th colspan="5" >销售情况</th>
  35.                         <th colspan="6" >款项情况</th>
  36.                     </tr>
  37.                     <tr>
  38.                         <th >项目名称</th>
  39.                         <th >服务商名称</th>
  40.                         <th >昨天订单总额</th>
  41.                         <th >今天订单总额</th>
  42.                         <th >去年订单总额</th>
  43.                         <th >今年订单总额</th>
  44.                         <th >订单总额</th>
  45.                         <th >客户应回款总额</th>
  46.                         <th >客户已回款总额</th>
  47.                         <th >客户未回款总额</th>
  48.                         <th >服务商应付款总额</th>
  49.                         <th >服务商已付款总额</th>
  50.                         <th >服务商未付款总额</th>
  51.                     </tr>
  52.                 </thead>
  53.                 <tbody id="table_body">
  54.                 </tbody>
  55.             </table>
  56.         
  57.    
复制代码

4、创建SignalR中心
中心是一个类,用于处理客户端服务器通信的高级管道。在SignalR_Demo项目文件夹中,创建Hubs文件夹。在Hubs文件夹中,使用已下代码创建ChatHub类。
  1. var builder = WebApplication.CreateBuilder(args);
  2. //添加SignalR服务
  3. builder.Services.AddSignalR();
  4. app.UseEndpoints(endpoints =>
  5. {
  6.     endpoints.MapControllerRoute(
  7.         name: "default",
  8.         pattern: "{controller=Home}/{action=Index}/{id?}"
  9.     );
  10.     //添加SignalR端点
  11.     endpoints.MapHub<ServerMonitorHub>("/serverMonitorHub");
  12. });
复制代码
3.2-客户端(Vue3+Vite+TS)

(1)安装SugbalR
  1.     public class ChatHub:Hub
  2.     {
  3.         /// <summary>
  4.         /// 发送消息
  5.         /// </summary>
  6.         /// <param name="user">用户名</param>
  7.         /// <param name="message">发送信息</param>
  8.         /// <returns></returns>
  9.         public async Task SendMessage(string user,string message)
  10.         {
  11.             await Clients.All.SendAsync("ReceiveMessage", user, message);
  12.         }
  13.     }
复制代码
版本截图:

(2)然后新建一个文件,用来封装业务逻辑相关代码。[西瓜程序猿]是在【src/utils】目录下,创建了一个名为【signalr.ts】的文件,也可以是js文件,根据自己项目的需要去新建。

代码:
  1. npm install @latelier/vue-signalr
复制代码
(3)然后再页面进行使用。[西瓜程序猿]这里以Echarts图表展示为例子,首先先安装Echarts包。
npm install echarts --save
版本截图:

(4)然后再页面写入如下代码,具体页面样式根据需要进行修改调整哦。
  1. import * as signalR  from '@microsoft/signalr';
  2. //如果需要身份验证
  3. //.withUrl('/messageHub', {accessTokenFactory: () => sessionStorage.getItem('token')})
  4. let connection;
  5. // 建立连接
  6. async function start(url) {
  7.   try {
  8.     connection = new signalR.HubConnectionBuilder()
  9.       .withUrl(url)//跨域需要使用绝对地址
  10.       .configureLogging(signalR.LogLevel.Information)
  11.       .withAutomaticReconnect() // 设置自动重连机制
  12.       .build();
  13.   } catch(err) {
  14.     console.log(err);
  15.     setTimeout(start, 10000);//错误重连
  16.   }
  17. }
  18. // 开始signalr连接
  19. const connect = async (url) => {
  20.   await start(url);
  21.   console.log(`【西瓜程序猿-${new Date().toLocaleString()}:SignalR已连接成功!`);
  22. };
  23. // 调用服务端方法
  24. async function send(methodName, param){
  25.   try {
  26.     await connection.invoke(methodName, param);
  27.   } catch (err) {
  28.     console.error(err);
  29.   }
  30. }
  31. //断开连接
  32. const disconnect = async ()=>{
  33.   await connection.stop();
  34.   console.log(`【西瓜程序猿-${new Date().toLocaleString()}:SignalR已断开连接!`);
  35. };
  36. export {
  37.   connection,
  38.   connect,
  39.   send,
  40.   disconnect
  41. };
复制代码
(5)效果展示:



原文链接:https://www.cnblogs.com/kimiliucn/p/17648543.html

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4