├── .gitignore ├── CONTRIBUTING.md ├── LICENSE ├── README.md ├── config ├── casper.js ├── const.js └── wxDom.js ├── index.js ├── note └── element.docx ├── package.json ├── src ├── directive │ ├── exact.js │ ├── explain │ │ ├── baike.js │ │ ├── ip.js │ │ ├── mPhone.js │ │ └── weather.js │ └── fuzzy.js ├── lib │ ├── hello.js │ ├── listen.js │ ├── logo.js │ ├── machine.js │ ├── message.js │ ├── open.js │ ├── openImage.sh │ ├── qrcode.js │ ├── searchWx.js │ └── turing.js └── utils │ ├── ajax.js │ └── message.js └── static ├── img ├── wechat-robot-deep.png └── wechat-robot.psd └── js └── jquery.js /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | static/img/* -------------------------------------------------------------------------------- /CONTRIBUTING.md: -------------------------------------------------------------------------------- 1 | # 指令和AI 2 | 在这之前你可能需要两个方法: 3 | 4 | **message.send()** 5 | ``` 6 | //路径:src/utils/message.js 7 | //回复消息的方法message.send 8 | //使用如: 9 | var message = require('/src/utils/message'); 10 | ... 11 | casper.then(function(){ 12 | message.send(this, '你好'); 13 | }) 14 | ``` 15 | 16 | **ajax() / ajax.get() / ajax.post() / ajax.getJSON()** 17 | ``` 18 | //路径:src/utils/ajax.js 19 | //在聊天器调ajax的方法 20 | //使用如: 21 | var ajax= require('/src/utils/ajax'); 22 | ... 23 | casper.then(function(){ 24 | ajax(this, "http://example.com/getInfo", 'get', {uid: 123}, function(res){ 25 | console.log(JSON.stringify(res)); 26 | }); 27 | 28 | //或者 29 | ajax.get(this, "http://example.com/getInfo", {uid: 123}, function(res){ 30 | console.log(JSON.stringify(res)); 31 | }); 32 | 33 | //jsonp 34 | ajax.getJSON(this, "http://example.com/getInfo?calback=?", {uid: 123}, function(res){ 35 | console.log(JSON.stringify(res)); 36 | }); 37 | }) 38 | ``` 39 | 你可以根据需要按以下方面扩充机器人的功能: 40 | ##### 编写指令 41 | ``` 42 | //你可以写一些指令而不是交给AI处理 43 | //指令分为精确匹配指令和模糊匹配指令 44 | 45 | //------------------------------------------------------------------- 46 | //精确匹配 47 | //精确指令在`/src/directive/exact.js`下编写。 48 | //key是指令名称,值是一个方法,接受参数msgContent(用户发送的消息)和casperIns(casper实例),如: 49 | //当对方发送'关闭'指令时程序提示并退出。 50 | module.exports = { 51 | '关闭': function (msgContent, casperIns) { 52 | message.send(casperIns, '[玫瑰]感谢您的使用[玫瑰]\n\r([闪电]需要开启请在控制台启动程序[闪电])'); 53 | casperIns.echo('微信发出关闭口令,程序退出。') 54 | return casperIns.exit(); 55 | } 56 | } 57 | 58 | //------------------------------------------------------------------- 59 | //模糊匹配 60 | //模糊匹配指令在`/src/directive/fuzzy.js`下编写。 61 | //接收参数前两个同精确指令,第三个是当前的正则表达式; 62 | //指令可单独防在src/directive/explain下方便维护,如weather.js: 63 | //匹配'地名 + 天气',调用天气api 64 | var weather = require('./explain/weather'); 65 | module.exports = { 66 | '/天气/g': weather 67 | } 68 | //weather.js实现如下: 69 | var ajax = require('../../utils/ajax'); 70 | var message = require('../../utils/message'); 71 | 72 | var formatWeather = function(local, weather) { 73 | //此方法对返回的json格式化,详情请查看源码 74 | } 75 | 76 | var weather = function(msgContent, casperIns, regex) { 77 | var local = msgContent.replace(/ |天气/, ''); 78 | var resource = 'http://wthrcdn.etouch.cn/weather_mini?city=' + encodeURIComponent(local); 79 | 80 | ajax.get(casperIns, resource, {}, function(res){ 81 | var weather = JSON.parse(res); 82 | if (weather.status == 1000) { 83 | message.send(casperIns, formatWeather(local, weather)); 84 | } else { 85 | message.send(casperIns, '未查找到相关天气信息。请尝试输入格式如"广州天气"。') 86 | } 87 | }); 88 | 89 | } 90 | module.exports = weather; 91 | //------------------------------------------------------------------- 92 | ``` 93 | 94 | ##### 完善AI 95 | 这个我目前也没研究。 96 | 97 | ##### 调试 98 | 调试代码时可将`capserjs`配置选像中的`logLevel`字段设为`info`将会显示更多`phantomjs`log,更多配置请移步[casperjs](http://casperjs.org/)文档。 99 | ``` 100 | //路径:config/casper.js 101 | module.exports = { 102 | clientScripts: [ 103 | 'static/js/jquery.js' 104 | ], 105 | pageSettings: { 106 | loadImages: true, 107 | userAgent: 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.14 Safari/537.36', 108 | }, 109 | logLevel: "info", //here 110 | viewportSize: {width: 1300, height: 900}, 111 | verbose: true, 112 | waitTimeout: 1000 * 60 * 60 * 24 * 365, 113 | onWaitTimeout: function(){ 114 | console.log( 'waitFor*方法超时...' ) 115 | } 116 | } 117 | 118 | ``` 119 | 另外,在执行各个步骤时会把浏览器截图保存到`/static/img`,方便查看浏览器渲染情况。比如每获取到一条新消息时会截图并覆盖为`lastNewMsgContent.png`。 120 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2017 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # **官方可能存在打压行为,项目可能已运行不正常。不再维护,感谢!** 2 | 3 | 4 | 5 |  6 | 7 | [](https://github.com/doterlin/wechat-robot) 8 | [](https://github.com/doterlin/wechat-robot/blob/master/LICENSE) 9 | [](https://github.com/doterlin) 10 | 11 | 12 | >基于`phantomjs`和web端微信开发的聊天机器人。使用的微信账号(即充当机器人的账号)为个人账号,可自定义指令。 13 | # 效果 14 | 可以指定一个微信账号: 15 | 16 |  17 | 18 | 自定义指令不会交给AI处理: 19 | 20 |  21 | 22 | 独乐乐不如众乐乐,可以指定到群聊上: 23 | 24 |  25 | 26 | 27 | 在后台程序查看log: 28 | 29 |  30 | 31 | 32 | 33 | # 使用场景 34 | > + 微信智能回复,监控,统计等 35 | > + 便捷查询 36 | > + 娱(liao)乐(mei) 37 | 38 | 39 | # 如何使用 40 | ### 1.安装环境 41 | 下载[源码](https://github.com/doterlin/wechat-robot): 42 | ``` 43 | git clone https://github.com/doterlin/wechat-robot.git 44 | ``` 45 | 46 | 安装以下环境: 47 | 48 | 1.[node.js](https://nodejs.org/) 49 | 50 | 2.[phantomjs](http://phantomjs.org/) 51 | 52 | 3.[casperjs](http://casperjs.org/) 53 | 54 | 55 | ### 2.配置 56 | 找到并配置好`/config/const.js` 57 | ``` 58 | //机器人名字 59 | var ROBOT_NAME = "小强"; 60 | 61 | module.exports = { 62 | //微信web版地址 63 | 'URL' : 'https://wx.qq.com/', 64 | 65 | //机器人名字 66 | 'ROBOT_NAME' : ROBOT_NAME, 67 | 68 | //图灵机器人apiKey和api地址 69 | //AI部分使用的是第三方机器人图灵(http://www.tuling123.com/) 70 | //这里只是示例,请大家自行到图灵官网注册并替换掉apikey免费版限5000次调用/天。有条件的同学可以付费支持下好产品 71 | 'TURING_APIKEY': '99fecec3424d416898b91b0998e2b26a', 72 | 'TURING_URL' : 'http://www.tuling123.com/openapi/api', 73 | 74 | //锁定的微信号备注,注意是备注;如果是群聊则填群聊名称即可。 75 | //填写的名称请预先在手机微信上搜索确认搜索结果是否出现在第一个 76 | 'TARGET_NICK' : '二十投小分队', 77 | 78 | //启动时打招呼消息 79 | 'HELLO_WORLD' : '[闪电]' + ROBOT_NAME + '[闪电]已启动...', 80 | 81 | } 82 | ``` 83 | 84 | 85 | ### 3.运行和登录微信 86 | **安装依赖和运行命令:** 87 | ``` 88 | npm install 89 | ``` 90 | ``` 91 | casperjs index.js 92 | ``` 93 | *如果提示`python找不到`之类的话请安装一下[python](https://www.python.org/)并保证其能运行在全局* 94 | 95 | **扫码登录微信** 96 | 运行后看到如下提示时则扫一下弹出来的二维码: 97 | ``` 98 | 正在加载二维码... 99 | 已保存二维码,路径:"static/img/qr.jpg". 100 | 正在使用默认软件打开二维码,请用手机微信扫一扫确认登录 (若没有请手动打开) 101 | ``` 102 | 请在`一分钟内`使用手机扫码并确认登录。出现`登录成功`和`发送欢迎语`提示后即可。这样就完成了使用步骤。下面的章节是介绍如何去扩展功能。 103 | 104 | # 指令和AI 105 | 在这之前你可能需要两个方法: 106 | 107 | **message.send()** 108 | ``` 109 | //路径:src/utils/message.js 110 | //回复消息的方法message.send 111 | //使用如: 112 | var message = require('/src/utils/message'); 113 | ... 114 | casper.then(function(){ 115 | message.send(this, '你好'); 116 | }) 117 | ``` 118 | 119 | **ajax() / ajax.get() / ajax.post() / ajax.getJSON()** 120 | ``` 121 | //路径:src/utils/ajax.js 122 | //在聊天器调ajax的方法 123 | //使用如: 124 | var ajax= require('/src/utils/ajax'); 125 | ... 126 | casper.then(function(){ 127 | ajax(this, "http://example.com/getInfo", 'get', {uid: 123}, function(res){ 128 | console.log(JSON.stringify(res)); 129 | }); 130 | 131 | //或者 132 | ajax.get(this, "http://example.com/getInfo", {uid: 123}, function(res){ 133 | console.log(JSON.stringify(res)); 134 | }); 135 | 136 | //jsonp 137 | ajax.getJSON(this, "http://example.com/getInfo?calback=?", {uid: 123}, function(res){ 138 | console.log(JSON.stringify(res)); 139 | }); 140 | }) 141 | ``` 142 | 你可以根据需要按以下方面扩充机器人的功能: 143 | ##### 编写指令 144 | ``` 145 | //你可以写一些指令而不是交给AI处理 146 | //指令分为精确匹配指令和模糊匹配指令 147 | 148 | //------------------------------------------------------------------- 149 | //精确匹配 150 | //精确指令在`/src/directive/exact.js`下编写。 151 | //key是指令名称,值是一个方法,接受参数msgContent(用户发送的消息)和casperIns(casper实例),如: 152 | //当对方发送'关闭'指令时程序提示并退出。 153 | module.exports = { 154 | '关闭': function (msgContent, casperIns) { 155 | message.send(casperIns, '[玫瑰]感谢您的使用[玫瑰]\n\r([闪电]需要开启请在控制台启动程序[闪电])'); 156 | casperIns.echo('微信发出关闭口令,程序退出。') 157 | return casperIns.exit(); 158 | } 159 | } 160 | 161 | //------------------------------------------------------------------- 162 | //模糊匹配 163 | //模糊匹配指令在`/src/directive/fuzzy.js`下编写。 164 | //接收参数前两个同精确指令,第三个是当前的正则表达式; 165 | //指令可单独防在src/directive/explain下方便维护,如weather.js: 166 | //匹配'地名 + 天气',调用天气api 167 | var weather = require('./explain/weather'); 168 | module.exports = { 169 | '/天气/g': weather 170 | } 171 | //weather.js实现如下: 172 | var ajax = require('../../utils/ajax'); 173 | var message = require('../../utils/message'); 174 | 175 | var formatWeather = function(local, weather) { 176 | //此方法对返回的json格式化,详情请查看源码 177 | } 178 | 179 | var weather = function(msgContent, casperIns, regex) { 180 | var local = msgContent.replace(/ |天气/, ''); 181 | var resource = 'http://wthrcdn.etouch.cn/weather_mini?city=' + encodeURIComponent(local); 182 | 183 | ajax.get(casperIns, resource, {}, function(res){ 184 | var weather = JSON.parse(res); 185 | if (weather.status == 1000) { 186 | message.send(casperIns, formatWeather(local, weather)); 187 | } else { 188 | message.send(casperIns, '未查找到相关天气信息。请尝试输入格式如"广州天气"。') 189 | } 190 | }); 191 | 192 | } 193 | module.exports = weather; 194 | //------------------------------------------------------------------- 195 | ``` 196 | 197 | ##### 完善AI 198 | 这个我目前也没研究。 199 | 200 | ##### 调试 201 | 调试代码时可将`capserjs`配置选像中的`logLevel`字段设为`info`将会显示更多`phantomjs`log,更多配置请移步[casperjs](http://casperjs.org/)文档。 202 | ``` 203 | //路径:config/casper.js 204 | module.exports = { 205 | clientScripts: [ 206 | 'static/js/jquery.js' 207 | ], 208 | pageSettings: { 209 | loadImages: true, 210 | userAgent: 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.14 Safari/537.36', 211 | }, 212 | logLevel: "info", //here 213 | viewportSize: {width: 1300, height: 900}, 214 | verbose: true, 215 | waitTimeout: 1000 * 60 * 60 * 24 * 365, 216 | onWaitTimeout: function(){ 217 | console.log( 'waitFor*方法超时...' ) 218 | } 219 | } 220 | 221 | ``` 222 | 另外,在执行各个步骤时会把浏览器截图保存到`/static/img`,方便查看浏览器渲染情况。比如每获取到一条新消息时会截图并覆盖为`lastNewMsgContent.png`。 223 | 224 | 225 | > **欢迎二次开发提PR,共同完善项目!** 226 | 227 | 228 | # MIT License 229 | 230 | Copyright (c) 2020 doterlin 231 | 232 | Permission is hereby granted, free of charge, to any person obtaining a copy 233 | of this software and associated documentation files (the "Software"), to deal 234 | in the Software without restriction, including without limitation the rights 235 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 236 | copies of the Software, and to permit persons to whom the Software is 237 | furnished to do so, subject to the following conditions: 238 | 239 | The above copyright notice and this permission notice shall be included in all 240 | copies or substantial portions of the Software. 241 | 242 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 243 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 244 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 245 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 246 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 247 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 248 | SOFTWARE. 249 | -------------------------------------------------------------------------------- /config/casper.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | clientScripts: [ 3 | 'static/js/jquery.js' 4 | ], 5 | pageSettings: { 6 | loadImages: true, 7 | userAgent: 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.14 Safari/537.36', 8 | }, 9 | // logLevel: "info", 10 | viewportSize: {width: 1300, height: 900}, 11 | verbose: true, 12 | waitTimeout: 1000 * 60 * 60 * 24 * 365, 13 | onWaitTimeout: function(){ 14 | console.log( 'waitFor*方法超时...' ) 15 | } 16 | } 17 | -------------------------------------------------------------------------------- /config/const.js: -------------------------------------------------------------------------------- 1 | // var emotionList = ['微笑', '撇嘴', '色', '发呆', '得意', '流泪', '害羞', '闭嘴', '睡', '大哭', '尴尬', '发怒', '调皮', '呲牙', '惊讶', '难过', '酷', '冷汗', '抓狂', '吐', '偷笑', '愉快', '白眼', '傲慢', '饥饿', '困', '惊恐', '流汗', '憨笑', '大兵', '奋斗', '咒骂', '疑问', '嘘', '晕', '折磨', '衰', '骷髅', '敲打', '再见', '擦汗', '抠鼻', '鼓掌', '糗大了', '坏笑', '左哼哼', '右哼哼', '哈欠', '鄙视', '委屈', '快哭了', '阴险', '亲亲', '吓', '可怜', '菜刀', '西瓜', '啤酒', '篮球', '乒乓', '咖啡', '饭', '猪头', '玫瑰', '凋谢', '示爱', '爱心', '心碎', '蛋糕', '闪电', '炸弹', '刀', '足球', '瓢虫', '便便', '月亮', '太阳', '礼物', '拥抱', '强', '弱', '握手', '胜利', '抱拳', '勾引', '拳头', '差劲', '爱你', 'NO', 'OK', '爱情', '飞吻', '跳跳', '发抖', '怄火', '转圈', '磕头', '回头', '跳绳', '挥手', '激动', '街舞', '献吻', '左太极', '右太极', '嘿哈', '捂脸', '奸笑', '机智', '皱眉', '耶', '红包', '鸡']; 2 | var ROBOT_NAME = "小强"; 3 | 4 | module.exports = { 5 | //微信web版地址 6 | 'URL' : 'https://wx.qq.com/', 7 | 8 | //机器人名字 9 | 'ROBOT_NAME' : ROBOT_NAME, 10 | 11 | //图灵机器人apiKey和api地址 12 | //AI部分使用的是第三方机器人图灵(http://www.tuling123.com/) 13 | 'TURING_APIKEY': '99fecec3424d416898b91b0998e2b26a', 14 | 'TURING_URL' : 'http://www.tuling123.com/openapi/api', 15 | 16 | //锁定的微信号备注,注意是备注;如果是群聊则填群聊名称即可。 17 | 'TARGET_NICK' : '二十投小分队', 18 | 19 | //启动时打招呼消息 20 | 'HELLO_WORLD' : '[闪电]' + ROBOT_NAME + '[闪电]已启动...\n\r\n\r现在的时间是:'+new Date().toLocaleString(), 21 | 22 | } 23 | -------------------------------------------------------------------------------- /config/wxDom.js: -------------------------------------------------------------------------------- 1 | //微信web版需用的各个选择器,方便更新 2 | var testByMe = false, 3 | massageClass = testByMe? '.me': '.you', 4 | MSG_ID_ATTR = 'data-cm'; 5 | 6 | module.exports = { 7 | //登录 8 | 'QR_CODE' : '.qrcode .img', 9 | 'LOGIN_BODY' : 'body.loaded', 10 | 'LOGIN_LOADING' : '.chat_list .chat_list .ico_loading', 11 | 12 | //搜索指定微信 13 | 'SEARCH_INPUT' : '.frm_search', 14 | 'SEARCH_RESULT' : '.recommendation', 15 | 'SEARCH_RESULT_ONE' : '.recommendation .contact_item', 16 | 'SEARCH_RESULT_FRIEND': '.recommendation .nickname', 17 | 18 | //输入框 19 | 'CHAT_NO_SELECT' : '.chat_bd .web_wechat_nomes_icon', 20 | 'CHAT_INPUT' : '#editArea', 21 | 'CHAT_SEND' : '.btn.btn_send', 22 | 23 | //消息相关 24 | 'MSG_ID_ATTR' : MSG_ID_ATTR, 25 | 'MSG' : '.chat_bd ' + massageClass + ' [' + MSG_ID_ATTR + ']:not(.avatar)', 26 | 'MSG_TEXT' : '.chat_bd ' + massageClass + ' [' + MSG_ID_ATTR + ']:not(.avatar) .js_message_plain', 27 | 'MSG_IMG' : '.chat_bd ' + massageClass + ' [' + MSG_ID_ATTR + ']:not(.avatar) img' 28 | } -------------------------------------------------------------------------------- /index.js: -------------------------------------------------------------------------------- 1 | var casper = require('casper').create(require('./config/casper')); 2 | var CONST = require('./config/const'); 3 | var WXDOM = require('./config/wxDom'); 4 | var qrcode = require('./src/lib/qrcode'); 5 | var searchWx = require('./src/lib/searchWx'); 6 | var listen = require('./src/lib/listen'); 7 | var hello = require('./src/lib/hello'); 8 | var machine = require('./src/lib/machine'); 9 | 10 | // welcome 11 | casper.echo(require('./src/lib/logo').string); 12 | console.log('正在加载网页...'); 13 | casper.start(CONST.URL); 14 | 15 | //二维码与登录 16 | casper.then(function () { 17 | qrcode.start(this); 18 | }); 19 | 20 | // 等待加载聊天信息 & 搜索目标微信号 21 | 22 | casper.then(function () { 23 | casper.waitWhileVisible(WXDOM.LOGIN_LOADING, function () { 24 | this.echo("加载最近聊天信息完毕!"); 25 | searchWx(this, CONST.TARGET_NICK) 26 | }, function () { 27 | this.echo('加载聊天信息失败!'); 28 | this.exit(); 29 | }, 30 * 1000) 30 | }) 31 | 32 | // 发送欢迎语 33 | casper.then(function () { 34 | hello(this); 35 | }) 36 | 37 | // 监听新消息 38 | casper.then(function () { 39 | listen.start(this); 40 | }) 41 | 42 | //处理新消息 43 | casper.on('newMsg', function (msg, isTextMsg) { 44 | this.echo('触发新文本消息事件,新消息:\n “' + msg + '”'); 45 | machine.reply(msg, isTextMsg, this); 46 | this.wait(10, function () { 47 | this.echo('已回复'); 48 | }) 49 | }) 50 | 51 | casper.on('exit', function () { 52 | this.echo('-----------------\n已退出程序!') 53 | }); 54 | 55 | casper.on("page.error", function(msg, trace) { 56 | this.echo("Browser error: " + msg, "ERROR"); 57 | }); 58 | 59 | casper.run(function () { 60 | this.echo('执行完毕,程序退出.'); 61 | this.exit(); 62 | }); -------------------------------------------------------------------------------- /note/element.docx: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/doterlin/wechat-robot/bb73f45dc343ef6816fcec940f783e13f907606a/note/element.docx -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "wx-robot", 3 | "version": "1.0.0", 4 | "description": "A WeChat automatic response robot.", 5 | "main": "index.js", 6 | "dependencies": { 7 | "opn": "^5.1.0" 8 | }, 9 | "devDependencies": {}, 10 | "scripts": { 11 | "test": "echo \"Error: no test specified\" && exit 1" 12 | }, 13 | "keywords": [ 14 | "wechat", 15 | "robot", 16 | "js" 17 | ], 18 | "author": "doterlin", 19 | "license": "MIT" 20 | } 21 | -------------------------------------------------------------------------------- /src/directive/exact.js: -------------------------------------------------------------------------------- 1 | var message = require('../utils/message'); 2 | 3 | module.exports = { 4 | '关闭': function (msgContent, casperIns) { 5 | message.send(casperIns, '[玫瑰]感谢您的使用[玫瑰]\n\r([闪电]需要开启请在控制台启动程序[闪电])'); 6 | casperIns.echo('微信发出关闭口令,程序退出。') 7 | return casperIns.exit(); 8 | }, 9 | 10 | '报时': function (msgContent, casperIns) { 11 | message.send(casperIns, '服务器的时间是:' + new Date().toLocaleString()); 12 | } 13 | } -------------------------------------------------------------------------------- /src/directive/explain/baike.js: -------------------------------------------------------------------------------- 1 | var ajax = require('../../utils/ajax'); 2 | var message = require('../../utils/message'); 3 | 4 | module.exports = function(msgContent, casperIns, regex) { 5 | var key = msgContent.replace(/ |百科/g, ''); 6 | if(!key){ 7 | return message.send(casperIns, '请输入要查找的内容。'); 8 | } 9 | casperIns.echo('正在查找百科:' + key); 10 | var resource = 'http://baike.baidu.com/api/openapi/BaikeLemmaCardApi?scope=103&format=json&appid=379020&bk_length=600&bk_key=' + encodeURIComponent(key); 11 | resource += "&callback=?"; //此接口需跨域处理 12 | 13 | ajax.getJSON(casperIns, resource, {}, function(res){ 14 | var str = res.title? ('【'+res.title+'】\n\r') : ''; 15 | str += res.desc? ('描述:'+res.desc+'\n\r') : ''; 16 | str += res.abstract || ''; 17 | console.log(JSON.stringify(str)); 18 | message.send(casperIns, str); 19 | }); 20 | } 21 | -------------------------------------------------------------------------------- /src/directive/explain/ip.js: -------------------------------------------------------------------------------- 1 | var ajax = require('../../utils/ajax'); 2 | var message = require('../../utils/message'); 3 | 4 | module.exports = function(msgContent, casperIns, regex) { 5 | var phone = msgContent; 6 | var resource = 'http://whois.pconline.com.cn/ip.jsp?ip=' + msgContent; 7 | 8 | ajax.get(casperIns, resource, {}, function(res){ 9 | var res = JSON.parse(res.replace('\n','')); 10 | var str = '【ip信息】\n\r'; 11 | message.send(casperIns, res); 12 | }); 13 | } 14 | -------------------------------------------------------------------------------- /src/directive/explain/mPhone.js: -------------------------------------------------------------------------------- 1 | var ajax = require('../../utils/ajax'); 2 | var message = require('../../utils/message'); 3 | 4 | module.exports = function(msgContent, casperIns, regex) { 5 | var phone = msgContent; 6 | var resource = 'https://tcc.taobao.com/cc/json/mobile_tel_segment.htm?tel=' + msgContent; 7 | 8 | ajax.get(casperIns, resource, {}, function(res){ 9 | var res = JSON.parse(res.replace('\n','')); 10 | var str = '【' + phone + ' 手机信息】\n\r\n\r'; 11 | if (res.carrier) { 12 | str += res.carrier? ('运营商:' + res.carrier + '\n\r') : ''; 13 | str += res.areaVid? ('areaVid' + res.areaVid + '\n\r') : ''; 14 | str += res.ispVid? ('ispVid' + res.ispVid + '\n\r') : ''; 15 | } else { 16 | message.send(casperIns, '未查找到相关手机信息。请确认该手机号存在。') 17 | } 18 | }); 19 | } 20 | -------------------------------------------------------------------------------- /src/directive/explain/weather.js: -------------------------------------------------------------------------------- 1 | var ajax = require('../../utils/ajax'); 2 | var message = require('../../utils/message'); 3 | 4 | var formatWeather = function(local, weather) { 5 | var weather = weather.data; 6 | var str = '【' + local +'天气】\n\r\n\r'; 7 | for (var i in weather.forecast[0]) { 8 | if (i == 'date') { 9 | str += '【' + weather.forecast[0][i] + "】\n\r"; 10 | } else { 11 | str += weather.forecast[0][i] + ' ' 12 | } 13 | } 14 | 15 | str += '\n\r'; 16 | 17 | for (var i in weather.forecast[1]) { 18 | if (i == 'date') { 19 | str += '【' + weather.forecast[1][i] + "】\n\r"; 20 | } else { 21 | str += weather.forecast[1][i] + ' ' 22 | } 23 | } 24 | 25 | if (weather.ganmao) { 26 | str += '\n\r\n\r[玫瑰]' + weather.ganmao + '[玫瑰]'; 27 | } 28 | return str; 29 | } 30 | 31 | var weather = function(msgContent, casperIns, regex) { 32 | var local = msgContent.replace(/ |天气/, ''); 33 | var resource = 'http://wthrcdn.etouch.cn/weather_mini?city=' + encodeURIComponent(local); 34 | 35 | ajax.get(casperIns, resource, {}, function(res){ 36 | var weather = JSON.parse(res); 37 | if (weather.status == 1000) { 38 | message.send(casperIns, formatWeather(local, weather)); 39 | } else { 40 | message.send(casperIns, '未查找到相关天气信息。请尝试输入格式如"广州天气"。') 41 | } 42 | }); 43 | 44 | } 45 | 46 | module.exports = weather; -------------------------------------------------------------------------------- /src/directive/fuzzy.js: -------------------------------------------------------------------------------- 1 | var weather = require('./explain/weather'); 2 | var baike = require('./explain/baike'); 3 | var mPhone = require('./explain/mPhone'); 4 | var ip = require('./explain/ip'); 5 | 6 | module.exports = { 7 | //天气 8 | '/天气/g': weather, 9 | 10 | //百科 11 | '/百科/g': baike, 12 | 13 | //手机查询 14 | '/^1[34578]\d{9}$/g': mPhone, 15 | 16 | //ip地址 17 | '/\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\b/g': ip, 18 | 19 | '/彩票/': function (msgContent, casperIns) { 20 | //'此指令正在开发...' 21 | } 22 | } -------------------------------------------------------------------------------- /src/lib/hello.js: -------------------------------------------------------------------------------- 1 | var WXDOM = require('../../config/wxDom'); 2 | var message = require('../utils/message'); 3 | 4 | 5 | var hello = function (casperIns) { 6 | var ts = casperIns; 7 | 8 | ts.waitForSelector(WXDOM.CHAT_INPUT, function () { 9 | ts.echo('已加载输入框...'); 10 | message.send(ts, CONST.HELLO_WORLD); 11 | ts.echo('已发送欢迎语...'); 12 | }); 13 | } 14 | 15 | module.exports = hello; -------------------------------------------------------------------------------- /src/lib/listen.js: -------------------------------------------------------------------------------- 1 | var listen = {}; 2 | 3 | var initData = { 4 | targetMessageIds: [], 5 | lastMsgId: '' 6 | }; 7 | 8 | listen.then = function (casperIns) { 9 | var ts = casperIns; 10 | ts.echo('-----------------------------\n监听到新消息,正在回复...'); 11 | ts.echo('新消息id: ' + initData.lastMsgId); 12 | ts.captureSelector('./static/img/lastNewMsgContent.png', 'html'); 13 | initData.targetMessageIds.push(initData.lastMsgId); 14 | 15 | ts.echo('当前对方消息数量:' + initData.targetMessageIds.length); 16 | 17 | var newMsgContent = ts.evaluate(function (MSG_SELECTOR, MSG_TEXT_SELECTOR) { 18 | //这里按元素选择器把消息分为两类,文本类和其他类 19 | var el_msg = document.querySelectorAll(MSG_SELECTOR); 20 | var el_msg_text = el_msg[el_msg.length - 1].querySelectorAll(MSG_TEXT_SELECTOR); 21 | var len = el_msg_text.length; 22 | if (len == 0) { 23 | return ''; 24 | } 25 | return el_msg_text[len - 1].innerHTML; 26 | }, WXDOM.MSG, WXDOM.MSG_TEXT); 27 | 28 | if (newMsgContent) { 29 | ts.emit('newMsg', newMsgContent, true); 30 | // message.send('您发送的消息:"' + newMsgContent + '"\n\r发送时间:' + new Date().toLocaleString()); 31 | } else { 32 | ts.emit('newMsg', newMsgContent, false); 33 | // message.send('无法识别您发的消息。' + '\n\r发送时间:' + new Date().toLocaleString()); 34 | } 35 | } 36 | 37 | listen.start = function (casperIns) { 38 | var ts = casperIns; 39 | // 监听新消息 40 | function loopListenNewMassage() { 41 | ts.echo('执行监听新消息:loopListenNewMassage()...') 42 | ts.unwait(); 43 | ts.waitFor(function checkMsgChange() { 44 | var msgAttrValues = ts.getElementsAttribute(WXDOM.MSG, WXDOM.MSG_ID_ATTR); 45 | if (msgAttrValues.length == 0) { 46 | return false; 47 | } 48 | var lastMsgId = JSON.parse(msgAttrValues[msgAttrValues.length - 1]).msgId; 49 | if (initData.targetMessageIds.indexOf(lastMsgId) < 0) { 50 | initData.lastMsgId = lastMsgId; 51 | return true; 52 | } 53 | return false; 54 | 55 | }, function then() { 56 | listen.then(ts); 57 | this.wait(10, loopListenNewMassage); 58 | 59 | }, function timeout() { }, 1000 * 60 * 60 * 24 * 365); 60 | } 61 | 62 | loopListenNewMassage(); 63 | } 64 | 65 | module.exports = listen; -------------------------------------------------------------------------------- /src/lib/logo.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | string: "\n\nMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM\n\ 3 | MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM\n\ 4 | MMMMMMMMMMMMMMMMMMMMMMMMMMWWWMMMWNWMMMMMMMMMMMMMMMMMMMMMMMWWMMMMMMMWWMMMMMMMMMMMMMMMWNNWMMMMMMMMMMMMMMMMWNNWMMWWMMMMMMWW\n\ 5 | MMMMNxlOWWklo0WXxokOollllloOWW0o;,;xNMKddKMMNxoOWMMXxlkNMWkclllllllONMM0olllllkNMMNkc;,lKMMKolllllkNMMXxc,;oXM0lclllllxX\n\ 6 | MMMMK;'kWK; ;0Wk':Ol......;KXo.... .dNx.cXMMO,;0MMWx..kWMK;... ..,OMMNo..... 'OW0:... .lNWd..... 'OWk;... .dXl.. ...dW\n\ 7 | MMMM0,;KXl..lN0;;0O,,xOOOk0Kc.;x0O:,x0:,OMMNl'xWMWO, ,0MMXOkx:.cOk0NMMO,,xOOl.;0k,'oOO;.:K0;,xOOl.;Ox',d0k,.lK0Ol.;xOOXM\n\ 8 | MMMMx.lKo...xK:,ONl.dWWWWMXc'oXMMMK0Nd.lXNNk':XMWO;..;KMMMM0o,;KMWWMWNl'xWMNo.ok,,OWMWd.cKo.lXNKc'dd':KMMNl.oWMXc'kWMMMM\n\ 9 | MMMWo.oo,:,'dc,kWO'.,:cOWWd'lNMMMMMM0;..,,,.'kWM0:cl.:XMMMNo..dWNxcoOx'.:ll;.cO:'kWMMNl.xO,..,,',dk,;0MMMK:'OMWk'cXMMMMM\n\ 10 | MMMX:.;,lx'';,dWXc....:KM0,,0MMMMMMNo..'''..cXM0::0d.lNMMMO,.:KMO' 'xc... .,dKd.lNMMMO,:0l..''..;Ol.xWMMWx'lNMX:'OMMMMMM\n\ 11 | MMM0,..cKx...oNWk':OK0KWWd.lNMMMWNM0,;ONN0;'OWK:.,c'.oWMMNl';xWW0xdkd';kl.,0MK:'kMMMK:,kk':0NXx.;d,,0MMWO,;0MWx'lNMMMMMM\n\ 12 | MMMk..:KNl..lXMX:'OMMMWWXc.cXMW0oONl'xWMWx'lNKc,::;..dWMMk':OXMMMMMK:'OWk.;KMO'.kMWO;'xKc'kWMNo.od.,0MNk,,kWMK;,OMMMMMMM\n\ 13 | MMWo.;0MX:.cXMWx..clllo0Xc..;c;.,OO':XMMK;,OKc;OWWX:.xMMXc'kWWMMMMWx'lNMk.;KMO'.'c:.'xNk..:ll;.cKx..,c;.;OWMWd.oNMMMMMMM\n\ 14 | MM0;.xWWk''OMM0;......:XWk' .,oKKc.xWMNo.cOc'xWMWO,'kMWx':XMMMMMM0;,OMWd.,OMXl. ..c0WK:.....'oXMK:. .'oKMMWO,,OMMMMMMMM\n\ 15 | MMXOOXMWKO0NMMXOkkkkkk0WMWKxxOXWMXOOXMMN0OKXOONMMWKOOXMW0OKWMMMMMWXO0NMWKOOXMMNOxkKWMMXOkkOOOXWMMMXkxkXWMMMWKO0NMMMMMMMM\n\ 16 | MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM\n\ 17 | MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM\n\ 18 | MMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMMM\n\n" 19 | } -------------------------------------------------------------------------------- /src/lib/machine.js: -------------------------------------------------------------------------------- 1 | var WXDOM = require('../../config/wxDom'); 2 | var message = require('../utils/message'); 3 | var turing = require('./turing'); 4 | var exactDirective = require('../directive/exact'); 5 | var fuzzyDirective = require('../directive/fuzzy'); 6 | 7 | var machine = {}; 8 | 9 | machine.reply = function(msg, isTextMsg, casperIns){ 10 | console.log('正在处理消息... '); 11 | 12 | //非文字类消息 13 | if(!isTextMsg) return dealUnknownMsg(casperIns); 14 | 15 | //是否是指令 16 | if(isDiretive(msg, casperIns)) return; 17 | 18 | //非指令消息交给第三方机器人 19 | return dealByMachine(casperIns, msg); 20 | } 21 | 22 | //判断是否是指令 23 | function isDiretive(msg, casperIns){ 24 | for(var diretive in exactDirective){ 25 | if(diretive == msg){ 26 | casperIns.echo('接受到精确匹配指令 ' + diretive + ' ,正在处理...'); 27 | exactDirective[diretive](msg, casperIns); 28 | return true; 29 | } 30 | } 31 | 32 | for(var regex in fuzzyDirective){ 33 | if(eval(regex).test(msg)){ 34 | casperIns.echo('接受到模糊匹配指令 ' + regex + ' ,正在处理...') 35 | fuzzyDirective[regex](msg, casperIns, regex); 36 | return true; 37 | } 38 | } 39 | 40 | return false; 41 | } 42 | 43 | function dealUnknownMsg(casperIns){ 44 | message.send(casperIns, '无法识别您发的消息。' + '\n\r发送时间:' + new Date().toLocaleString()); 45 | } 46 | 47 | function dealByMachine(casperIns, msg){ 48 | turing(casperIns, msg) 49 | } 50 | 51 | module.exports = machine; -------------------------------------------------------------------------------- /src/lib/message.js: -------------------------------------------------------------------------------- 1 | /*发送消息模块*/ 2 | var WXDOM = require('../../config/wxDom'); 3 | 4 | var message = {}; 5 | 6 | message.send = function(casperIns, msg){ 7 | if (!casperIns.visible(WXDOM.CHAT_INPUT) || !casperIns.visible(WXDOM.CHAT_INPUT)) { 8 | casperIns.captureSelector('./static/img/sendMessageError.png', 'html'); 9 | return casperIns.echo("Module message:息输入框框或者发送按钮未加载!"); 10 | } 11 | try { 12 | casperIns.sendKeys(WXDOM.CHAT_INPUT, msg); 13 | casperIns.click(WXDOM.CHAT_SEND); 14 | } catch (error) { 15 | console.log('Module message: 发送消息失败-' + error.message, 'ERROR') 16 | } 17 | } 18 | 19 | module.exports = message; -------------------------------------------------------------------------------- /src/lib/open.js: -------------------------------------------------------------------------------- 1 | var opn = require('opn'); 2 | console.log('打开二维码...') 3 | // Opens the image in the default image viewer 4 | opn('static/img/qr.jpg').then(() => { 5 | console.log('关闭二维码!') 6 | }); -------------------------------------------------------------------------------- /src/lib/openImage.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | var opn = require('opn'); 3 | 4 | // Opens the image in the default image viewer 5 | opn('static/img/qr.jpg').then(() => { 6 | console.log('关闭二维码!') 7 | }); -------------------------------------------------------------------------------- /src/lib/qrcode.js: -------------------------------------------------------------------------------- 1 | /*处理二维码和登录*/ 2 | var execFile = require("child_process").execFile; 3 | var WXDOM = require('../../config/wxDom'); 4 | var qrcode = {}; 5 | 6 | //登录是否成功处理 7 | function isLogin(casperIns) { 8 | var ts = casperIns; 9 | ts.waitForSelector(WXDOM.LOGIN_BODY, function () { 10 | ts.captureSelector('./static/img/login_success.png', 'html'); 11 | ts.echo('登录成功!'); 12 | }, function () { 13 | ts.echo('您在一分钟未登录或者二维码加载失败,程序退出!', 'ERROR'); 14 | ts.exit(); 15 | }, 60 * 1000); 16 | } 17 | 18 | 19 | //开始生成和处理二维码 20 | qrcode.start = function (casperIns) { 21 | var ts = casperIns; 22 | var qrUrl = ts.getElementAttribute(WXDOM.QR_CODE, 'src'); 23 | 24 | ts.echo('正在加载二维码...'); 25 | ts.waitForResource(qrUrl, function () { 26 | ts.captureSelector('./static/img/qr.jpg', '.login_box', { 27 | format: 'jpg', 28 | quality: 100 29 | }); 30 | ts.echo('已保存二维码,路径:"static/img/qr.jpg".\n正在使用默认软件打开二维码,请用手机微信扫一扫确认登录 (若没有请手动打开)'); 31 | execFile("node", ["./src/lib/open.js"], null); 32 | 33 | isLogin(ts); 34 | }, function () { }, 120 * 1000); 35 | } 36 | 37 | module.exports = qrcode; -------------------------------------------------------------------------------- /src/lib/searchWx.js: -------------------------------------------------------------------------------- 1 | /*搜索目标微信*/ 2 | var WXDOM = require('../../config/wxDom'); 3 | 4 | var searchWx = function (casperIns, wxNick) { 5 | var ts = casperIns; 6 | 7 | ts.waitForSelector(WXDOM.SEARCH_INPUT, function () { 8 | ts.echo('好友搜索框已加载!'); 9 | ts.sendKeys(WXDOM.SEARCH_INPUT, CONST.TARGET_NICK); 10 | ts.captureSelector('./static/img/SEARCH_INPUT.png', 'html'); 11 | }, function () { 12 | ts.echo('好友搜索框加载超时!') 13 | }, 30 * 1000); 14 | 15 | // ts.waitForSelector(WXDOM.SEARCH_RESULT, function(){ 16 | 17 | ts.waitForSelector(WXDOM.SEARCH_RESULT_FRIEND, function () { 18 | ts.waitForText(CONST.TARGET_NICK, function () { 19 | ts.echo('已加载搜索结果:./static/img/SEARCH_RESULT.png'); 20 | ts.click(WXDOM.SEARCH_RESULT_ONE); 21 | ts.captureSelector('./static/img/SEARCH_RESULT.png', 'html'); 22 | }), function () { 23 | ts.echo('搜索结果未出现匹配的好友..请检查目标微信号昵称是否好友', 'ERROR'); 24 | ts.exit(); 25 | }, 10 * 1000; 26 | }, function () { 27 | ts.captureSelector('./static/img/SEARCH_RESULT_0.png', 'html'); 28 | ts.echo('搜索结果为零,请确认目标微信号昵称是否在您的联系人列表中', 'ERROR'); 29 | ts.exit(); 30 | }, 30 * 1000) 31 | 32 | // }) 33 | }; 34 | 35 | module.exports = searchWx; -------------------------------------------------------------------------------- /src/lib/turing.js: -------------------------------------------------------------------------------- 1 | var CONST = require('../../config/const'); 2 | var message = require('../utils/message'); 3 | var ajax = require('../utils/ajax'); 4 | 5 | var formatTuring = function (res) { 6 | var replyMsg = ''; 7 | 8 | replyMsg += res.text || ''; 9 | if (res.url) { 10 | replyMsg += '\n\r' + res.url; 11 | } 12 | if (res.list) { 13 | replyMsg += '\n\r\n\r'; 14 | res.list.forEach(function (item) { 15 | replyMsg += !item.source ? '' : ('【' + item.source + '】'); 16 | replyMsg += (item.article || ''); 17 | replyMsg += !item.name ? '' : ('【' + item.name + '】'); 18 | replyMsg += !item.info ? '' : ('\n\r' + item.info); 19 | replyMsg += !item.detailUrl ? '' : ('\n\r' + item.detailUrl); 20 | replyMsg += '\n\r\n\r'; 21 | }) 22 | } 23 | return replyMsg; 24 | } 25 | 26 | var turing = function (casperIns, msg) { 27 | var param = { 28 | "key": CONST.TURING_APIKEY, 29 | "info": msg, 30 | "userid": CONST.TARGET_NICK 31 | } 32 | 33 | casperIns.echo('turing....'); 34 | ajax.post(casperIns, CONST.TURING_URL, param, function (res) { 35 | console.log('turing res:' + JSON.stringify(res)) 36 | var errorCode = [40001, 40002, 40004, 40007]; 37 | 38 | if (errorCode.indexOf(res.code) >= 0) { 39 | return message.send(casperIns, '我出错啦,暂时不能回答您。'); 40 | } 41 | 42 | message.send(casperIns, formatTuring(res)); 43 | }); 44 | 45 | // //在页面使用ajax请求 46 | // casperIns.evaluate(function (resource, param) { 47 | // $.get(resource, param, function (data) { window.to_casper_turing = data; }) 48 | // }, CONST.TURING_URL, param); 49 | 50 | // casperIns.waitFor(function check() { 51 | // return casperIns.evaluate(function () { 52 | // return window.to_casper_turing 53 | // }) 54 | // }, function () { 55 | // var res = casperIns.evaluate(function () { 56 | // var turing = window.to_casper_turing; 57 | // window.to_casper_turing = null; 58 | // return turing; 59 | // }); 60 | 61 | 62 | // }, function () { 63 | // console.log('请求天气超时...') 64 | // }, 10000) 65 | 66 | } 67 | 68 | 69 | module.exports = turing; 70 | 71 | -------------------------------------------------------------------------------- /src/utils/ajax.js: -------------------------------------------------------------------------------- 1 | /** 2 | * 在浏览器发ajax请求并接受回脚本程序 3 | * @casperIns casper实例 4 | * @url String ajax地址 5 | * @type String 请求类型,'get'或者'post' 6 | * @pram Object 请求参数 7 | * @callback Function 回调 8 | */ 9 | 10 | var ajax = function(casperIns, url, type, param, callback){ 11 | if(type != 'get' && type !='post' && type!='getJSON'){ 12 | return console.log('type不合法!') 13 | } 14 | 15 | casperIns.echo('开始Ajax ' + type +'请求: ' + url); 16 | casperIns.echo('Ajax参数: ' + JSON.stringify(param)); 17 | casperIns.evaluate(function(url, type, param) { 18 | $[type](url, param, function(data) { 19 | window.wr_ajax_respond= data; 20 | }) 21 | }, url, type, param); 22 | 23 | casperIns.waitFor(function checkRespond() { 24 | return casperIns.evaluate(function() { 25 | return window.wr_ajax_respond; 26 | }) 27 | }, function() { 28 | var ajaxRespond = casperIns.evaluate(function() { 29 | var ajaxRespond = window.wr_ajax_respond; 30 | window.wr_ajax_respond = null; 31 | return ajaxRespond; 32 | }); 33 | 34 | setTimeout(function() { 35 | console.log('ajax callback....') 36 | callback && callback(ajaxRespond); 37 | }, 10); 38 | 39 | }, function() { 40 | casperIns.echo('请求超时...'); 41 | }, 10000) 42 | } 43 | 44 | ajax.get = function(casperIns, url, param, callback){ 45 | ajax(casperIns, url, 'get', param, callback); 46 | } 47 | 48 | ajax.getJSON = function(casperIns, url, param, callback){ 49 | ajax(casperIns, url, 'getJSON', param, callback); 50 | } 51 | 52 | ajax.post = function(casperIns, url, param, callback){ 53 | ajax(casperIns, url, 'post', param, callback); 54 | } 55 | 56 | module.exports = ajax; -------------------------------------------------------------------------------- /src/utils/message.js: -------------------------------------------------------------------------------- 1 | /*发送消息模块*/ 2 | var WXDOM = require('../../config/wxDom'); 3 | 4 | var message = {}; 5 | 6 | message.send = function(casperIns, msg){ 7 | if (!casperIns.visible(WXDOM.CHAT_INPUT) || !casperIns.visible(WXDOM.CHAT_INPUT)) { 8 | casperIns.captureSelector('./static/img/sendMessageError.png', 'html'); 9 | return casperIns.echo("Module message:输入框或者发送按钮未加载!"); 10 | } 11 | try { 12 | casperIns.sendKeys(WXDOM.CHAT_INPUT, msg); 13 | casperIns.click(WXDOM.CHAT_SEND); 14 | } catch (error) { 15 | console.log('Module message: 发送消息失败-' + error.message, 'ERROR') 16 | } 17 | } 18 | 19 | module.exports = message; -------------------------------------------------------------------------------- /static/img/wechat-robot-deep.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/doterlin/wechat-robot/bb73f45dc343ef6816fcec940f783e13f907606a/static/img/wechat-robot-deep.png -------------------------------------------------------------------------------- /static/img/wechat-robot.psd: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/doterlin/wechat-robot/bb73f45dc343ef6816fcec940f783e13f907606a/static/img/wechat-robot.psd -------------------------------------------------------------------------------- /static/js/jquery.js: -------------------------------------------------------------------------------- 1 | /*! jQuery v1.10.2 | (c) 2005, 2013 jQuery Foundation, Inc. | jquery.org/license 2 | //@ sourceMappingURL=jquery.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="
",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="",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="","#"===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="",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="t |