index.js 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700
  1. const SHARE_TO = (() => {
  2. const ShareType = {
  3. CREATE: 'create',
  4. UPDATE: 'update',
  5. CANCEL: 'cancel',
  6. };
  7. const { SharePermissionChangeType: PermissionType, PageTarget, ShareLibType } = commonConstants;
  8. const Mode = {
  9. PROJECT: 1,
  10. RATION_LIB: 2,
  11. GLJ_LIB: 3,
  12. BLOCK_LIB: 4,
  13. };
  14. const ModeToLibType = {
  15. [Mode.RATION_LIB]: ShareLibType.RATION_LIB,
  16. [Mode.GLJ_LIB]: ShareLibType.GLJ_LIB,
  17. [Mode.BLOCK_LIB]: ShareLibType.BLOCK_LIB,
  18. };
  19. // 当前模式
  20. let curMode = Mode.PROJECT;
  21. // 当前分享的项目ID
  22. let curProjectID;
  23. // 当前项目的已分享列表
  24. let curSharedUsers = [];
  25. // 清除缓存
  26. function clearCache() {
  27. curProjectID = null;
  28. curSharedUsers = [];
  29. }
  30. // 最近联系人显示数量
  31. const rencentCount = 5;
  32. // 获取初始数据:1.最近分享人 2.联系人 3.已分享人
  33. async function getInitialData(projectID) {
  34. return await ajaxPost('/pm/api/getInitialShareData', { user_id: userID, count: rencentCount, projectID }, false);
  35. }
  36. // 获取分享库的初始数据
  37. async function getInitialLibData(libType) {
  38. return await ajaxPost('/pm/api/getInitialShareLibData', { user_id: userID, count: rencentCount, libType }, false);
  39. }
  40. // 获取头像视图html
  41. function getAvatarHTML(mobile, realName) {
  42. // 手机最后一位
  43. const lastMobileNumer = mobile.substr(-1);
  44. // 显示名称为真实名称后两位
  45. const nickName = realName.substr(-2);
  46. return `<span class="avatar bg-${lastMobileNumer}">${nickName}</span>`;
  47. }
  48. /**
  49. * 获取用户列表视图html:最近分享和联系人用
  50. * @param {Array} users - 排序过的用户数据
  51. * @param {Boolean} showAlphabet - 是否显示字母表分类
  52. * @return {HTMLString}
  53. * */
  54. function getUserHTML(users, showAlphabet) {
  55. let curLetter = '';
  56. return users.reduce((html, user) => {
  57. const mobile = user.mobile || '';
  58. const realName = user.real_name || '';
  59. const company = user.company || '';
  60. if (showAlphabet) {
  61. // 名字首个字符对应拼音
  62. const letter = pinyinUtil.getFirstLetter(realName).substr(0, 1);
  63. if (letter !== curLetter) {
  64. curLetter = letter;
  65. html += `<li class="letter">${letter}</li>`;
  66. }
  67. }
  68. const avatarHtml = getAvatarHTML(mobile, realName);
  69. return html +
  70. `<li>
  71. ${avatarHtml}
  72. <div class="book-body">
  73. <h5 class="mt-0" title="${company}">${realName}</h5>
  74. <span>${mobile}</span>
  75. </div>
  76. </li>`;
  77. }, '');
  78. }
  79. // 初始化最近分享视图
  80. function initRecentView(recentUsers) {
  81. const recentShareHTML = getUserHTML(recentUsers, false);
  82. $('#recent-share').html(recentShareHTML);
  83. // 点击最近分享列表自动添加该用户添加到搜索框中
  84. $('#recent-share li').click(function () {
  85. const mobile = $(this).find('div span')[0].textContent;
  86. $('#share-phone').val(mobile);
  87. handleSearch()
  88. })
  89. }
  90. // 初始化联系人视图
  91. function initContactsView(contacts) {
  92. // 联系人按拼英首字母降序排序
  93. contacts.sort((a, b) => {
  94. const realNameA = a.real_name || '';
  95. const realNameB = b.real_name || '';
  96. return realNameA.localeCompare(realNameB, 'zh-Hans-CN', { sensitivity: 'accent' })
  97. });
  98. const contactsHTML = getUserHTML(contacts, true);
  99. $('#contacts').html(contactsHTML);
  100. // 点击联系人自动添加该联系人添加到搜索框中
  101. $('#contacts li:not(.letter)').click(function () {
  102. const mobile = $(this).find('div span')[0].textContent;
  103. $('#share-phone').val(mobile);
  104. $('#contacts-menue').removeClass('show');
  105. handleSearch()
  106. });
  107. }
  108. // 初始化已分享视图
  109. function initSharedView(sharedUsers) {
  110. const html = sharedUsers.reduce((html, user, index) => {
  111. const mobile = user.mobile || '';
  112. const realName = user.real_name || '';
  113. const company = user.company || '';
  114. const avatarHTML = getAvatarHTML(mobile, realName);
  115. const copyLabelFor = `allowCopy${index}`;
  116. const editLabelFor = `allowEdit${index}`;
  117. return html +
  118. `<li class="card mb-1">
  119. <div class="card-body p-1 row m-0">
  120. ${avatarHTML}
  121. <div class="book-body col-auto pl-0">
  122. <h5 class="mt-0">${realName}</h5>
  123. ${mobile}
  124. </div>
  125. <div class="col-5">${company}</div>
  126. <div class="col ml-auto p-0">
  127. <div class="d-flex justify-content-end">
  128. ${curMode === Mode.PROJECT ? `<div>
  129. <div class="custom-control custom-checkbox">
  130. <input type="checkbox" class="custom-control-input allow-copy" id="${copyLabelFor}" data-user="${user._id}" ${user.allowCopy ? 'checked' : ''}>
  131. <label class="custom-control-label" for="${copyLabelFor}">允许拷贝</label>
  132. </div>
  133. <div class="custom-control custom-checkbox">
  134. <input type="checkbox" class="custom-control-input allow-edit" id="${editLabelFor}" data-user="${user._id}" ${user.allowCooperate ? 'checked' : ''}>
  135. <label class="custom-control-label" for="${editLabelFor}">允许编辑</label>
  136. </div>
  137. </div>` : ''}
  138. <div class="ml-3 d-flex align-items-center">
  139. <button class="btn btn-sm btn-outline-danger cancel-share" data-user="${user._id}">取消分享</button>
  140. </div>
  141. </div>
  142. </div>
  143. </div>
  144. </li>`;
  145. }, '');
  146. $('#shared-list').html(html);
  147. // 编辑允许拷贝
  148. $('#shared-list .allow-copy').click(function () {
  149. handleCheckBoxClick.call(this);
  150. handleShareAction.call(this, ShareType.UPDATE);
  151. });
  152. // 编辑允许编辑
  153. $('#shared-list .allow-edit').click(function () {
  154. handleCheckBoxClick.call(this);
  155. handleShareAction.call(this, ShareType.UPDATE);
  156. });
  157. // 取消分享
  158. $('#shared-list .cancel-share').click(function () {
  159. if (curMode === Mode.PROJECT) {
  160. handleShareAction.call(this, ShareType.CANCEL);
  161. } else {
  162. handleShareLibAction.call(this, ShareType.CANCEL, ModeToLibType[curMode]);
  163. }
  164. });
  165. }
  166. // 初始化搜索结果视图
  167. function initSearchResultView(user) {
  168. if (!user) {
  169. $('#share-search-result').html('')
  170. } else {
  171. const mobile = user.mobile || '';
  172. const realName = user.real_name || '';
  173. const company = user.company || '';
  174. const avatarHTML = getAvatarHTML(mobile, realName);
  175. const html =
  176. `<li class="card mb-1">
  177. <div class="card-body p-1 row m-0">
  178. ${avatarHTML}
  179. <div class="book-body col-auto pl-0">
  180. <h5 class="mt-0">${realName}</h5>
  181. ${mobile}
  182. </div>
  183. <div class="col-5">${company}</div>
  184. <div class="col ml-auto p-0">
  185. <div class="d-flex justify-content-end">
  186. ${curMode === Mode.PROJECT ? `<div>
  187. <div class="custom-control custom-checkbox">
  188. <input type="checkbox" class="custom-control-input" id="allow-copy" checked="">
  189. <label class="custom-control-label" for="allow-copy">允许拷贝</label>
  190. </div>
  191. <div class="custom-control custom-checkbox">
  192. <input type="checkbox" class="custom-control-input" id="allow-edit">
  193. <label class="custom-control-label" for="allow-edit">允许编辑</label>
  194. </div>
  195. </div>` : ''}
  196. <div class="ml-3 d-flex align-items-center"><button class="btn btn-sm btn-primary" id="share-to" data-user="${user._id}">分享给Ta</button></div>
  197. </div>
  198. </div>
  199. </div>
  200. </li>`;
  201. $('#share-search-result').html(html);
  202. // 允许拷贝
  203. $('#allow-copy').click(function () {
  204. handleCheckBoxClick.call(this);
  205. });
  206. // 允许编辑
  207. $('#allow-edit').click(function () {
  208. handleCheckBoxClick.call(this);
  209. });
  210. // 分享给事件
  211. $('#share-to').click(function () {
  212. if (curMode === Mode.PROJECT) {
  213. handleShareAction.call(this, ShareType.CREATE, user);
  214. } else {
  215. handleShareLibAction.call(this, ShareType.CREATE, ModeToLibType[curMode], user);
  216. }
  217. });
  218. }
  219. }
  220. // 复选框框的状态随点击更改,不处理的话input的checked并不会自动处理
  221. function handleCheckBoxClick() {
  222. const curChecked = !$(this).attr('checked');
  223. if (curChecked) {
  224. $(this).attr('checked', 'checked');
  225. } else {
  226. $(this).removeAttr('checked');
  227. }
  228. }
  229. async function handleShareLibAction(shareType, libType, user) {
  230. try {
  231. $.bootstrapLoading.start();
  232. const receiver = $(this).data('user');
  233. let shareData;
  234. if (shareType === ShareType.CREATE) {
  235. shareData = [{ receiver }];
  236. } else if (shareType === ShareType.CANCEL) {
  237. shareData = [{ receiver, isCancel: true }];
  238. }
  239. const postData = {
  240. type: shareType,
  241. user_id: userID,
  242. count: rencentCount,
  243. libType,
  244. shareData
  245. };
  246. const rst = await ajaxPost('/pm/api/shareLib', postData);
  247. // 请求成功后刷新视图
  248. if (shareType === ShareType.CREATE || shareType === ShareType.CANCEL) {
  249. if (shareType === ShareType.CREATE) {
  250. curSharedUsers.unshift(user);
  251. $('#share-phone').val('');
  252. initSearchResultView();
  253. } else {
  254. curSharedUsers = curSharedUsers.filter(user => user._id !== receiver);
  255. }
  256. if (libType === ShareLibType.BLOCK_LIB) {
  257. const shareTip = curSharedUsers.length ? curSharedUsers.reduce((acc, user) => acc += ` ${user.real_name}`, '已分享给') : '';
  258. blockLibObj.initialShareTip = shareTip;
  259. $('#btn_block_share').attr('data-original-title', shareTip);
  260. }
  261. if (Array.isArray(rst.recentUsers)) {
  262. initRecentView(rst.recentUsers);
  263. }
  264. if (Array.isArray(rst.contacts)) {
  265. initContactsView(rst.contacts)
  266. }
  267. initSharedView(curSharedUsers);
  268. refreshShareTip(curSharedUsers);
  269. refreshTreeView();
  270. }
  271. } catch (err) {
  272. console.log(err);
  273. alert(`${String(err)} 请重试。`);
  274. initSharedView(curSharedUsers);
  275. } finally {
  276. $.bootstrapLoading.end();
  277. }
  278. }
  279. // 添加分享、编辑分享、取消分享的动作
  280. async function handleShareAction(shareType, user) {
  281. try {
  282. $.bootstrapLoading.start();
  283. const receiver = $(this).data('user');
  284. let shareData;
  285. let type = ShareType.UPDATE;
  286. if (shareType === ShareType.CREATE) {
  287. const allowCopy = $('#allow-copy').prop('checked');
  288. const allowCooperate = $('#allow-edit').prop('checked');
  289. shareData = [{ userID: receiver, allowCopy, allowCooperate }];
  290. type = ShareType.CREATE; // 上传的服务器的type,删除跟更新是一样的
  291. } else if (shareType === ShareType.UPDATE) {
  292. const allowCopy = $(`[data-user=${receiver}].allow-copy`).prop('checked');
  293. const allowCooperate = $(`[data-user=${receiver}].allow-edit`).prop('checked');
  294. shareData = [{ userID: receiver, allowCopy, allowCooperate }];
  295. } else {
  296. shareData = [{ userID: receiver, isCancel: true }];
  297. }
  298. // 获取权限变更的类型
  299. const permissionType = getPermissionType(shareType, curSharedUsers, shareData[0]);
  300. const postData = {
  301. user_id: userID,
  302. type,
  303. permissionType,
  304. projectID: curProjectID,
  305. count: rencentCount,
  306. shareData
  307. };
  308. const rst = await ajaxPost('/pm/api/share', postData);
  309. // 请求成功后刷新视图
  310. if (shareType === ShareType.CREATE || shareType === ShareType.CANCEL) {
  311. if (shareType === ShareType.CREATE) {
  312. user.allowCopy = shareData[0].allowCopy;
  313. user.allowCooperate = shareData[0].allowCooperate;
  314. curSharedUsers.unshift(user);
  315. $('#share-phone').val('');
  316. initSearchResultView();
  317. } else {
  318. curSharedUsers = curSharedUsers.filter(user => user._id !== receiver);
  319. }
  320. if (Array.isArray(rst.recentUsers)) {
  321. initRecentView(rst.recentUsers);
  322. }
  323. if (Array.isArray(rst.contacts)) {
  324. initContactsView(rst.contacts)
  325. }
  326. initSharedView(curSharedUsers);
  327. refreshShareTip(curSharedUsers);
  328. refreshTreeView();
  329. } else {
  330. const matchItem = curSharedUsers.find(item => item._id === receiver);
  331. if (matchItem) {
  332. matchItem.allowCopy = shareData[0].allowCopy;
  333. matchItem.allowCooperate = shareData[0].allowCooperate;
  334. }
  335. }
  336. if (permissionType !== null) {
  337. const payload = [PermissionType.UPDATE_COOPERATE, PermissionType.UPDATE_COPY].includes(permissionType)
  338. ? { prop: { allowCopy: shareData[0].allowCopy, allowCooperate: shareData[0].allowCooperate } }
  339. : null;
  340. emitPermissionChange(permissionType, receiver, curProjectID, rst.emitTenders, payload);
  341. }
  342. } catch (err) {
  343. console.log(err);
  344. alert(`${String(err)} 请重试。`);
  345. initSharedView(curSharedUsers);
  346. } finally {
  347. $.bootstrapLoading.end();
  348. }
  349. }
  350. // 获取权限变更的类型
  351. function getPermissionType(shareType, cache, curShareData) {
  352. if (shareType === ShareType.CANCEL) {
  353. return PermissionType.CANCEL;
  354. }
  355. if (shareType === ShareType.CREATE) {
  356. return PermissionType.SHARE;
  357. }
  358. if (!cache) {
  359. return null;
  360. }
  361. const match = cache.find(item => item._id === curShareData.userID);
  362. if (!match) {
  363. return null;
  364. }
  365. if (match.allowCooperate !== curShareData.allowCooperate) {
  366. return PermissionType.UPDATE_COOPERATE;
  367. }
  368. if (match.allowCopy !== curShareData.allowCopy) {
  369. return PermissionType.UPDATE_COPY;
  370. }
  371. return null;
  372. }
  373. // 权限变更消息推送
  374. function emitPermissionChange(permissionType, userID, projectID, emitTenders, payload) {
  375. const compilationID = typeof projectObj !== 'undefined' ? projectObj.project.projectInfo.compilation : compilationData._id;
  376. socket.emit('sharePermissionChange', { permissionType, userID, compilationID, projectID, emitTenders, payload });
  377. }
  378. // 权限变更处理监听
  379. function permissionChangeListener() {
  380. socket.on('sharePermissionChange', ({ permissionType, target, payload }) => {
  381. const type = `${permissionType}-${target}`;
  382. switch (type) {
  383. case `${PermissionType.READ}-${PageTarget.PM}`:
  384. handleShareItemRead(payload);
  385. break;
  386. case `${PermissionType.UPDATE_COOPERATE}-${PageTarget.PM}`:
  387. case `${PermissionType.UPDATE_COPY}-${PageTarget.PM}`:
  388. handlePMPropChange(payload);
  389. break;
  390. case `${PermissionType.UPDATE_COOPERATE}-${PageTarget.MAIN}`:
  391. handleMainCooperateChange(payload);
  392. break;
  393. case `${PermissionType.CANCEL}-${PageTarget.PM}`:
  394. handlePMCancelPermission(payload);
  395. break;
  396. case `${PermissionType.CANCEL}-${PageTarget.MAIN}`:
  397. handleMainCancelPermission();
  398. break;
  399. case `${PermissionType.SHARE}-${PageTarget.PM}`:
  400. handlePMSharePermission(payload);
  401. break;
  402. }
  403. });
  404. }
  405. // 项目管理页面分享权限变更后相关处理函数
  406. // 已读分享项目
  407. function handleShareItemRead({ markReadProjectIDs }) {
  408. const isActive = $('#tab_pm_share').hasClass('active');
  409. if (isActive) {
  410. pmShare.handleMarkRead(unreadShareList, markReadProjectIDs);
  411. } else {
  412. markReadProjectIDs.forEach(projectID => {
  413. removeUnread(projectID, unreadShareList);
  414. });
  415. }
  416. }
  417. // 分享条数属性变更(是否可拷贝、是否可编辑)
  418. // @param {Object} prop - 属性变更对象 eg: { allowCopy: false }
  419. function handlePMPropChange({ projectID, prop }) {
  420. // 如果当前在项目管理的分享页面,需要刷新相关节点
  421. const isActive = $('#tab_pm_share').hasClass('active');
  422. if (isActive) {
  423. pmShare.handlePropChange(projectID, prop);
  424. }
  425. }
  426. // 已读某未读分享项目
  427. function removeUnread(projectID, list) {
  428. const idx = list.indexOf(projectID);
  429. if (idx >= 0) {
  430. list.splice(idx, 1);
  431. refreshUnreadCount(list.length);
  432. }
  433. }
  434. // 取消分享后项目管理页面接收到推送后到操作
  435. function handlePMCancelPermission({ projectID }) {
  436. removeUnread(projectID, unreadShareList);
  437. // 如果当前停留在分享标签界面,清楚分享项目
  438. const isActive = $('#tab_pm_share').hasClass('active');
  439. if (isActive) {
  440. pmShare.handleCancelShare(projectID);
  441. }
  442. }
  443. // 通知信息点击事件
  444. function notify() {
  445. pmShare.initShareTree();
  446. $("#notify").hide();
  447. }
  448. // 新增分享后项目管理页面接收到推送后的操作
  449. function handlePMSharePermission({ projectID }) {
  450. unreadShareList.push(projectID);
  451. unreadShareList = [...new Set(unreadShareList)];
  452. refreshUnreadCount(unreadShareList.length);
  453. // 如果当前停留在分享标签界面,需要弹出提示
  454. const isActive = $('#tab_pm_share').hasClass('active');
  455. if (isActive) {
  456. $("#message").html('您有新的分享项目,<a href="javascript:void(0);" id="load-data" onclick="SHARE_TO.notify()">点击刷新列表</a>');
  457. $("#notify").show();
  458. }
  459. }
  460. // 刷新未读分享标记
  461. function refreshUnreadCount(count) {
  462. if (count) {
  463. $('#unread-share-count').text(count);
  464. $('#unread-share-count').show();
  465. } else {
  466. $('#unread-share-count').hide();
  467. }
  468. }
  469. // 主页面的分享权限变更后相关处理函数
  470. function handleMainCooperateChange({ allowCooperate }) {
  471. if (projectObj.project.projectInfo.shareState && projectObj.project.projectInfo.shareState.allowCooperate !== allowCooperate) {
  472. setLocalCache(commonConstants.StorageKey.ONCE_MAIN_LOADED, '分享设置已被修改,当前项目已自动刷新。');
  473. window.location.reload();
  474. }
  475. }
  476. function handleMainCancelPermission() {
  477. // 定位到空白页
  478. window.location.replace(`/blank?type=${commonConstants.BlankType.SHARE_CANCEL}`);
  479. }
  480. // 刷新项目管理树视图
  481. // 如果是在项目管理页面,需要刷新树(分享图标可能需要清除)
  482. function refreshTreeView() {
  483. if (typeof projTreeObj !== 'undefined' && projTreeObj.tree.selected) {
  484. projTreeObj.tree.selected.data.shareInfo = curSharedUsers;
  485. const sheet = projTreeObj.workBook.getSheet(0);
  486. projTreeObj.renderSheetFuc(sheet, function () {
  487. sheet.invalidateLayout();
  488. sheet.repaint();
  489. });
  490. }
  491. }
  492. // 刷新造价书的分享按钮tooltip提示
  493. function refreshShareTip(sharedUsers) {
  494. const $shareTip = $('#share-tip');
  495. if (!$shareTip) {
  496. return;
  497. }
  498. const limit = 2;
  499. const count = sharedUsers.length;
  500. const users = sharedUsers.slice(0, 2);
  501. const tip = users.reduce((acc, user, index) => {
  502. if (index === 0) {
  503. acc += '已分享给';
  504. acc += user.real_name;
  505. } else {
  506. acc += ` ${user.real_name}`;
  507. }
  508. if (index === users.length - 1 && count > limit) {
  509. acc += `等${count}人`;
  510. }
  511. return acc;
  512. }, '');
  513. $shareTip.attr('data-original-title', tip);
  514. }
  515. // 初始化分享给的页面
  516. // mode: 模式,分享项目、分享库
  517. async function initModal(mode, projectID) {
  518. try {
  519. curMode = mode;
  520. $.bootstrapLoading.start();
  521. // 恢复
  522. $('#share-phone').val('');
  523. initSearchResultView();
  524. $('#share-hint').text('');
  525. let sharedUsers = [];
  526. let recentUsers = [];
  527. let contacts = [];
  528. let data;
  529. if (mode === Mode.PROJECT) {
  530. curProjectID = projectID;
  531. data = await getInitialData(projectID);
  532. if (data.isFree) {
  533. return hintBox.versionBox('此功能仅在专业版中提供,免费版可选择单位工程进行分享。');
  534. }
  535. } else {
  536. data = await getInitialLibData(ModeToLibType[curMode]);
  537. }
  538. sharedUsers = data.sharedUsers;
  539. recentUsers = data.recentUsers;
  540. contacts = data.contacts;
  541. curSharedUsers = sharedUsers;
  542. initSharedView(sharedUsers);
  543. initRecentView(recentUsers);
  544. initContactsView(contacts);
  545. setTimeout(() => $('#share-phone').focus(), 200);
  546. $('#share').modal('show');
  547. } catch (err) {
  548. console.log(err);
  549. alert(err);
  550. } finally {
  551. $.bootstrapLoading.end();
  552. }
  553. }
  554. // 退出分享给页面
  555. function exitModal() {
  556. clearCache();
  557. }
  558. // 分享给
  559. async function handleSearch() {
  560. let phone = $('#share-phone').val();
  561. phone = phone && phone.trim() || '';
  562. //$('#share-hint').text('');
  563. initSearchResultView();
  564. if (!phone) {
  565. $('#share-hint').text('请输入手机号码。');
  566. return;
  567. }
  568. // 根据手机号获取用户
  569. const user = await ajaxPost('/user/getUserByMobile', { mobile: phone });
  570. if (!user) {
  571. $('#share-hint').text('账号不存在。');
  572. return;
  573. }
  574. if (user._id === userID) {
  575. $('#share-hint').text('不可分享给自己。');
  576. return;
  577. }
  578. const matched = curSharedUsers.find(item => item._id === user._id);
  579. if (matched) {
  580. $('#share-hint').text('已与该用户分享。');
  581. return;
  582. }
  583. $('#share-hint').text('');
  584. initSearchResultView(user);
  585. }
  586. // 一些事件的监听
  587. function handleEventListener() {
  588. // 界面消失
  589. $('#share').on('hide.bs.modal', function () {
  590. exitModal();
  591. });
  592. // 联系人下拉
  593. $('#contacts-dropdown').click(function () {
  594. const $subMenu = $('#contacts-menue');
  595. const visible = $subMenu.is(':visible');
  596. if (visible) {
  597. $subMenu.removeClass('show');
  598. } else {
  599. $subMenu.addClass('show');
  600. }
  601. });
  602. // 点击body时,联系人菜单的处理
  603. $('body').click(function (e) {
  604. const body = $(this)[0];
  605. const $contactsMenu = $('#contacts-menue');
  606. const contactsMenu = $contactsMenu[0];
  607. const dropdownButton = $('#contacts-dropdown')[0]
  608. if (!$contactsMenu.is(':visible')) {
  609. return;
  610. }
  611. let target = e.target;
  612. while (target !== body) {
  613. if ([contactsMenu, dropdownButton].includes(target)) {
  614. return;
  615. }
  616. target = target.parentElement;
  617. }
  618. $(contactsMenu.parentElement).removeClass('show');
  619. $contactsMenu.removeClass('show');
  620. });
  621. // 输入手机号查找要分享给的用户
  622. let keyupTime = 0;
  623. let delayTime = 500;
  624. function delayKeyup(callback) {
  625. const nowTime = Date.now();
  626. keyupTime = nowTime;
  627. setTimeout(function () {
  628. if (nowTime - keyupTime == 0) {
  629. callback();
  630. }
  631. }, delayTime);
  632. }
  633. $('#share-phone').on('keyup', function () {
  634. delayKeyup(function () {
  635. console.log(curSharedUsers);
  636. handleSearch();
  637. });
  638. });
  639. $('#sharePhone').on('keypress', function (e) {
  640. if (e.keyCode === 13) {
  641. $(this).blur();
  642. }
  643. });
  644. }
  645. return {
  646. Mode,
  647. initModal,
  648. handleEventListener,
  649. permissionChangeListener,
  650. emitPermissionChange,
  651. getAvatarHTML,
  652. notify,
  653. removeUnread,
  654. }
  655. })();