App.vue 46 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757
  1. <template>
  2. <div id="app">
  3. <div class="scroll-top" @click="scrollToTop">🚀</div>
  4. <div class="logo">
  5. <div class="logo-text">
  6. <b>Lemon</b> IMUI<span class="logo-badge">{{
  7. this.packageData.version
  8. }}</span>
  9. </div>
  10. <div class="logo-sub">{{ this.packageData.description }}</div>
  11. <div class="link">
  12. <span>源码下载&nbsp;&nbsp;</span>
  13. <a target="_blank" href="https://github.com/fanjyy/lemon-imui"
  14. >Github</a
  15. >
  16. <a target="_blank" href="https://gitee.com/june000/lemon-im">Gitee</a>
  17. <a
  18. target="_blank"
  19. href="https://qm.qq.com/cgi-bin/qm/qr?k=xzUa9CPYQ5KCNQ86h7ep4Z3TtkqJxRZE&jump_from=webapi"
  20. >QQ交流群:1081773406</a
  21. >
  22. </div>
  23. <br />
  24. <div><a style="font-size:14px;" href="#help1">1.如何创建消息?</a></div>
  25. <div>
  26. <a style="font-size:14px;" href="#help2">2.如何对接后端接口?</a>
  27. </div>
  28. </div>
  29. <div class="imui-center">
  30. <lemon-imui
  31. :user="user"
  32. ref="IMUI"
  33. :contextmenu="contextmenu"
  34. :contact-contextmenu="contactContextmenu"
  35. :theme="theme"
  36. :hide-menu="hideMenu"
  37. :hide-menu-avatar="hideMenuAvatar"
  38. :hide-message-name="hideMessageName"
  39. :hide-message-time="hideMessageTime"
  40. @change-menu="handleChangeMenu"
  41. @change-contact="handleChangeContact"
  42. @pull-messages="handlePullMessages"
  43. @message-click="handleMessageClick"
  44. @menu-avatar-click="handleMenuAvatarClick"
  45. @send="handleSend"
  46. >
  47. <template #cover>
  48. <div class="cover">
  49. <i class="lemon-icon-message"></i>
  50. <p><b>自定义封面 Lemon</b> IMUI</p>
  51. </div>
  52. </template>
  53. <template #message-title="contact">
  54. <span>{{ contact.displayName }}</span>
  55. <small class="more" @click="changeDrawer(contact, $refs.IMUI)"
  56. >{{
  57. ($refs.IMUI ? $refs.IMUI.drawerVisible : false) ? "关闭" : "打开"
  58. }}抽屉</small
  59. >
  60. <br />
  61. </template>
  62. </lemon-imui>
  63. <a
  64. target="_blank"
  65. style="font-size:14px"
  66. href="https://codesandbox.io/s/sweet-chaplygin-s24mb?fontsize=14&hidenavigation=1&theme=dark"
  67. >在线编辑代码</a
  68. >
  69. <div class="action">
  70. <lemon-button @click="appendMessage">发送消息</lemon-button>
  71. <lemon-button @click="appendEventMessage">发送 event 消息</lemon-button>
  72. <lemon-button @click="removeMessage">删除最近一条消息</lemon-button>
  73. <lemon-button @click="updateMessage">修改消息</lemon-button>
  74. <lemon-button @click="appendCustomMessage">发送消息</lemon-button>
  75. <br />
  76. <lemon-button @click="updateContact">修改联系人信息</lemon-button>
  77. <lemon-button @click="changeMenuVisible">切换导航显示</lemon-button>
  78. <lemon-button @click="changeMenuAvatarVisible"
  79. >切换头像显示</lemon-button
  80. >
  81. <lemon-button @click="changeMessageNameVisible"
  82. >切换聊天窗口内名字显示</lemon-button
  83. >
  84. <lemon-button @click="changeMessageTimeVisible"
  85. >切换聊天窗口内时间显示</lemon-button
  86. >
  87. <lemon-button @click="changeTheme"
  88. >切换主题,当前主题:{{ this.theme }}</lemon-button
  89. >
  90. </div>
  91. </div>
  92. <div style="display:flex;">
  93. <div>
  94. <div class="title">自定义</div>
  95. <div class="imui-center"><qq-imui>12312312</qq-imui></div>
  96. </div>
  97. <div style="margin:0 55px;">
  98. <div class="title">精简模式</div>
  99. <div class="imui-center">
  100. <lemon-imui
  101. class="lemon-simple"
  102. :user="user"
  103. ref="SimpleIMUI"
  104. width="340px"
  105. :avatar-cricle="true"
  106. simple
  107. @pull-messages="handlePullMessages"
  108. @message-click="handleMessageClick"
  109. @send="handleSend"
  110. ></lemon-imui>
  111. <a
  112. target="_blank"
  113. style="font-size:14px"
  114. href="https://codesandbox.io/s/lemon-imui-jingjianmoshi-forked-1lvoh?fontsize=14&hidenavigation=1&theme=dark"
  115. >在线编辑代码</a
  116. >
  117. </div>
  118. </div>
  119. </div>
  120. <div class="title">联系人 Contact</div>
  121. <table class="table">
  122. <tr class="table-head">
  123. <th>参数</th>
  124. <th>说明</th>
  125. <th>类型</th>
  126. <th>默认值</th>
  127. <th>示例</th>
  128. </tr>
  129. <tr>
  130. <td width="150">id</td>
  131. <td width="350">唯一ID</td>
  132. <td width="150">String/Number</td>
  133. <td width="100">-</td>
  134. <td></td>
  135. </tr>
  136. <tr>
  137. <td>displayName</td>
  138. <td>名称</td>
  139. <td>String</td>
  140. <td>-</td>
  141. <td></td>
  142. </tr>
  143. <tr></tr>
  144. <tr>
  145. <td>avatar</td>
  146. <td>头像</td>
  147. <td>String</td>
  148. <td>-</td>
  149. <td></td>
  150. </tr>
  151. <tr>
  152. <td>index</td>
  153. <td>
  154. 通讯录索引,传入字母或数字进行排序,索引可以显示自定义文字“[1]群组”
  155. </td>
  156. <td>String</td>
  157. <td>-</td>
  158. <td></td>
  159. </tr>
  160. <tr>
  161. <td>unread</td>
  162. <td>未读消息数</td>
  163. <td>Number</td>
  164. <td>0</td>
  165. <td></td>
  166. </tr>
  167. <tr>
  168. <td>lastSendTime</td>
  169. <td>最近一条消息的时间戳,13位毫秒</td>
  170. <td>timestamp</td>
  171. <td>0</td>
  172. <td></td>
  173. </tr>
  174. <tr>
  175. <td>lastContent</td>
  176. <td>最近一条消息的内容</td>
  177. <td>String | Vnode</td>
  178. <td></td>
  179. <td></td>
  180. </tr>
  181. </table>
  182. <div class="title">消息体 Message</div>
  183. <table class="table">
  184. <tr class="table-head">
  185. <th>参数</th>
  186. <th>说明</th>
  187. <th>类型</th>
  188. <th>默认值</th>
  189. <th>示例</th>
  190. </tr>
  191. <tr>
  192. <td width="150">id</td>
  193. <td width="350">唯一ID</td>
  194. <td width="150">String/Number</td>
  195. <td width="100">-</td>
  196. <td></td>
  197. </tr>
  198. <tr>
  199. <td>status</td>
  200. <td>消息发送的状态:going | failed | succeed</td>
  201. <td>String</td>
  202. <td>-</td>
  203. <td></td>
  204. </tr>
  205. <tr></tr>
  206. <tr>
  207. <td>type</td>
  208. <td>消息类型:file | image | text | event</td>
  209. <td>String | Vnode</td>
  210. <td>-</td>
  211. <td></td>
  212. </tr>
  213. <tr>
  214. <td>sendTime</td>
  215. <td>消息发送时间,13位毫秒</td>
  216. <td>timestamp</td>
  217. <td>-</td>
  218. <td></td>
  219. </tr>
  220. <tr>
  221. <td>content</td>
  222. <td>消息内容,如果type=file,此属性表示文件的URL地址</td>
  223. <td>String</td>
  224. <td>-</td>
  225. <td></td>
  226. </tr>
  227. <tr>
  228. <td>fileSize</td>
  229. <td>文件大小</td>
  230. <td>Number</td>
  231. <td>0</td>
  232. <td></td>
  233. </tr>
  234. <tr>
  235. <td>fileName</td>
  236. <td>文件名称</td>
  237. <td>String</td>
  238. <td>-</td>
  239. <td></td>
  240. </tr>
  241. <tr>
  242. <td>toContactId</td>
  243. <td>接收消息的联系人ID</td>
  244. <td>String | Number</td>
  245. <td>-</td>
  246. <td></td>
  247. </tr>
  248. <tr>
  249. <td>fromUser</td>
  250. <td>消息发送人的信息</td>
  251. <td>Object</td>
  252. <td>-</td>
  253. <td>{id: "1",displayName: "测试",avatar: "url"};</td>
  254. </tr>
  255. </table>
  256. <div class="title">右键菜单 ContextmenuItem</div>
  257. <table class="table">
  258. <tr class="table-head">
  259. <th>名称</th>
  260. <th>说明</th>
  261. <th>类型</th>
  262. <th>示例</th>
  263. </tr>
  264. <tr>
  265. <td width="150">text</td>
  266. <td width="350">显示文字</td>
  267. <td width="150">String</td>
  268. <td width="100">-</td>
  269. <td></td>
  270. </tr>
  271. <tr>
  272. <td width="150">color</td>
  273. <td width="350">颜色</td>
  274. <td width="150">String</td>
  275. <td width="100">-</td>
  276. <td></td>
  277. </tr>
  278. <tr>
  279. <td width="150">icon</td>
  280. <td width="350">图标 class</td>
  281. <td width="150">String</td>
  282. <td width="100">-</td>
  283. <td></td>
  284. </tr>
  285. <tr>
  286. <td width="150">click</td>
  287. <td width="350">点击事件,调用hide方法隐藏右键菜单。</td>
  288. <td width="150">Function(e,instance,hide)</td>
  289. <td width="100">-</td>
  290. <td></td>
  291. </tr>
  292. <tr>
  293. <td width="150">visible</td>
  294. <td width="350">是否显示的判断函数</td>
  295. <td width="150">Function(instance)</td>
  296. <td width="100">-</td>
  297. <td></td>
  298. </tr>
  299. <tr>
  300. <td width="150">render</td>
  301. <td width="350">
  302. 负责样式的渲染函数,使用render的时候text属性会失去作用,调用hide方法隐藏右键菜单。
  303. </td>
  304. <td width="150">Function(e,instance,hide)</td>
  305. <td width="100">-</td>
  306. <td></td>
  307. </tr>
  308. </table>
  309. <div class="title">抽屉配置 DrawerOption</div>
  310. <table class="table">
  311. <tr class="table-head">
  312. <th>名称</th>
  313. <th>说明</th>
  314. <th>类型</th>
  315. <th>示例</th>
  316. </tr>
  317. <tr>
  318. <td width="150">width</td>
  319. <td width="350">宽度,可以设置百分比</td>
  320. <td width="150">String | Number</td>
  321. <td width="100">-</td>
  322. <td></td>
  323. </tr>
  324. <tr>
  325. <td width="150">height</td>
  326. <td width="350">高度,可以设置百分比</td>
  327. <td width="150">String | Number</td>
  328. <td width="100">-</td>
  329. <td></td>
  330. </tr>
  331. <tr>
  332. <td width="150">offsetX</td>
  333. <td width="350">X偏移值,可以设置百分比</td>
  334. <td width="150">String | Number</td>
  335. <td width="100">-</td>
  336. <td></td>
  337. </tr>
  338. <tr>
  339. <td width="150">offsetY</td>
  340. <td width="350">Y偏移值,可以设置百分比</td>
  341. <td width="150">String | Number</td>
  342. <td width="100">-</td>
  343. <td></td>
  344. </tr>
  345. <tr>
  346. <td width="150">position</td>
  347. <td width="350">位置</td>
  348. <td width="150">right | rightInside | center</td>
  349. <td width="100">-</td>
  350. <td></td>
  351. </tr>
  352. </table>
  353. <div class="title">组件属性</div>
  354. <table class="table">
  355. <tr class="table-head">
  356. <th>参数</th>
  357. <th>说明</th>
  358. <th>类型</th>
  359. <th>默认值</th>
  360. <th>示例</th>
  361. </tr>
  362. <tr>
  363. <td width="150">user</td>
  364. <td width="350">用户信息</td>
  365. <td width="150">Object</td>
  366. <td width="100">-</td>
  367. <td>{id: "1",displayName: "测试",avatar: "url"};</td>
  368. </tr>
  369. <tr>
  370. <td width="150">width</td>
  371. <td width="350">宽度</td>
  372. <td width="150">String</td>
  373. <td width="100">850px</td>
  374. <td></td>
  375. </tr>
  376. <tr>
  377. <td width="150">height</td>
  378. <td width="350">高度</td>
  379. <td width="150">String</td>
  380. <td width="100">580px</td>
  381. <td></td>
  382. </tr>
  383. <tr>
  384. <td width="150">theme</td>
  385. <td width="350">主题</td>
  386. <td width="150">default | blue</td>
  387. <td width="100">default</td>
  388. <td></td>
  389. </tr>
  390. <tr>
  391. <td width="150">avatarCricle</td>
  392. <td width="350">使用圆形头像</td>
  393. <td width="150">Boolean</td>
  394. <td width="100">false</td>
  395. <td></td>
  396. </tr>
  397. <tr>
  398. <td width="150">sendText</td>
  399. <td width="350">发送消息按钮的文字</td>
  400. <td width="150">String</td>
  401. <td width="100">发送消息</td>
  402. <td></td>
  403. </tr>
  404. <tr>
  405. <td width="150">sendKey</td>
  406. <td width="350">快捷发送键检查函数</td>
  407. <td width="150">Function(event)=>Boolean</td>
  408. <td width="100"></td>
  409. <td>(e)=>e.keyCode == 13 && e.ctrlKey</td>
  410. </tr>
  411. <tr>
  412. <td width="150">simple</td>
  413. <td width="350">精简模式</td>
  414. <td width="150">Boolean</td>
  415. <td width="100">false</td>
  416. <td>
  417. 精简模式下左侧的导航和联系人列表会隐藏,初始化时需要手动调用
  418. changeContact 切换到聊天视图。
  419. </td>
  420. </tr>
  421. <tr>
  422. <td>messageTimeFormat</td>
  423. <td>消息列表时间格式化函数</td>
  424. <td>Function(time)=>String</td>
  425. <td>-</td>
  426. <td></td>
  427. </tr>
  428. <tr>
  429. <td>contactTimeFormat</td>
  430. <td>联系人时间格式化规则</td>
  431. <td>Function(time)=>String</td>
  432. <td>-</td>
  433. <td></td>
  434. </tr>
  435. <tr>
  436. <td>hideDrawer</td>
  437. <td>是否隐藏抽屉</td>
  438. <td>Boolean</td>
  439. <td>true</td>
  440. <td></td>
  441. </tr>
  442. <tr>
  443. <td>hideMenuAvatar</td>
  444. <td>是否隐藏导航头像</td>
  445. <td>Boolean</td>
  446. <td>false</td>
  447. <td></td>
  448. </tr>
  449. <tr>
  450. <td>hideMenu</td>
  451. <td>是否隐藏左侧导航</td>
  452. <td>Boolean</td>
  453. <td>false</td>
  454. <td></td>
  455. </tr>
  456. <tr>
  457. <td>hideMessageName</td>
  458. <td>是否隐藏聊天窗口内的联系人名字</td>
  459. <td>Boolean</td>
  460. <td>false</td>
  461. <td></td>
  462. </tr>
  463. <tr>
  464. <td>hideMessageTime</td>
  465. <td>是否隐藏聊天窗口内的消息发送时间</td>
  466. <td>Boolean</td>
  467. <td>false</td>
  468. <td></td>
  469. </tr>
  470. <tr>
  471. <td>contextmenu</td>
  472. <td>聊天消息右键菜单配置</td>
  473. <td>[ContextmenuItem]</td>
  474. <td>-</td>
  475. <td></td>
  476. </tr>
  477. <tr>
  478. <td>contactContextmenu</td>
  479. <td>联系人右键菜单配置</td>
  480. <td>[ContextmenuItem]</td>
  481. <td>-</td>
  482. <td></td>
  483. </tr>
  484. </table>
  485. <div class="title">组件方法</div>
  486. <table class="table">
  487. <tr class="table-head">
  488. <th>参数</th>
  489. <th>说明</th>
  490. <th>类型</th>
  491. <th>默认值</th>
  492. <th>示例</th>
  493. </tr>
  494. <tr>
  495. <td width="150">initMenus</td>
  496. <td width="350">初始化导航</td>
  497. <td width="150">Function([Object])</td>
  498. <td width="100">[ { name: "messages" }, { name: "contacts" }]</td>
  499. <td>
  500. { name: "custom2", title: "自定义按钮2", unread: 0, click: () => {
  501. alert("拦截导航点击事件"); }, render: menu => { return '...'; },
  502. isBottom: true }
  503. </td>
  504. </tr>
  505. <tr>
  506. <td>initContacts</td>
  507. <td>初始化联系人</td>
  508. <td>Function([Contact])</td>
  509. <td>-</td>
  510. <td></td>
  511. </tr>
  512. <tr>
  513. <td>initEditorTools</td>
  514. <td>初始化工具栏</td>
  515. <td>Function([Object])</td>
  516. <td>[{name:'emoji'},{name:'uploadFile'},{name:'uploadImage'}]</td>
  517. <td>
  518. [{ name:"test2", isRight:true, title:'上传 Excel', click:()=>{
  519. alert('点击') }, render:()=>{ return '...' } }]
  520. </td>
  521. </tr>
  522. <tr>
  523. <td>initEmoji</td>
  524. <td>初始化表情数据</td>
  525. <td>Function([Object])</td>
  526. <td>-</td>
  527. <td>
  528. <div>
  529. 有分类:[{ label: '默认表情', children: [ { name: '1f62c', title:
  530. '微笑', src: 'https://twemoji.maxcdn.com/2/72x72/1f62c.png' } ] }]
  531. </div>
  532. <div>
  533. 无分类:[{ name: '1f62c', title: '微笑', src:
  534. 'https://twemoji.maxcdn.com/2/72x72/1f62c.png' }]
  535. </div>
  536. </td>
  537. </tr>
  538. <tr>
  539. <td>appendMessage</td>
  540. <td>
  541. 新增一条消息, 如果当前焦点在该联系人的聊天窗口,设置
  542. scrollToBottom=true 添加之后自动定位到消息窗口底部
  543. </td>
  544. <td>Function(Message,scrollToBottom=false)</td>
  545. <td>-</td>
  546. <td></td>
  547. </tr>
  548. <tr>
  549. <td>removeMessage</td>
  550. <td>删除聊天消息</td>
  551. <td>Function(Message.id)</td>
  552. <td>-</td>
  553. <td></td>
  554. </tr>
  555. <tr>
  556. <td>updateMessage</td>
  557. <td>
  558. 修改消息,根据 Message.id
  559. 查找聊天消息并覆盖传入的值(toContactId会被忽略)
  560. </td>
  561. <td>Function(Message)</td>
  562. <td>-</td>
  563. <td></td>
  564. </tr>
  565. <tr>
  566. <td>appendContact</td>
  567. <td>添加联系人</td>
  568. <td>Function(Contact)</td>
  569. <td>-</td>
  570. <td></td>
  571. </tr>
  572. <tr>
  573. <td>removeContact</td>
  574. <td>删除联系人</td>
  575. <td>Function(Contact.id)</td>
  576. <td>-</td>
  577. <td></td>
  578. </tr>
  579. <tr>
  580. <td>updateContact</td>
  581. <td>修改联系人,根据 Contact.id 查找联系人并覆盖传入的值</td>
  582. <td>Function(Contact)</td>
  583. <td>-</td>
  584. <td></td>
  585. </tr>
  586. <tr>
  587. <td>getMessages</td>
  588. <td>返回所有本地消息,传入 Contact.id 则只返回与该联系人的消息</td>
  589. <td>Function(Contact.id)=>[Message]</td>
  590. <td>-</td>
  591. <td></td>
  592. </tr>
  593. <tr>
  594. <td>getCurrentContact</td>
  595. <td>返回当前聊天窗口的联系人信息</td>
  596. <td>Function()=>Contact</td>
  597. <td>-</td>
  598. <td></td>
  599. </tr>
  600. <tr>
  601. <td>getCurrentMessages</td>
  602. <td>返回当前聊天窗口的所有消息</td>
  603. <td>Function()=>[Message]</td>
  604. <td>-</td>
  605. <td></td>
  606. </tr>
  607. <tr>
  608. <td>getContacts</td>
  609. <td>返回所有本地联系人</td>
  610. <td>Function()=>[Contact]</td>
  611. <td>-</td>
  612. <td></td>
  613. </tr>
  614. <tr>
  615. <td>openDrawer</td>
  616. <td>打开联系人右侧抽屉,vnode 为抽屉内容</td>
  617. <td>Function(vnode)</td>
  618. <td>-</td>
  619. <td></td>
  620. </tr>
  621. <tr>
  622. <td>changeDrawer</td>
  623. <td>切换右侧抽屉显示/隐藏,vnode 为抽屉内容</td>
  624. <td>Function(DrawerOption)</td>
  625. <td>-</td>
  626. <td></td>
  627. </tr>
  628. <tr>
  629. <td>closeDrawer</td>
  630. <td>关闭抽屉</td>
  631. <td>Function()</td>
  632. <td>-</td>
  633. <td></td>
  634. </tr>
  635. <tr>
  636. <td>changeMenu</td>
  637. <td>切换左侧导航</td>
  638. <td>Function(Menu.name)</td>
  639. <td>-</td>
  640. <td></td>
  641. </tr>
  642. <tr>
  643. <td>changeContact</td>
  644. <td>切换聊天窗口</td>
  645. <td>Function(Contact.id,instance)</td>
  646. <td>-</td>
  647. <td></td>
  648. </tr>
  649. <tr>
  650. <td>messageViewToBottom</td>
  651. <td>将当前聊天窗口滚动到底部</td>
  652. <td>Function()</td>
  653. <td>-</td>
  654. <td></td>
  655. </tr>
  656. <tr>
  657. <td>setLastContentRender</td>
  658. <td>设置左侧联系人最新消息的渲染函数</td>
  659. <td>Function(Message.type, (Message)=>vnode)</td>
  660. <td>-</td>
  661. <td>
  662. setLastContentRender('image', message => { return
  663. <span>[最新图片]</span>
  664. })
  665. </td>
  666. </tr>
  667. <tr>
  668. <td>lastContentRender</td>
  669. <td>用来生成 Message.lastContent 需要的vnode结构。</td>
  670. <td>Function(Message)</td>
  671. <td>-</td>
  672. <td></td>
  673. </tr>
  674. <tr>
  675. <td>setEditorValue</td>
  676. <td>设置编辑框内容</td>
  677. <td>Function(string)</td>
  678. <td>-</td>
  679. <td></td>
  680. </tr>
  681. <tr>
  682. <td>getEditorValue</td>
  683. <td>获取编辑框内容</td>
  684. <td>Function()=>string</td>
  685. <td>-</td>
  686. <td></td>
  687. </tr>
  688. </table>
  689. <div class="title">组件插槽</div>
  690. <table class="table">
  691. <tr class="table-head">
  692. <th>插槽名</th>
  693. <th>说明</th>
  694. <th>参数</th>
  695. </tr>
  696. <tr>
  697. <td width="150">cover</td>
  698. <td width="350">初始化时的封面</td>
  699. <td width="150">-</td>
  700. </tr>
  701. <tr>
  702. <td width="150">editor-footer</td>
  703. <td width="350">消息输入框底部</td>
  704. <td width="150">-</td>
  705. </tr>
  706. <tr>
  707. <td width="150">message-title</td>
  708. <td width="350">消息列表的标题</td>
  709. <td width="150">Contact</td>
  710. </tr>
  711. <tr>
  712. <td width="150">message-after</td>
  713. <td width="350">每条消息的尾部</td>
  714. <td width="150">Message</td>
  715. </tr>
  716. <tr>
  717. <td width="150">sidebar-message</td>
  718. <td width="350">左侧最新消息列表插槽</td>
  719. <td width="150">Contact</td>
  720. </tr>
  721. <tr>
  722. <td width="150">sidebar-contact</td>
  723. <td width="350">左侧联系人列表插槽</td>
  724. <td width="150">Contact</td>
  725. </tr>
  726. <tr>
  727. <td width="150">sidebar-message-top</td>
  728. <td width="350">左侧最新消息列表的顶部,会随列表滚动</td>
  729. <td width="150">instance</td>
  730. </tr>
  731. <tr>
  732. <td width="150">sidebar-contact-top</td>
  733. <td width="350">左侧联系人列表的顶部,会随列表滚动</td>
  734. <td width="150">instance</td>
  735. </tr>
  736. <tr>
  737. <td width="150">sidebar-message-fixedtop</td>
  738. <td width="350">固定在左侧最新消息列表的顶部</td>
  739. <td width="150">instance</td>
  740. </tr>
  741. <tr>
  742. <td width="150">sidebar-contact-fixedtop</td>
  743. <td width="350">固定在左侧联系人列表的顶部</td>
  744. <td width="150">instance</td>
  745. </tr>
  746. <tr>
  747. <td width="150">contact-info</td>
  748. <td width="350">左侧联系人详细页</td>
  749. <td width="150">Contact</td>
  750. </tr>
  751. </table>
  752. <div class="title">组件事件</div>
  753. <table class="table">
  754. <tr class="table-head">
  755. <th>事件名</th>
  756. <th>说明</th>
  757. <th>参数</th>
  758. </tr>
  759. <tr>
  760. <td width="150">change-menu</td>
  761. <td width="350">当左侧导航选项卡切换的时候会触发该事件</td>
  762. <td width="150">Menu.name</td>
  763. </tr>
  764. <tr>
  765. <td width="150">menu-avatar-click</td>
  766. <td width="350">当左侧导航内的头像被点击时回触发该事件</td>
  767. <td width="150">Contact</td>
  768. </tr>
  769. <tr>
  770. <td width="150">change-contact</td>
  771. <td width="350">当左侧联系人点击时会触发该事件</td>
  772. <td width="150">Contact</td>
  773. </tr>
  774. <tr>
  775. <td width="150">pull-messages</td>
  776. <td width="350">
  777. 当切换聊天对象或者聊天窗口滚动到顶部时会触发该事件,调用next方法结束loading状态,如果设置了isEnd=true,下次聊天窗口滚动到顶部将不会再触发该事件
  778. </td>
  779. <td width="150">Contact,next([Message],isEnd),instance</td>
  780. </tr>
  781. <tr>
  782. <td width="150">message-click</td>
  783. <td width="350">点击聊天窗口中的消息时会触发该事件</td>
  784. <td width="150">event,key,Message,instance</td>
  785. </tr>
  786. <tr>
  787. <td width="150">send</td>
  788. <td width="350">当发送新消息时会触发该事件</td>
  789. <td width="150">
  790. Message,Function(Message):调用该函数完成消息发送,可以传入Message来改变消息内容,file:上传时的文件
  791. </td>
  792. </tr>
  793. </table>
  794. <div class="title" id="help1">如何创建消息?</div>
  795. <div>
  796. <p>
  797. Lemon-IMUI
  798. 目前内置了file、image、text、event四种消息类型,在实际应用当中肯定是不够的哦,咋办?没事的,我们继续往下see。<br />要创建消息首先要确定新消息的
  799. Message 结构。
  800. </p>
  801. <pre>
  802. {
  803. //值为 voice,用于解析的组件 name 必须为 lemonMessageVoice
  804. type: "voice",
  805. content: '语音消息',
  806. //自定义参数
  807. params1:'参数1',
  808. params2:'参数2',
  809. //必传参数
  810. id: "message-id",
  811. //必传参数
  812. fromUser:{
  813. avatar: ""
  814. displayName: "June"
  815. id: "1"
  816. },
  817. //必传参数
  818. sendTime: 1610872045162
  819. //必传参数
  820. status: "succeed"
  821. //必传参数
  822. toContactId: "contact-id"
  823. }
  824. </pre
  825. >
  826. <p>创建用于解析该消息的组件。</p>
  827. <pre>{{ tip }}</pre>
  828. <p>IMUI初始化的时候,设置左侧联系人最新消息的渲染内容</p>
  829. <pre>
  830. IMUI.setLastContentRender('voice', message => {
  831. return &lt;span&gt;[语音]&lt;/span&gt;
  832. })
  833. </pre
  834. >
  835. <p>最后一步,注册组件,必须使用全局注册的方式。</p>
  836. <pre>
  837. import Vue from 'vue';
  838. import LemonMessageVoice from './lemon-message-voice';
  839. Vue.component(LemonMessageVoice.name,LemonMessageVoice);
  840. </pre
  841. >
  842. <p>如果还有不明白的,可以到 examples/App.vue 查看示例代码</p>
  843. </div>
  844. <div class="title" id="help2">如何对接后端接口?</div>
  845. <p>1.初始化用户的信息</p>
  846. <pre
  847. v-text="
  848. `data(){
  849. return {
  850. user:{id:1:displayName:'June',avatar:''}
  851. }
  852. }`
  853. "
  854. ></pre>
  855. <pre
  856. v-text="`<lemon-imui :user='this.user' ref='IMUI'></lemon-imui>`"
  857. ></pre>
  858. <p>2.初始化联系人数据</p>
  859. <pre
  860. v-text="
  861. `mounted(){
  862. const { IMUI } = this.$refs;
  863. //初始化表情包。
  864. IMUI.initEmoji(...);
  865. //从后端请求联系人数据,包装成下面的样子
  866. const contacts = [{
  867. id: 2,
  868. displayName: '丽安娜',
  869. avatar:'',
  870. index: 'L',
  871. unread: 0,
  872. //最近一条消息的内容,如果值为空,不会出现在“聊天”列表里面。
  873. //lastContentRender 函数会将 file 消息转换为 '[文件]', image 消息转换为 '[图片]',对 text 会将文字里的表情标识替换为img标签,
  874. lastContent: IMUI.lastContentRender({type:'text',content:'你在干嘛呢?'})
  875. //最近一条消息的发送时间
  876. lastSendTime: 1566047865417,
  877. }];
  878. IMUI.initContacts(contacts);
  879. }`
  880. "
  881. ></pre>
  882. <p>3.拉取消息列表</p>
  883. <p>
  884. 现在刷新页面应该能够看到联系人了,但是点击联系人的话右边会一直处于加载中,这时需要监听
  885. pull-messages 事件。
  886. </p>
  887. <pre
  888. v-text="
  889. `<lemon-imui :user='this.user' ref='IMUI' @pull-messages='handlePullMessages'></lemon-imui>`
  890. "
  891. ></pre>
  892. <pre
  893. v-text="
  894. `methods:{
  895. handlePullMessages(contact, next) {
  896. //从后端请求消息数据,包装成下面的样子
  897. const messages = [{
  898. id: '唯一消息ID',
  899. status: 'succeed',
  900. type: 'text',
  901. sendTime: 1566047865417,
  902. content: '你什么才能对接完?',
  903. toContactId: contact.id,
  904. fromUser:this.user
  905. }]
  906. //将第二个参数设为true,表示已到末尾,聊天窗口顶部会显示“暂无更多消息”,不然会一直转圈。
  907. next(messages,true);
  908. },
  909. }`
  910. "
  911. ></pre>
  912. <p>4.发送消息</p>
  913. <p>现在在消息框发送新消息会一直转圈,这时需要监听 send 事件。</p>
  914. <pre
  915. v-text="
  916. `methods:{
  917. handleSend(message, next, file) {
  918. ... 调用你的消息发送业务接口
  919. //执行到next消息会停止转圈,如果接口调用失败,可以修改消息的状态 next({status:'failed'});
  920. next();
  921. },
  922. }`
  923. "
  924. ></pre>
  925. <p>5.接收消息</p>
  926. <pre
  927. v-text="
  928. `mounted(){
  929. WebSocket.onmessage = function(event) {
  930. //将接收到的数据包装成下面的样子
  931. const data = {
  932. id: '唯一消息ID',
  933. status: 'succeed',
  934. type: 'text',
  935. sendTime: 1566047865417,
  936. content: '马上就对接完了!',
  937. toContactId: 2,
  938. fromUser:{
  939. //如果 id == this.user.id消息会显示在右侧,否则在左侧
  940. id:2,
  941. displayName:'丽安娜',
  942. avatar:'',
  943. }
  944. };
  945. IMUI.appendMessage(data);
  946. };
  947. }`
  948. "
  949. ></pre>
  950. </div>
  951. </template>
  952. <script>
  953. import Vue from "vue";
  954. import LemonMessageVoice from "./lemon-message-voice";
  955. import QQIMUI from "./qq";
  956. import packageData from "../package.json";
  957. import EmojiData from "./database/emoji";
  958. Vue.component(LemonMessageVoice.name, LemonMessageVoice);
  959. Vue.component(QQIMUI.name, QQIMUI);
  960. const tip = `export default {
  961. //组件的name必须以lemonMessage开头,后面跟上 Message.type
  962. name: "lemonMessageVoice",
  963. inheritAttrs: false,
  964. //如果需要使用父组件的方法,可以使用注入。
  965. inject: ["IMUI"],
  966. render() {
  967. //lemon-message-basic 组件对气泡框、头像、事件等信息进行了公共的处理。
  968. return (
  969. <lemon-message-basic
  970. class="lemon-message-voice"
  971. props={{ ...this.$attrs }}
  972. scopedSlots={{
  973. content: props => {
  974. //返回HTML结构
  975. return <span>{props.content}&nbsp;🔈</span>
  976. }
  977. }}
  978. />
  979. );
  980. }
  981. };
  982. <style lang="stylus">
  983. .lemon-message.lemon-message-voice
  984. user-select none
  985. .lemon-message__content
  986. border 2px solid #000
  987. font-size 12px
  988. cursor pointer
  989. &::before
  990. display none
  991. </style>`;
  992. const getTime = () => {
  993. return new Date().getTime();
  994. };
  995. const generateRandId = () => {
  996. return Math.random()
  997. .toString(36)
  998. .substr(-8);
  999. };
  1000. const generateRandWord = () => {
  1001. return Math.random()
  1002. .toString(36)
  1003. .substr(2);
  1004. };
  1005. const generateMessage = (toContactId = "", fromUser) => {
  1006. if (!fromUser) {
  1007. fromUser = {
  1008. id: "system",
  1009. displayName: "系统测试",
  1010. avatar: "http://upload.qqbodys.com/allimg/1710/1035512943-0.jpg",
  1011. };
  1012. }
  1013. return {
  1014. id: generateRandId(),
  1015. status: "succeed",
  1016. type: "text",
  1017. sendTime: getTime(),
  1018. content: generateRandWord(),
  1019. //fileSize: 1231,
  1020. //fileName: "asdasd.doc",
  1021. toContactId,
  1022. fromUser,
  1023. };
  1024. };
  1025. export default {
  1026. name: "app",
  1027. data() {
  1028. return {
  1029. theme: "default",
  1030. contactContextmenu: [
  1031. {
  1032. text: "删除该聊天",
  1033. click: (e, instance, hide) => {
  1034. const { IMUI, contact } = instance;
  1035. IMUI.updateContact({
  1036. id: contact.id,
  1037. lastContent: null,
  1038. });
  1039. if (IMUI.currentContactId == contact.id) IMUI.changeContact(null);
  1040. hide();
  1041. },
  1042. },
  1043. {
  1044. text: "设置备注和标签",
  1045. },
  1046. {
  1047. text: "投诉",
  1048. },
  1049. {
  1050. icon: "lemon-icon-message",
  1051. render: (h, instance, hide) => {
  1052. return (
  1053. <div style="display:flex;justify-content:space-between;align-items:center;width:130px">
  1054. <span>加入黑名单</span>
  1055. <span>
  1056. <input type="checkbox" id="switch" />
  1057. <label id="switch-label" for="switch">
  1058. Toggle
  1059. </label>
  1060. </span>
  1061. </div>
  1062. );
  1063. },
  1064. },
  1065. {
  1066. click(e, instance, hide) {
  1067. const { IMUI, contact } = instance;
  1068. IMUI.removeContact(contact.id);
  1069. if (IMUI.currentContactId == contact.id) IMUI.changeContact(null);
  1070. hide();
  1071. },
  1072. color: "red",
  1073. text: "删除好友",
  1074. },
  1075. ],
  1076. contextmenu: [
  1077. {
  1078. click: (e, instance, hide) => {
  1079. const { IMUI, message } = instance;
  1080. const data = {
  1081. id: generateRandId(),
  1082. type: "event",
  1083. //使用 jsx 时 click必须使用箭头函数(使上下文停留在vue内)
  1084. content: (
  1085. <div>
  1086. <span>
  1087. 你撤回了一条消息{" "}
  1088. <span
  1089. v-show={message.type == "text"}
  1090. style="color:#333;cursor:pointer"
  1091. content={message.content}
  1092. on-click={e => {
  1093. IMUI.setEditorValue(e.target.getAttribute("content"));
  1094. }}
  1095. >
  1096. 重新编辑
  1097. </span>
  1098. </span>
  1099. </div>
  1100. ),
  1101. toContactId: message.toContactId,
  1102. sendTime: getTime(),
  1103. };
  1104. IMUI.removeMessage(message.id);
  1105. IMUI.appendMessage(data, true);
  1106. hide();
  1107. },
  1108. visible: instance => {
  1109. return instance.message.fromUser.id == this.user.id;
  1110. },
  1111. text: "撤回消息",
  1112. },
  1113. {
  1114. visible: instance => {
  1115. return instance.message.fromUser.id != this.user.id;
  1116. },
  1117. text: "举报",
  1118. },
  1119. {
  1120. text: "转发",
  1121. },
  1122. {
  1123. visible: instance => {
  1124. return instance.message.type == "text";
  1125. },
  1126. text: "复制文字",
  1127. },
  1128. {
  1129. visible: instance => {
  1130. return instance.message.type == "image";
  1131. },
  1132. text: "下载图片",
  1133. },
  1134. {
  1135. visible: instance => {
  1136. return instance.message.type == "file";
  1137. },
  1138. text: "下载文件",
  1139. },
  1140. {
  1141. click: (e, instance, hide) => {
  1142. const { IMUI, message } = instance;
  1143. IMUI.removeMessage(message.id);
  1144. hide();
  1145. },
  1146. icon: "lemon-icon-folder",
  1147. color: "red",
  1148. text: "删除",
  1149. },
  1150. ],
  1151. tip: tip,
  1152. packageData,
  1153. hideMenuAvatar: false,
  1154. hideMenu: false,
  1155. hideMessageName: false,
  1156. hideMessageTime: true,
  1157. user: {
  1158. id: "1",
  1159. displayName: "June",
  1160. avatar: "",
  1161. },
  1162. };
  1163. },
  1164. mounted() {
  1165. const contactData1 = {
  1166. id: "contact-1",
  1167. displayName: "工作协作群",
  1168. avatar: "http://upload.qqbodys.com/img/weixin/20170804/ji5qxg1am5ztm.jpg",
  1169. index: "[1]群组",
  1170. unread: 0,
  1171. lastSendTime: 1566047865417,
  1172. lastContent: "2",
  1173. };
  1174. const contactData2 = {
  1175. id: "contact-2",
  1176. displayName: "自定义内容",
  1177. avatar: "http://upload.qqbodys.com/img/weixin/20170807/jibfvfd00npin.jpg",
  1178. //index: "Z",
  1179. click(next) {
  1180. next();
  1181. },
  1182. renderContainer: () => {
  1183. return <h1 style="text-indent:20px">自定义页面</h1>;
  1184. },
  1185. lastSendTime: 1345209465000,
  1186. lastContent: "12312",
  1187. unread: 2,
  1188. };
  1189. const contactData3 = {
  1190. id: "contact-3",
  1191. displayName: "铁牛",
  1192. avatar: "http://upload.qqbodys.com/img/weixin/20170803/jiq4nzrkrnd0e.jpg",
  1193. index: "T",
  1194. unread: 32,
  1195. lastSendTime: 3,
  1196. lastContent: "你好123",
  1197. };
  1198. const contactData4 = {
  1199. id: "contact-4",
  1200. displayName: "如花",
  1201. avatar:
  1202. "https://dss2.bdstatic.com/70cFvnSh_Q1YnxGkpoWK1HF6hhy/it/u=4275424924,2201401076&fm=111&gp=0.jpg",
  1203. index: "",
  1204. unread: 1,
  1205. lastSendTime: 3,
  1206. lastContent: "吃饭了嘛",
  1207. };
  1208. const { IMUI } = this.$refs;
  1209. setTimeout(() => {
  1210. IMUI.changeContact("contact-1");
  1211. }, 500);
  1212. IMUI.setLastContentRender("event", message => {
  1213. return `[自定义通知内容]`;
  1214. });
  1215. let contactList = [
  1216. { ...contactData1 },
  1217. { ...contactData2 },
  1218. { ...contactData3 },
  1219. //...Array(100).fill(contactData1)
  1220. ];
  1221. IMUI.initContacts(contactList);
  1222. IMUI.initMenus([
  1223. {
  1224. name: "messages",
  1225. },
  1226. {
  1227. name: "contacts",
  1228. },
  1229. {
  1230. name: "custom1",
  1231. title: "自定义按钮1",
  1232. unread: 0,
  1233. render: menu => {
  1234. return <i class="lemon-icon-attah" />;
  1235. },
  1236. renderContainer: () => {
  1237. return (
  1238. <div class="article">
  1239. <ul>
  1240. <li class="article-item">
  1241. <h2>人民日报谈网红带货:产品真的值得买吗?</h2>
  1242. </li>
  1243. <li class="article-item">
  1244. 甘肃夏河县发生5.7级地震 暂未接到人员伤亡报告
  1245. </li>
  1246. <li class="article-item">
  1247. 北方多地风力仍强沙尘相伴,东北内蒙古等地迎雨雪
  1248. </li>
  1249. <li class="article-item">
  1250. 英货车案:越南警方采集疑死者家属DNA作比对
  1251. </li>
  1252. <li class="article-item">
  1253. 知名连锁咖啡店的蛋糕吃出活虫 曝光内幕太震惊
  1254. </li>
  1255. </ul>
  1256. <lemon-contact
  1257. props={{ contact: contactData1 }}
  1258. style="margin:20px"
  1259. />
  1260. <lemon-contact
  1261. props={{ contact: contactData3 }}
  1262. style="margin:20px"
  1263. />
  1264. </div>
  1265. );
  1266. },
  1267. isBottom: true,
  1268. },
  1269. {
  1270. name: "custom2",
  1271. title: "自定义按钮2",
  1272. unread: 0,
  1273. click: () => {
  1274. alert("拦截导航点击事件");
  1275. },
  1276. render: menu => {
  1277. return <i class="lemon-icon-group" />;
  1278. },
  1279. isBottom: true,
  1280. },
  1281. ]);
  1282. IMUI.initEditorTools([
  1283. {
  1284. name: "emoji",
  1285. },
  1286. {
  1287. name: "uploadFile",
  1288. },
  1289. {
  1290. name: "uploadImage",
  1291. },
  1292. {
  1293. name: "test1",
  1294. click: () => {
  1295. IMUI.$refs.editor.selectFile("application/vnd.ms-excel");
  1296. },
  1297. render: () => {
  1298. return <span>Excel</span>;
  1299. },
  1300. },
  1301. {
  1302. name: "test1",
  1303. click: () => {
  1304. IMUI.initEditorTools([{ name: "uploadFile" }, { name: "emoji" }]);
  1305. },
  1306. render: () => {
  1307. return <span>重制工具栏</span>;
  1308. },
  1309. },
  1310. {
  1311. name: "test2",
  1312. isRight: true,
  1313. title: "上传 Excel",
  1314. click: () => {
  1315. alert("点击了 ··· ");
  1316. },
  1317. render: () => {
  1318. return <b>···</b>;
  1319. },
  1320. },
  1321. ]);
  1322. IMUI.initEmoji(EmojiData);
  1323. IMUI.setLastContentRender("voice", message => {
  1324. return <span>[语音]</span>;
  1325. });
  1326. const { SimpleIMUI } = this.$refs;
  1327. contactData1.id = "11";
  1328. SimpleIMUI.initContacts([contactData1]);
  1329. SimpleIMUI.initEmoji(EmojiData);
  1330. SimpleIMUI.changeContact(contactData1.id);
  1331. },
  1332. methods: {
  1333. changeTheme() {
  1334. this.theme = this.theme == "default" ? "blue" : "default";
  1335. },
  1336. scrollToTop() {
  1337. document.body.scrollIntoView();
  1338. },
  1339. handleMenuAvatarClick() {
  1340. console.log("Event:menu-avatar-click");
  1341. },
  1342. handleMessageClick(e, key, message, instance) {
  1343. console.log("点击了消息", e, key, message);
  1344. if (key == "status") {
  1345. instance.updateMessage({
  1346. id: message.id,
  1347. status: "going",
  1348. content: "正在重新发送消息...",
  1349. });
  1350. setTimeout(() => {
  1351. instance.updateMessage({
  1352. id: message.id,
  1353. status: "succeed",
  1354. content: "发送成功",
  1355. });
  1356. }, 2000);
  1357. }
  1358. },
  1359. changeMenuAvatarVisible() {
  1360. this.hideMenuAvatar = !this.hideMenuAvatar;
  1361. },
  1362. changeMenuVisible() {
  1363. this.hideMenu = !this.hideMenu;
  1364. },
  1365. changeMessageNameVisible() {
  1366. this.hideMessageName = !this.hideMessageName;
  1367. },
  1368. changeMessageTimeVisible() {
  1369. this.hideMessageTime = !this.hideMessageTime;
  1370. },
  1371. removeMessage() {
  1372. const { IMUI } = this.$refs;
  1373. const messages = IMUI.getCurrentMessages();
  1374. const id = messages[messages.length - 1].id;
  1375. if (messages.length > 0) {
  1376. IMUI.removeMessage(id);
  1377. }
  1378. },
  1379. updateMessage() {
  1380. const { IMUI } = this.$refs;
  1381. const messages = IMUI.getCurrentMessages();
  1382. const message = messages[messages.length - 1];
  1383. if (messages.length > 0) {
  1384. const update = {
  1385. id: message.id,
  1386. status: "succeed",
  1387. type: "file",
  1388. fileName: "被修改成文件了.txt",
  1389. fileSize: "4200000",
  1390. };
  1391. if (message.type == "event") {
  1392. update.fromUser = this.user;
  1393. }
  1394. IMUI.updateMessage(update);
  1395. IMUI.messageViewToBottom();
  1396. }
  1397. },
  1398. appendCustomMessage() {
  1399. const { IMUI } = this.$refs;
  1400. const message = {
  1401. id: generateRandId(),
  1402. status: "succeed",
  1403. type: "voice",
  1404. sendTime: getTime(),
  1405. content: "语音消息",
  1406. params1: "1",
  1407. params2: "2",
  1408. toContactId: "contact-1",
  1409. fromUser: this.user,
  1410. };
  1411. IMUI.appendMessage(message, true);
  1412. },
  1413. appendMessage() {
  1414. const { IMUI } = this.$refs;
  1415. const contact = IMUI.currentContact;
  1416. const message = generateMessage("contact-3");
  1417. message.fromUser = {
  1418. ...message.fromUser,
  1419. ...this.user,
  1420. };
  1421. IMUI.appendMessage(message, true);
  1422. },
  1423. appendEventMessage() {
  1424. const { IMUI } = this.$refs;
  1425. const message = {
  1426. id: generateRandId(),
  1427. type: "event",
  1428. content: (
  1429. <span>
  1430. 邀请你加入群聊{" "}
  1431. <span
  1432. style="color:#333;cursor:pointer"
  1433. on-click={() => alert("OK")}
  1434. >
  1435. 接受
  1436. </span>
  1437. </span>
  1438. ),
  1439. toContactId: "contact-3",
  1440. sendTime: getTime(),
  1441. };
  1442. IMUI.appendMessage(message, true);
  1443. },
  1444. updateContact() {
  1445. this.$refs.IMUI.updateContact({
  1446. id: "contact-3",
  1447. unread: 10,
  1448. displayName: generateRandWord(),
  1449. lastSendTime: getTime(),
  1450. lastContent: "修改昵称为随机字母",
  1451. });
  1452. },
  1453. changeDrawer(contact, instance) {
  1454. instance.changeDrawer({
  1455. //width: 240,
  1456. //height: "90%",
  1457. //offsetX:0 ,
  1458. //offsetY: ,
  1459. //position: "center",
  1460. // inside: true,
  1461. // offsetX: -280,
  1462. // offsetY: -100,
  1463. render: () => {
  1464. return (
  1465. <div class="drawer-content">
  1466. <p>
  1467. <b>自定义抽屉</b>
  1468. </p>
  1469. <p>{contact.displayName}</p>
  1470. </div>
  1471. );
  1472. },
  1473. });
  1474. },
  1475. handleChangeContact(contact, instance) {
  1476. console.log("Event:change-contact");
  1477. instance.updateContact({
  1478. id: contact.id,
  1479. unread: 0,
  1480. });
  1481. instance.closeDrawer();
  1482. },
  1483. handleSend(message, next, file) {
  1484. console.log(message, next, file);
  1485. setTimeout(() => {
  1486. next();
  1487. }, 1000);
  1488. },
  1489. handlePullMessages(contact, next, instance) {
  1490. const otheruser = {
  1491. id: contact.id,
  1492. displayName: contact.displayName,
  1493. avatar: contact.avatar,
  1494. };
  1495. setTimeout(() => {
  1496. const messages = [
  1497. generateMessage(instance.currentContactId, this.user),
  1498. generateMessage(instance.currentContactId, otheruser),
  1499. generateMessage(instance.currentContactId, this.user),
  1500. generateMessage(instance.currentContactId, otheruser),
  1501. generateMessage(instance.currentContactId, this.user),
  1502. generateMessage(instance.currentContactId, this.user),
  1503. generateMessage(instance.currentContactId, otheruser),
  1504. {
  1505. ...generateMessage(instance.currentContactId, this.user),
  1506. ...{ status: "failed" },
  1507. },
  1508. ];
  1509. let isEnd = false;
  1510. if (
  1511. instance.getMessages(instance.currentContactId).length +
  1512. messages.length >
  1513. 11
  1514. )
  1515. isEnd = true;
  1516. next(messages, isEnd);
  1517. }, 500);
  1518. },
  1519. handleChangeMenu() {
  1520. console.log("Event:change-menu");
  1521. },
  1522. openCustomContainer() {},
  1523. },
  1524. };
  1525. </script>
  1526. <style lang="stylus">
  1527. ::selection{background:#000;color:#fff;}
  1528. body
  1529. font-family "Microsoft YaHei"
  1530. background #f6f6f6 !important
  1531. #app
  1532. width 90%
  1533. margin 0 auto
  1534. padding-bottom 100px
  1535. .scroll-top
  1536. cursor pointer
  1537. position fixed
  1538. bottom 40px
  1539. left 50%
  1540. border-radius 50%
  1541. background #fff
  1542. font-size 18px
  1543. overflow hidden
  1544. width 40px
  1545. height 40px
  1546. line-height 40px
  1547. user-select none
  1548. text-align center
  1549. transform rotate(-45deg) translateX(-50%)
  1550. box-shadow 0 0 30px rgba(0,0,0,0.1);
  1551. &:hover
  1552. font-size 22px
  1553. a
  1554. color #0c5ed9
  1555. text-decoration none
  1556. font-size 12px
  1557. .action
  1558. margin-top 20px
  1559. .lemon-button
  1560. margin-right 10px
  1561. margin-bottom 10px
  1562. .link
  1563. font-size 14px
  1564. margin-top 15px
  1565. a
  1566. display inline-block
  1567. margin 0 5px
  1568. text-decoration none
  1569. background #ffba00
  1570. border-radius 4px
  1571. padding 5px 10px
  1572. color rgba(0,0,0,0.8)
  1573. .logo
  1574. position relative
  1575. display inline-block
  1576. margin 60px auto
  1577. user-select none
  1578. .logo-text
  1579. font-size 38px
  1580. .logo-sub
  1581. font-size 18px
  1582. color #999
  1583. font-weight 300
  1584. .logo-badge
  1585. position absolute
  1586. top -10px
  1587. left 230px
  1588. background #000
  1589. border-radius 16px
  1590. color #f9f9f9
  1591. font-size 12px
  1592. padding 4px 8px
  1593. .title
  1594. font-size 24px
  1595. line-height 26px
  1596. border-left 1px solid #ffba00
  1597. padding-left 15px
  1598. margin-bottom 15px
  1599. margin-top 30px
  1600. user-select none
  1601. .table
  1602. width 100%
  1603. border-radius 10px
  1604. background #fff
  1605. border-collapse collapse
  1606. tr
  1607. cursor pointer
  1608. tr:not(.table-head):hover
  1609. background #ffba00 !important
  1610. tr:nth-of-type(even)
  1611. background #f9f9f9
  1612. th
  1613. user-select none
  1614. color #999
  1615. td,
  1616. th
  1617. text-align left
  1618. padding 10px 15px
  1619. font-size 14px
  1620. font-weight normal
  1621. .imui-center
  1622. margin-bottom 60px
  1623. .lemon-wrapper
  1624. border:1px solid #ddd;
  1625. .lemon-drawer
  1626. border:1px solid #ddd;
  1627. border-left:0;
  1628. .drawer-content
  1629. padding 15px
  1630. .more
  1631. font-size 12px
  1632. line-height 24px
  1633. height 24px
  1634. position absolute
  1635. top 14px
  1636. right 14px
  1637. cursor pointer
  1638. user-select none
  1639. color #f1f1f1
  1640. display inline-block
  1641. border-radius 4px
  1642. background #111
  1643. padding 0 8px
  1644. &:active
  1645. background #999
  1646. .bar
  1647. text-align center
  1648. line-height 30px
  1649. background #fff
  1650. margin 15px
  1651. color #666
  1652. user-select none
  1653. font-size 12px
  1654. .cover
  1655. text-align center
  1656. user-select none
  1657. position absolute
  1658. top 50%
  1659. left 50%
  1660. transform translate(-50%,-50%)
  1661. i
  1662. font-size 84px
  1663. color #e6e6e6
  1664. p
  1665. font-size 18px
  1666. color #ddd
  1667. line-height 50px
  1668. .article-item
  1669. line-height 34px
  1670. cursor pointer
  1671. &:hover
  1672. text-decoration underline
  1673. color #318efd
  1674. pre
  1675. background #fff
  1676. border-radius 8px
  1677. padding 15px
  1678. .lemon-simple .lemon-container{
  1679. z-index:5
  1680. }
  1681. .lemon-simple .lemon-drawer{
  1682. z-index:4
  1683. }
  1684. input#switch[type=checkbox]{
  1685. height: 0;
  1686. width: 0;
  1687. display:none;
  1688. }
  1689. label#switch-label {
  1690. cursor: pointer;
  1691. text-indent: -9999px;
  1692. width: 34px;
  1693. height: 20px;
  1694. background: #aaa;
  1695. display: block;
  1696. border-radius: 100px;
  1697. position: relative;
  1698. }
  1699. label#switch-label:after {
  1700. content: '';
  1701. position: absolute;
  1702. top: 2px;
  1703. left: 2px;
  1704. width: 16px;
  1705. height: 16px;
  1706. background: #fff;
  1707. border-radius: 20px;
  1708. transition: 0.3s;
  1709. }
  1710. input#switch:checked + label {
  1711. background: #0fd547;
  1712. }
  1713. input#switch:checked + label:after {
  1714. left: calc(100% - 2px);
  1715. transform: translateX(-100%);
  1716. }
  1717. label#switch-label:active:after {
  1718. width: 20px;
  1719. }
  1720. </style>