index.html 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="utf-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <title>后台首页</title>
  7. <link rel="shortcut icon" href="favicon.ico">
  8. <link href="https://cdn.bootcss.com/bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet">
  9. <link href="https://cdn.staticfile.org/font-awesome/4.4.0/css/font-awesome.css?v=4.4.0" rel="stylesheet">
  10. <link href="https://cdn.bootcss.com/bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet">
  11. <link href="__CSS__/bootstrap.min.css?v=3.3.6" rel="stylesheet">
  12. <link href="__CSS__/font-awesome.min.css?v=4.4.0" rel="stylesheet">
  13. <link href="__CSS__/plugins/bootstrap-table/bootstrap-table.min.css" rel="stylesheet">
  14. <link href="__CSS__/animate.min.css" rel="stylesheet">
  15. <link href="__CSS__/style.min.css?v=4.1.0" rel="stylesheet">
  16. <link href="__JS__/layui/css/myLayui.css" rel="stylesheet">
  17. </head>
  18. <body class="gray-bg">
  19. <div class="wrapper wrapper-content">
  20. <div class="row">
  21. <div class="col-sm-2">
  22. <div class="ibox float-e-margins">
  23. <div class="ibox-title" style="display: flex; justify-content: center;">
  24. <!--<span class="label label-primary pull-right">今天</span>-->
  25. <h3>敏感词报警</h3>
  26. </div>
  27. <div class="ibox-content" style="text-align: center">
  28. <h1 class="no-margins"><span id="allSensitive">0</span><span style="font-size: 14px; margin-left: 10px">次</span></h1>
  29. <small></small>
  30. </div>
  31. <div class="ibox-content" style="font-size: 12px; padding: 5px 20px 5px; height: 45px;">
  32. <div style="display: flex">
  33. <div style="width: 80px; text-align: right;">访客次数</div>
  34. <div style="margin-left: 20px; text-align: right;" id="userSensitive">0</div>
  35. </div>
  36. <div style="display: flex">
  37. <div style="width: 80px; text-align: right;">客服次数</div>
  38. <div style="margin-left: 20px; text-align: right;" id="serverSensitive">0</div>
  39. </div>
  40. </div>
  41. </div>
  42. </div>
  43. <div class="col-sm-2">
  44. <div class="ibox float-e-margins">
  45. <div class="ibox-title" style="display: flex; justify-content: center;">
  46. <!--<span class="label label-primary pull-right">今天</span>-->
  47. <h3>平均响应超时</h3>
  48. </div>
  49. <div class="ibox-content" style="text-align: center">
  50. <h1 class="no-margins"><span id="csdTime">0</span><span style="font-size: 14px; margin-left: 10px">秒</span></h1>
  51. <small></small>
  52. </div>
  53. <div class="ibox-content" style="font-size: 12px; padding: 5px 20px 5px; height: 45px; line-height: 3;">
  54. <div style="display: flex">
  55. <div style="width: 80px; text-align: right;">响应超时次数</div>
  56. <div style="margin-left: 20px; text-align: right;" id="csdNumber">0</div>
  57. </div>
  58. </div>
  59. </div>
  60. </div>
  61. <div class="col-sm-2">
  62. <div class="ibox float-e-margins">
  63. <div class="ibox-title" style="display: flex; justify-content: center;">
  64. <!--<span class="label label-primary pull-right">今天</span>-->
  65. <h3>平均会话超时</h3>
  66. </div>
  67. <div class="ibox-content" style="text-align: center">
  68. <h1 class="no-margins"><span id="overtimeTime">0</span><span style="font-size: 14px; margin-left: 10px">秒</span></h1>
  69. <small></small>
  70. </div>
  71. <div class="ibox-content" style="font-size: 12px; padding: 5px 20px 5px; height: 45px; line-height: 3;">
  72. <div style="display: flex">
  73. <div style="width: 80px; text-align: right;">会话超时次数</div>
  74. <div style="margin-left: 20px; text-align: right;" id="overtimeNumber">0</div>
  75. </div>
  76. </div>
  77. </div>
  78. </div>
  79. <div class="col-sm-2">
  80. <div class="ibox float-e-margins">
  81. <div class="ibox-title" style="display: flex; justify-content: center;">
  82. <!--<span class="label label-primary pull-right">今天</span>-->
  83. <h3>满意度报警</h3>
  84. </div>
  85. <div class="ibox-content" style="text-align: center">
  86. <h1 class="no-margins"><span id="evaluateCount1">0</span><span style="font-size: 14px; margin-left: 10px">次</span></h1>
  87. <small></small>
  88. </div>
  89. <div class="ibox-content" style="font-size: 12px; padding: 5px 20px 5px; height: 45px; line-height: 3;">
  90. <div style="display: flex">
  91. <div style="width: 80px; text-align: right;">不满意次数</div>
  92. <div style="margin-left: 20px; text-align: right;" id="evaluateCount2">0</div>
  93. </div>
  94. </div>
  95. </div>
  96. </div>
  97. </div>
  98. <div class="ibox float-e-margins">
  99. <input style="display:none;" type="text" value="1" id="type">
  100. <div style="display: flex;">
  101. <div class="ibox-title" style="width: 100px; cursor:pointer;" id="current1" onclick="current(1)">
  102. <h5>当前会话</h5>
  103. </div>
  104. <div class="ibox-title" id="current2" style="width: 100px; cursor:pointer; background: #eee" onclick="current(2)">
  105. <h5>在线客服</h5>
  106. </div>
  107. <div class="ibox-title" style="width: calc(100% - 200px); background: #eee;">
  108. </div>
  109. </div>
  110. <div class="ibox-content">
  111. <!--搜索框开始-->
  112. <form id="realTimeForm" role="form" method="post" class="form-inline">
  113. <div class="content clearfix m-b">
  114. <div class="form-group">
  115. <label>分组名称:</label>
  116. <div class="input-group col-sm-4 layui-form" style="width: 120px;">
  117. <input type="hidden" id="group_id"/>
  118. <select lay-verify="required" lay-filter="group">
  119. <option value="">全部客服组</option>
  120. {if !empty($groups)}
  121. {foreach name="groups" item="vo"}
  122. <option value="{$vo['id']}">{$vo['name']}</option>
  123. {/foreach}
  124. {/if}
  125. </select>
  126. </div>
  127. </div>
  128. <div class="form-group" style="margin-left: 40px">
  129. <label>报警状态:</label>
  130. <div class="input-group col-sm-4 layui-form" style="width: 120px;">
  131. <input type="hidden" id="alarm_id"/>
  132. <select lay-verify="required" lay-filter="alarm">
  133. <option value="">所有会话</option>
  134. <option value="1">正常会话</option>
  135. <option value="2">报警会话</option>
  136. </select>
  137. </div>
  138. </div>
  139. </div>
  140. </form>
  141. <!--搜索框结束-->
  142. <div id="realTimeTable" class="example-wrap">
  143. <div class="example">
  144. <table class="table table-hover table-striped">
  145. <thead>
  146. <tr>
  147. <th>会话ID</th>
  148. <th>访客进线时间</th>
  149. <th>接待客服</th>
  150. <th>访客名</th>
  151. <th>会话开始时间</th>
  152. <th>报警次数</th>
  153. <th>操作</th>
  154. </tr>
  155. </thead>
  156. <tbody id="table">
  157. <tr>
  158. <td colspan="999" style="text-align: center">暂无数据</td>
  159. </tr>
  160. </tbody>
  161. </table>
  162. </div>
  163. </div>
  164. <!-- End Example Pagination -->
  165. <!--搜索框开始-->
  166. <form id='commentForm' style="display: none" role="form" method="get" class="form-inline">
  167. <div class="content clearfix m-b">
  168. <div class="form-group">
  169. <label>分组名称:</label>
  170. <div class="input-group col-sm-4 layui-form" style="width: 120px;">
  171. <select lay-verify="required" lay-filter="group" name="type1" id="type1">
  172. <option value="">全部客服组</option>
  173. {if !empty($groups)}
  174. {foreach name="groups" item="vo"}
  175. <option value="{$vo['id']}">{$vo['name']}</option>
  176. {/foreach}
  177. {/if}
  178. </select>
  179. </div>
  180. </div>
  181. <div class="form-group" style="margin-left: 40px">
  182. <label>报警状态:</label>
  183. <div class="input-group col-sm-4 layui-form" style="width: 120px;">
  184. <select lay-verify="required" lay-filter="alarm" name="type2" id="type2">
  185. <option value="">所有状态</option>
  186. <option value="1">在线状态</option>
  187. <option value="2">休息状态</option>
  188. <option value="3">隐身状态</option>
  189. </select>
  190. </div>
  191. </div>
  192. <div class="form-group" style="display: none">
  193. <lable>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</lable>
  194. <select name="type3" id="type3" class="form-control">
  195. <option value="0">所有会话</option>
  196. <option value="1">正常会话</option>
  197. <option value="2">报警回话</option>
  198. </select>
  199. </div>
  200. <div class="form-group">
  201. <lable>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;</lable>
  202. <button class="btn btn-primary" type="button" style="margin-top:5px" id="search"><strong>搜
  203. 索</strong>
  204. </button>
  205. </div>
  206. </div>
  207. </form>
  208. <!--搜索框结束-->
  209. <div id="commentTable" style="display: none" class="example-wrap">
  210. <div class="example">
  211. <table id="cusTable">
  212. <thead>
  213. <th data-field="uidname">客服</th>
  214. <th data-field="hhl">当前会话量</th>
  215. <th data-field="jdl">接待量</th>
  216. <th data-field="pjhysc">平均会话时长</th>
  217. <th data-field="cpl">参评率</th>
  218. <th data-field="mydl">满意度率</th>
  219. <th data-field="pjxysc">平均响应时长</th>
  220. <th data-field="jrxxsc">今日休息时长</th>
  221. <th data-field="fxx">发消息</th>
  222. </thead>
  223. </table>
  224. </div>
  225. </div>
  226. <!-- End Example Pagination -->
  227. </div>
  228. </div>
  229. </div>
  230. <script src="https://cdn.staticfile.org/jquery/2.1.4/jquery.min.js"></script>
  231. <script src="https://cdn.bootcss.com/bootstrap/3.3.6/js/bootstrap.min.js"></script>
  232. <script src="/static/admin/js/plugins/echarts/echarts.min.js"></script>
  233. <script src="/static/customer/js/md5.js"></script>
  234. <script src="__JS__/jquery.min.js?v=2.1.4"></script>
  235. <script src="__JS__/bootstrap.min.js?v=3.3.6"></script>
  236. <script src="__JS__/content.min.js?v=1.0.0"></script>
  237. <script src="__JS__/plugins/bootstrap-table/bootstrap-table.min.js"></script>
  238. <script src="__JS__/plugins/bootstrap-table/bootstrap-table-mobile.min.js"></script>
  239. <script src="__JS__/plugins/bootstrap-table/locale/bootstrap-table-zh-CN.min.js"></script>
  240. <script src="__JS__/plugins/layer/layer.min.js"></script>
  241. <script src="__JS__/plugins/validate/jquery.validate.min.js"></script>
  242. <script src="__JS__/plugins/validate/messages_zh.min.js"></script>
  243. <script src="__JS__/layui/layui.js"></script>
  244. <script src="__JS__/jquery.form.js"></script>
  245. <script type="text/javascript">
  246. function current(id) {
  247. $("#type").val(id);
  248. if (id == 1) {
  249. $("#current1").css("background","#fff");
  250. $("#current2").css("background","#eee");
  251. $("#realTimeForm").css("display","block");
  252. $("#realTimeTable").css("display","block");
  253. $("#commentForm").css("display","none");
  254. $("#commentTable").css("display","none");
  255. } else if (id == 2) {
  256. $("#current2").css({"background":"#fff"});
  257. $("#current1").css({"background":"#eee"});
  258. $("#realTimeForm").css({"display":"none"});
  259. $("#realTimeTable").css({"display":"none"});
  260. $("#commentForm").css({"display":"block"});
  261. $("#commentTable").css({"display":"block"});
  262. }
  263. initTable()
  264. }
  265. layui.use(['form', 'upload'], function(){
  266. var form = layui.form;
  267. form.on('select(group)', function(value){
  268. $("#group_id").val(value.value);
  269. onSearch()
  270. });
  271. form.on('select(alarm)', function(value){
  272. $("#alarm_id").val(value.value);
  273. onSearch()
  274. });
  275. });
  276. function initTable() {
  277. //先销毁表格
  278. $('#cusTable').bootstrapTable('destroy');
  279. //初始化表格,动态从服务器加载数据
  280. $("#cusTable").bootstrapTable({
  281. method: "get", //使用get请求到服务器获取数据
  282. url: "{:url('kfonitoring/kfjiankong')}", //获取数据的地址
  283. striped: true, //表格显示条纹
  284. pagination: true, //启动分页
  285. pageSize: 50, //每页显示的记录数
  286. pageNumber: 1, //当前第几页
  287. pageList: [5, 10, 15, 20, 25], //记录数可选列表
  288. sidePagination: "server", //表示服务端请求
  289. paginationFirstText: "首页",
  290. paginationPreText: "上一页",
  291. paginationNextText: "下一页",
  292. paginationLastText: "尾页",
  293. queryParamsType: "undefined",
  294. queryParams: function queryParams(params) { //设置查询参数
  295. var param = {
  296. pageNumber: params.pageNumber,
  297. pageSize: params.pageSize,
  298. searchText: $('#type1').val() + ',' + $('#type2').val() + ',' + $('#type3').val()
  299. };
  300. return param;
  301. },
  302. onLoadSuccess: function (res) { //加载成功时执行
  303. if (111 == res.code) {
  304. window.location.reload();
  305. }
  306. layer.msg("加载成功", {time: 1000});
  307. },
  308. onLoadError: function () { //加载失败时执行
  309. layer.msg("加载数据失败");
  310. }
  311. });
  312. }
  313. $(document).ready(function () {
  314. //调用函数,初始化表格
  315. initTable();
  316. //当点击查询按钮的时候执行
  317. $("#search").bind("click", initTable);
  318. });
  319. </script>
  320. <!--webSocket-->
  321. <script>
  322. let config = {
  323. socket: '{$socket}',
  324. token: '{$token}',
  325. };
  326. let myData = [];
  327. let date = new Date(new Date().setHours(0, 0, 0, 0)) / 1000;
  328. let getLocation = window.location.href;
  329. let host = getLocation.split('/admin')[0];
  330. let apiToken = hex_md5('customer-service'+date+host);
  331. let socket = new WebSocket('ws://' + config.socket+'?apiToken=' + apiToken);
  332. socket.onopen = function(res) {
  333. console.log('握手成功');
  334. // 登录
  335. let login_data = JSON.stringify({
  336. type: 'adminInit',
  337. token: config.token
  338. });
  339. socket.send(login_data);
  340. };
  341. socket.onmessage = function(res) {
  342. var data = eval("("+res.data+")");
  343. switch(data['message_type']){
  344. // 服务端ping客户端
  345. case 'monitor':
  346. myData = data.data.cvtList;
  347. putMonitor(data.data);
  348. break;
  349. }
  350. };
  351. /**
  352. * 时间戳转化为年 月 日 时 分 秒
  353. * time: 传入时间戳
  354. * format:返回格式,支持自定义,但参数必须与formateArr里保持一致
  355. */
  356. function customFormatDateTime(timeStamp,custom,isDtae) {
  357. var date = new Date();
  358. date.setTime(timeStamp * 1000);
  359. var y = date.getFullYear();
  360. var m = date.getMonth() + 1;
  361. m = m < 10 ? ('0' + m) : m;
  362. var d = date.getDate();
  363. d = d < 10 ? ('0' + d) : d;
  364. var h = date.getHours();
  365. h = h < 10 ? ('0' + h) : h;
  366. var minute = date.getMinutes();
  367. var second = date.getSeconds();
  368. minute = minute < 10 ? ('0' + minute) : minute;
  369. second = second < 10 ? ('0' + second) : second;
  370. if (isDtae == 0){
  371. return y + custom + m //+'student';
  372. }else if (isDtae ==1){
  373. return y + custom + m + custom + d;
  374. } else if (isDtae ==2) {
  375. return h + ':' + minute;
  376. } else {
  377. return y + '-' + m + '-' + d + '-' + h + ':' + minute + ':' + second;
  378. }
  379. };
  380. function onSearch() {
  381. let table = "";
  382. let cvtList = searchData();
  383. for(key in cvtList){
  384. let start_time = cvtList[key].start_time;
  385. let intime = cvtList[key].intime
  386. if (key != "length") {
  387. table += "<tr>" +
  388. "<td>"+cvtList[key].servicelog_id+"</td>" +
  389. "<td>"+customFormatDateTime(intime,'',2)+"</td>" +
  390. "<td>"+cvtList[key].server_name+"</td>" +
  391. "<td>"+cvtList[key].user_name+"</td>" +
  392. "<td>"+customFormatDateTime(start_time,'',2)+"</td>" +
  393. "<td>"+cvtList[key].allCount+"</td>" +
  394. "<td>" +
  395. "<a href='/admin/system/detail/id/"+cvtList[key].servicelog_id+".html'>" +
  396. "<button type='button' class='btn btn-primary btn-sm'>" +
  397. "<i class='fa fa-paste'></i> 详情" +
  398. "</button>" +
  399. "</a>" +
  400. "</td>" +
  401. "</tr>"
  402. }
  403. }
  404. if (table) {
  405. $("#table").html(table);
  406. } else {
  407. $("#table").html("<tr>" +
  408. "<td colspan='6' style='text-align: center'>暂无数据</td>" +
  409. "</tr>");
  410. }
  411. }
  412. function searchData() {
  413. let group_id = $("#group_id").val();
  414. let alarm_id = $("#alarm_id").val();
  415. let newData = [];
  416. if (group_id && alarm_id) {
  417. if (alarm_id == 1) {
  418. for (keys in myData) {
  419. if (group_id == myData[key].group_id && !myData[key].allCount) {
  420. newData.push(myData[key])
  421. }
  422. }
  423. } else if (alarm_id == 2) {
  424. for (keys in myData) {
  425. if (group_id == myData[key].group_id && myData[key].allCount) {
  426. newData.push(myData[key])
  427. }
  428. }
  429. }
  430. } else if(group_id) {
  431. for (keys in myData) {
  432. if (group_id == myData[key].group_id) {
  433. newData.push(myData[key])
  434. }
  435. }
  436. } else if(alarm_id) {
  437. if (alarm_id == 1) {
  438. for (keys in myData) {
  439. if (!myData[key].allCount) {
  440. newData.push(myData[key])
  441. }
  442. }
  443. } else if (alarm_id == 2) {
  444. for (keys in myData) {
  445. if (myData[key].allCount) {
  446. newData.push(myData[key])
  447. }
  448. }
  449. }
  450. } else {
  451. newData = myData
  452. }
  453. return newData;
  454. }
  455. function msg(id) {
  456. layui.use('layer', function () {
  457. layer.open({
  458. type: 2,
  459. title: '请输入消息内容',
  460. area: ['500px', '600px'], //自定义文本域宽高
  461. content:'/admin/kfnotice/add.html?uid='+id
  462. });
  463. }
  464. )
  465. }
  466. // 渲染
  467. function putMonitor(data) {
  468. $("#allSensitive").html(data.userSensitive + data.serverSensitive);
  469. $("#userSensitive").html(data.userSensitive);
  470. $("#serverSensitive").html(data.serverSensitive);
  471. $("#csdNumber").html(data.csdNumber);
  472. $("#overtimeNumber").html(data.overtimeNumber);
  473. $("#evaluateCount1").html(data.evaluateCount);
  474. $("#evaluateCount2").html(data.evaluateCount);
  475. // 响应超时.
  476. let n = data.csdTime.length;
  477. let csdTime = 0;
  478. for(keys in data.csdTime){
  479. csdTime += data.csdTime[keys]
  480. }
  481. csdTime = csdTime ? Math.ceil(csdTime/n) : 0;
  482. $("#csdTime").html(csdTime);
  483. // 会话超时.
  484. let m = data.overtimeTime.length;
  485. let overtimeTime = 0;
  486. for(keys in data.overtimeTime){
  487. overtimeTime += data.overtimeTime[keys]
  488. }
  489. overtimeTime = overtimeTime ? Math.ceil(overtimeTime/m) : 0;
  490. $("#overtimeTime").html(overtimeTime);
  491. onSearch();
  492. }
  493. </script>
  494. </body>
  495. </html>