msgbuoy.js 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505
  1. /**
  2. *
  3. * 浮漂msg消息公用方法
  4. * 插入dom节点方式运,为了兼容 三端,采用元素js 操作 dom + css
  5. * @param type
  6. * @returns {Object|*}
  7. */
  8. var getMsgBuoyActive = {
  9. // 版本号
  10. version: '0.0.2',
  11. platform: '', // work_bench、wx、app、pc
  12. msgids: '',
  13. floatDOM: '',
  14. messageList: [],
  15. messageInfo: {},
  16. textmas: '剑鱼标讯-招标信息查询_全国招标采购信息订阅推送_专业招标大数据服务平台',
  17. init: function () {
  18. // 初始化平台计算
  19. this.getPlatform()
  20. // 限制初始化页面
  21. if (this.platform === 'app' || this.platform === 'wx') {
  22. if (this.checkNowInMobileHomePage()) {
  23. this.getBuoyMsgAjax()
  24. }
  25. } else {
  26. this.listenNotification()
  27. this.getBuoyMsgAjax()
  28. }
  29. },
  30. /**
  31. * 检查当前是否是 移动端首页 页面
  32. * @returns {boolean}
  33. */
  34. checkNowInMobileHomePage () {
  35. var Pages = [
  36. '/jyapp/jylab/mainSearch',
  37. '/jylab/mainSearch',
  38. '/jy_mobile/tabbar/home'
  39. ]
  40. return Pages.some(function (v) {
  41. return location.href.indexOf(v) > -1
  42. })
  43. },
  44. getPlatform: function () {
  45. this.platform = this.platformOS()
  46. },
  47. initDOMNode: function () {
  48. // 判断是否存在,需要先移除dom
  49. var oldNode = $('.msggroup')
  50. if (oldNode && typeof oldNode.remove === 'function') {
  51. oldNode.remove()
  52. }
  53. // 没有消息时退出执行
  54. if (!(this.messageList && this.messageList.length)) {
  55. return
  56. }
  57. var message = this.messageList[0]
  58. var textmas = this.textmas
  59. if (message) {
  60. textmas = message.buoyDetail
  61. this.msgids = message.id
  62. }
  63. let scanHTML = `<div class="msggroup" style="display:none;">
  64. <div class="msgbox">
  65. <div id="msg_content" class="msg_content">
  66. <span class="text-content">${textmas}</span>
  67. </div>
  68. </div>
  69. <div class="msg_close">
  70. </div>
  71. </div>`
  72. var scanDOM = $(scanHTML)
  73. this.floatDOM = scanDOM.show()
  74. scanDOM.find('#msg_content').html(`<span class="text-content">${textmas}</span>`)
  75. if (this.platform === 'pc') {
  76. $('#public-nav .iner').append(scanDOM)
  77. } else if (this.platform === 'work_bench') {
  78. // 插入节点
  79. $('.el-header > .navbar-group').append(scanDOM)
  80. } else {
  81. $('body').append(scanDOM)
  82. }
  83. // 初始化样式表
  84. this.createStyle()
  85. // 初始化监听事件
  86. this.addEventClick()
  87. // 初始化样式表style
  88. this.scroll()
  89. },
  90. getBuoyMsgAjax: function () {
  91. var _this = this
  92. $.ajax({
  93. url: '/jyapi/messageCenter/GetBuoyMsg',
  94. type: 'POST',
  95. }).done(function (res) {
  96. if (res && res.error_code === 0 && res.data) {
  97. try {
  98. _this.messageList = res.data
  99. if (res.data && res.data.length) {
  100. _this.messageInfo = _this.messageList[0]
  101. _this.initDOMNode()
  102. }
  103. } catch (error) {
  104. console.warn(error)
  105. }
  106. }
  107. })
  108. },
  109. istoReaded: function (ids, url) {
  110. // 消息盒子 消失
  111. // 效果彩蛋 ~~ 1200ms逐渐消失
  112. $(".msggroup").fadeOut(1200);
  113. $(".msggroup").fadeTo("1200", 0.6);
  114. $.ajax({
  115. type: 'POST',
  116. url: '/jymessageCenter/markRead',
  117. async: false,
  118. data: {
  119. msgId: ids
  120. },
  121. success: function (res) {
  122. if (res.error_code === 0 && res.data) {
  123. console.log(res);
  124. }
  125. }
  126. })
  127. },
  128. // 监听点击事件 行为
  129. addEventClick: function (praume) {
  130. var _this = this
  131. // 直接使用选择器进行绑定监听
  132. this.floatDOM.find('.msg_content').on('click', function () {
  133. _this.floatDOM.find('.msg_close').trigger('click')
  134. const ua = window.navigator.userAgent.toLocaleLowerCase()
  135. const isIos = /iphone|ipad|ipod/.test(ua)
  136. const isAndroid = /android/.test(ua)
  137. try {
  138. // 跳转链接
  139. switch (getMsgBuoyActive.platformOS()) {
  140. case 'pc':
  141. location.href = _this.messageInfo.pcUrl ? _this.messageInfo.pcUrl : '/swordfish/frontPage/messageCenter/sess/index'
  142. break;
  143. case 'work_bench':
  144. window.open(_this.messageInfo.pcUrl ? _this.messageInfo.pcUrl : '/swordfish/frontPage/messageCenter/sess/index')
  145. break;
  146. case 'wx': {
  147. location.href = _this.messageInfo.weChatUrl ? _this.messageInfo.weChatUrl :'/weixin/frontPage/messageCenter/sess/index'
  148. break;
  149. }
  150. case 'app': {
  151. if(isIos && _this.messageInfo.iosUrl){
  152. location.href = _this.messageInfo.iosUrl
  153. } else if(isAndroid && _this.messageInfo.androidUrl) {
  154. location.href = _this.messageInfo.androidUrl
  155. } else {
  156. _this.isAppH5Jump()
  157. }
  158. break;
  159. }
  160. default:
  161. break;
  162. }
  163. } catch (e) {
  164. console.log(e);
  165. }
  166. })
  167. // 关闭消息
  168. this.floatDOM.find('.msg_close').on('click', function (e) {
  169. getMsgBuoyActive.istoReaded(getMsgBuoyActive.msgids)
  170. })
  171. },
  172. isAppH5Jump: function(){
  173. // H5 环境支持跳转
  174. if (window.utilsEnv && window.utilsEnv.platformEnvs.platform === 'h5') {
  175. location.href = '/jy_mobile/tabbar/message'
  176. } else {
  177. try {
  178. // app跳转到消息中心tab
  179. JyObj.skipAppointTab('message')
  180. } catch (error) {
  181. console.log('error')
  182. }
  183. }
  184. },
  185. scroll: function () {
  186. // 可视窗口宽度 \
  187. let windowWidth = document.body.offsetWidth, Pc_Topstyle
  188. var floatDOM = this.floatDOM
  189. // 设置每个弹框相对位置定位style - 根据平台
  190. // pc 和移动端 定位盒子
  191. // Pc_Topstyle = getMsgBuoyActive.platformOS() === 'pc'
  192. // ? `width:315px;right:${Number(windowWidth - 1200)/2}px;top:80px;`
  193. // : `width:92%;right: 4%; bottom: 17vw;`
  194. switch (getMsgBuoyActive.platformOS()) {
  195. case 'pc':
  196. floatDOM.addClass('message-float-pc')
  197. break;
  198. case 'work_bench':
  199. Pc_Topstyle = `width:315px;right:186px;top:80px;`
  200. break;
  201. case 'app':
  202. floatDOM.addClass('message-float-mobile')
  203. Pc_Topstyle = `bottom: 8vw;`
  204. break;
  205. case 'wx':
  206. floatDOM.addClass('message-float-mobile')
  207. Pc_Topstyle = `bottom: calc(8vw + 9.33vw);`
  208. break;
  209. default:
  210. break;
  211. }
  212. floatDOM.show()
  213. floatDOM[0].style = `display:block;${Pc_Topstyle}`
  214. if (getMsgBuoyActive.platformOS() === 'app') {
  215. floatDOM.find('.msg_close')[0].style = `background: #000;`
  216. }
  217. var content = floatDOM.find('.msg_content')[0],
  218. box = floatDOM[0],
  219. contentWidth = content.offsetWidth,
  220. boxWidth = box.offsetWidth
  221. box.style.setProperty('animation', '5s')
  222. var _this = this
  223. if (!contentWidth || !boxWidth) {
  224. setTimeout(function () {
  225. _this.scroll()
  226. }, 200)
  227. return
  228. }
  229. if (contentWidth <= boxWidth) {
  230. // 内容宽度小于盒子宽度,停止滚动
  231. content.style.setProperty('animation', '5s');
  232. } else {
  233. var computedContentWidth = contentWidth * 2 + 20
  234. // 根据宽度设置滚动距离和动画时长,复制实现无缝滚动
  235. content.style.setProperty('animation', 'move ' + computedContentWidth / 50 + 's linear infinite');
  236. $(content).append($(".msg_content .text-content").clone())
  237. // // 修改@keyframes的值
  238. // const frame = `@Keyframes move {
  239. // from {
  240. // transform: translate(0);
  241. // }
  242. // to {
  243. // transform: translateX(-${contentWidth - boxWidth}px)
  244. // }
  245. // }`;
  246. // // 找到对应的css样式表,先删除再新增
  247. // let sheets = document.styleSheets;
  248. // for (let i = 0; i < sheets.length; ++i) {
  249. // const item = sheets[i];
  250. // if (item.cssRules[0] && item.cssRules[0].name && item.cssRules[0].name === 'move') {
  251. // item.deleteRule(0);
  252. // item.insertRule(frame, 0)
  253. // }
  254. // }
  255. }
  256. },
  257. // 动态创建样式==
  258. createStyle: function () {
  259. var frame = `
  260. @Keyframes move {
  261. from {
  262. transform: translate(0);
  263. }
  264. to {
  265. transform: translateX(-50%)
  266. }
  267. }
  268. .msggroup {
  269. position: absolute;
  270. width: 351px;
  271. height: 44px;
  272. right: 3%;
  273. background: linear-gradient(180deg, #E8FFFF 0%, #FFFFFF 100%);
  274. box-shadow: 0px 4px 16px rgb(8 31 38 / 8%);
  275. border-radius: 8px;
  276. border: 0.5px solid;
  277. border-image-source: linear-gradient(163deg, #1a1c1d 0%, rgba(42, 190, 209, 0) 100%);
  278. z-index: 7;
  279. line-height: 44px;
  280. font-size: 13px;
  281. padding: 0 5px;
  282. background-size: 100% 100%;
  283. background-image: url();
  284. background-repeat: no-repeat;
  285. }
  286. .msgbox {
  287. width: 100%;
  288. overflow: hidden;
  289. white-space: nowrap;
  290. }
  291. .msg_content {
  292. animation: move 5s linear infinite;
  293. display: inline-block;
  294. cursor: pointer;
  295. }
  296. .msg_content .text-content {
  297. padding: 0 10px;
  298. }
  299. .msg_close {
  300. position: absolute;
  301. width: 18px;
  302. height: 18px;
  303. right: -6px;
  304. top: -10px;
  305. cursor: pointer;
  306. background: #2ABED1;
  307. opacity: 0.8;
  308. /*background:url('/images/close_pc_msg.png') 0 0;
  309. background-size: cover;*/
  310. -moz-border-radius: 50px;
  311. -webkit-border-radius: 50px;
  312. border-radius: 50px;
  313. }
  314. .msg_close img {
  315. width:100%;
  316. height:100%;
  317. }
  318. .msg_close:before,
  319. .msg_close:after {
  320. content: "";
  321. position: absolute;
  322. height: 1.3px;
  323. width: 65%;
  324. top: 50%;
  325. left: 20%;
  326. margin-top: -1px;
  327. background: #fff;
  328. }
  329. .msg_close:before {
  330. -webkit-transform: rotate(45deg);
  331. -moz-transform: rotate(45deg);
  332. -ms-transform: rotate(45deg);
  333. -o-transform: rotate(45deg);
  334. transform: rotate(45deg);
  335. }
  336. .msg_close:after {
  337. -webkit-transform: rotate(-45deg);
  338. -moz-transform: rotate(-45deg);
  339. -ms-transform: rotate(-45deg);
  340. -o-transform: rotate(-45deg);
  341. transform: rotate(-45deg);
  342. }
  343. `
  344. frame += this.addPlatformStyle()
  345. // 创建style标签
  346. const style = document.createElement('style');
  347. style.type = 'text/css';
  348. style.innerHTML = frame;
  349. document.getElementsByTagName('head')[0].appendChild(style);
  350. },
  351. addPlatformStyle: function () {
  352. var text = ''
  353. if (this.platform === 'pc') {
  354. text = `
  355. .msggroup.message-float-pc {
  356. right: 0;
  357. bottom: -10px;
  358. transform: translate(0, 100%);
  359. }
  360. `
  361. } else if (this.platform === 'wx' || this.platform === 'app') {
  362. text = `
  363. .msggroup.message-float-mobile {
  364. position: fixed;
  365. width: calc(100% - 8vw);
  366. /* width:315px; */
  367. left: 50%;
  368. bottom: 0;
  369. z-index: 9;
  370. transform: translate(-50%, 0);
  371. }
  372. `
  373. }
  374. return text
  375. },
  376. // 判断平台环境,用于定位样式
  377. platformOS: function () {
  378. let ua = navigator.userAgent || window.navigator.userAgent,
  379. IsWeChatBrowser = ua.toLowerCase().indexOf('micromessenger') !== -1,
  380. androidOrIOS = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(ua),
  381. isAndroid = /(?:Android)/.test(ua),
  382. isPad = /(?:iPad|PlayBook)/.test(ua) || (isAndroid && !/(?:Mobile)/.test(ua)),
  383. isPhone = /(?:iPhone)/.test(ua) && !isPad,
  384. isPc = !isPhone && !isAndroid && !IsWeChatBrowser && !androidOrIOS,
  385. iswork_bench = /work-bench|dashboard|workspace/i.test(location.href)
  386. if (IsWeChatBrowser) {
  387. return 'wx'
  388. } else if (iswork_bench) {
  389. return 'work_bench'
  390. } else if (androidOrIOS) {
  391. return 'app'
  392. } else if (isPc) {
  393. return 'pc'
  394. } else if (isPhone) {
  395. return 'ios'
  396. } else if (isAndroid) {
  397. return 'Android'
  398. }
  399. },
  400. /**
  401. * @date 2023/1/4 消息通知改为 webscoket 获取并调用浏览器消息通知
  402. */
  403. listenNotification () {
  404. /**
  405. * 进行浏览器通知,判断上次通知是否与当前一致,一致则不进行重复通知
  406. * @param datas - 信息数据
  407. */
  408. function sendNotification (datas) {
  409. var waitShowMessageId = localStorage.getItem('notification-login-clear-info') || -1
  410. if (waitShowMessageId === datas.id) {
  411. return console.warn('重复消息ID,不进行重复通知')
  412. }
  413. try {
  414. Notification.requestPermission(function(status) {
  415. var notifyStatus = Notification.permission || status
  416. if(notifyStatus === 'granted'){
  417. localStorage.setItem('notification-login-clear-info', datas.id)
  418. var notify = new Notification(datas.title,{
  419. icon: '/images/t3_new.jpg',
  420. body: datas.show_content || datas.content || ''
  421. })
  422. // 点击时桌面消息时触发
  423. notify.onclick = function () {
  424. try {
  425. getMsgBuoyActive.istoReaded(datas.id)
  426. notify.close()
  427. switch (getMsgBuoyActive.platformOS()) {
  428. case 'pc':
  429. location.href = '/swordfish/frontPage/messageCenter/sess/index'
  430. break;
  431. case 'work_bench':
  432. window.open('/swordfish/frontPage/messageCenter/sess/index')
  433. break;
  434. }
  435. } catch (e) {
  436. console.warn(e)
  437. }
  438. }
  439. } else {
  440. Notification.requestPermission(function () {
  441. sendNotification(datas)
  442. })
  443. }
  444. })
  445. } catch (e) {
  446. console.warn('不支持通知')
  447. }
  448. }
  449. // 定义 scoket 信息
  450. var pushMessageScoket = {
  451. url: "ws"+(!isIE9&&"https:"==document.location.protocol?"s":"")+"://"+window.location.host,
  452. init: function () {
  453. if (typeof CommonWebScoket === 'function') {
  454. this._scoket = new CommonWebScoket(this.url + '/webscoket/getBuoyMsg', '', {
  455. ajaxReq: function() {},
  456. reconnect: function (commom) {
  457. commom.isOver = true
  458. commom.over()
  459. },
  460. complete: function (data, type) {
  461. try {
  462. var pushMessage = JSON.parse(data)
  463. if (pushMessage.id) {
  464. sendNotification(pushMessage)
  465. }
  466. } catch (e) {
  467. console.warn(e)
  468. }
  469. }
  470. }).init()
  471. }
  472. }
  473. }
  474. // 初始化
  475. pushMessageScoket.init()
  476. }
  477. }
  478. // 初始化
  479. $(function () {
  480. try {
  481. // 尝试判断是否在iframe中
  482. if (!goTemplateData.inIframe) {
  483. getMsgBuoyActive.init()
  484. }
  485. } catch (error) {
  486. getMsgBuoyActive.init()
  487. }
  488. })