searchDept.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563
  1. <template>
  2. <view class="content">
  3. <view class="search-box">
  4. <seiminSearch class="mSearch-input-box" :mode="2" button="inside" placeholder="请输入科室名称" @search="doSearch(false)"
  5. @input="changeInp" @confirm="doSearch(false)" v-model="keyword"></seiminSearch>
  6. </view>
  7. <view class="search-keyword">
  8. <scroll-view class="keyword-list-box" v-show="isShowKeywordList" scroll-y>
  9. <block v-for="(row, index) in keywordList" :key="index">
  10. <view class="keyword-entry" hover-class="keyword-entry-tap">
  11. <view class="keyword-text" @tap.stop="doSearch(keywordList[index].keyword)">
  12. <rich-text :nodes="row.htmlStr"></rich-text>
  13. </view>
  14. <view class="keyword-img" @tap.stop="doSearch(keywordList[index].keyword)">
  15. <image src="/static/HM-search/back.png"></image>
  16. </view>
  17. </view>
  18. </block>
  19. </scroll-view>
  20. <scroll-view class="keyword-box" v-show="!isShowKeywordList" scroll-y>
  21. <view class="keyword-block" v-if="oldKeywordList.length > 0">
  22. <view class="keyword-list-header">
  23. <view>历史搜索</view>
  24. <view>
  25. <image @tap="oldDelete" src="/static/HM-search/delete.png"></image>
  26. </view>
  27. </view>
  28. <view class="keyword">
  29. <view v-for="(keyword, index) in oldKeywordList" @tap="changeInp(keyword)" :key="index">{{ keyword }}
  30. </view>
  31. </view>
  32. </view>
  33. </scroll-view>
  34. </view>
  35. <seiminModel ref="seiminModel"></seiminModel>
  36. </view>
  37. </template>
  38. <script>
  39. import {
  40. reqFetchDataList,
  41. reqChangeHospital,
  42. reqUpdData,
  43. reqGetCurrentUser,
  44. } from "../../request/api.js";
  45. import {
  46. mapState,
  47. mapMutations
  48. } from "vuex";
  49. export default {
  50. data() {
  51. return {
  52. keyword: "",
  53. oldKeywordList: [],
  54. keywordList: [],
  55. isShowKeywordList: false,
  56. deptList: [],
  57. timer: null, //定时器
  58. searchText: "", //搜索文本
  59. searchData: [], //搜索结果
  60. };
  61. },
  62. computed: {
  63. ...mapState("user", ["loginInfo"]),
  64. ...mapState('other', [
  65. "deptDisplay",
  66. "searchDeptParams",
  67. "searchDeptResult",
  68. "searchDeptResultList",
  69. ]),
  70. },
  71. onUnload() {
  72. if (this.timer) {
  73. clearTimeout(this.timer);
  74. this.timer = null;
  75. }
  76. },
  77. onLoad(options) {
  78. this.init();
  79. },
  80. methods: {
  81. ...mapMutations('other', [
  82. "changeSearchDeptResult",
  83. "changeSeiminModel",
  84. "changeSearchDeptResultList",
  85. ]),
  86. ...mapMutations("user", ["changeLoginInfo"]),
  87. init() {
  88. this.loadOldKeyword();
  89. },
  90. blur() {
  91. uni.hideKeyboard();
  92. },
  93. //加载历史搜索,自动读取本地Storage
  94. loadOldKeyword() {
  95. uni.getStorage({
  96. key: "OldKeys",
  97. success: (res) => {
  98. var OldKeys = JSON.parse(res.data);
  99. this.oldKeywordList = OldKeys;
  100. },
  101. });
  102. },
  103. //防抖搜索
  104. changeInp(event) {
  105. this.searchText = event;
  106. clearTimeout(this.timer);
  107. this.timer = setTimeout(() => {
  108. this.inputChange(event);
  109. }, 500);
  110. },
  111. //监听输入
  112. inputChange(event) {
  113. //兼容引入组件时传入参数情况
  114. var keyword = event.detail ? event.detail.value : event;
  115. if (!keyword) {
  116. this.keywordList = [];
  117. this.isShowKeywordList = false;
  118. return;
  119. }
  120. this.keyword = keyword;
  121. this.isShowKeywordList = true;
  122. let postData = {
  123. idx: 0,
  124. sum: 20,
  125. department: {},
  126. };
  127. // 院区参数----start
  128. if (this.searchDeptParams.type === "changeDept_index") {
  129. //首页切换科室
  130. postData.department.hospital = {
  131. id: this.searchDeptParams.hospital.value,
  132. };
  133. postData.department.keyWord = keyword;
  134. postData.department.nurseSign = 1;
  135. } else if (
  136. this.searchDeptParams.type === "selectDept_start_qucikCreateOrder" ||
  137. this.searchDeptParams.type === "selectDept_end_qucikCreateOrder"
  138. ) {
  139. //快捷建单选择起点科室或选择终点科室
  140. postData.department.hospital = {
  141. id: this.loginInfo.user.currentHospital.id,
  142. };
  143. postData.department.searchType = 1;
  144. postData.department.ids = this.searchDeptParams.ids;
  145. postData.department.keyWord = keyword;
  146. postData.department.nurseSign = 1;
  147. if (this.searchDeptParams.departmentStrategy == 205) {
  148. // 固定科室类型
  149. postData.department.type = {
  150. id: this.searchDeptParams.deptType,
  151. };
  152. }
  153. } else {
  154. postData.department.hospital = {
  155. id: this.loginInfo.user.currentHospital.id,
  156. };
  157. // 科室名称或科室别称开关----start
  158. if (this.deptDisplay == 1) {
  159. //科室名称
  160. postData.department.dept = keyword;
  161. } else if (this.deptDisplay == 2) {
  162. //科室别称
  163. postData.department.deptalias = keyword;
  164. }
  165. // 科室名称或科室别称开关----end
  166. }
  167. // 院区参数----end
  168. uni.showLoading({
  169. title: "加载中",
  170. mask: true,
  171. });
  172. reqFetchDataList("data", "department", postData).then((res) => {
  173. uni.hideLoading();
  174. if (res.status == 200) {
  175. this.searchData.push({
  176. name: keyword,
  177. list: res.list || [],
  178. });
  179. let searchText = this.searchText.detail ?
  180. this.searchText.detail.value :
  181. this.searchText;
  182. let index = this.searchData.findIndex(
  183. (item) => item.name === searchText
  184. );
  185. this.deptList = index >= 0 ? this.searchData[index].list : [];
  186. this.keywordList = this.drawCorrelativeKeyword(
  187. this.deptList,
  188. keyword
  189. );
  190. } else {
  191. this.$refs.seiminModel.showChangeDept({
  192. skin: "toast",
  193. icon: "error",
  194. content: res.msg || "获取数据失败",
  195. });
  196. throw new Error(res.msg || "获取数据失败");
  197. }
  198. });
  199. },
  200. //高亮关键字
  201. drawCorrelativeKeyword(keywords, keyword) {
  202. var len = keywords.length,
  203. keywordArr = [];
  204. for (var i = 0; i < len; i++) {
  205. var row = keywords[i];
  206. //定义高亮#9f9f9f
  207. var html = "";
  208. if (this.deptDisplay == 1) {
  209. // 科室名称
  210. html = row.dept.replace(
  211. keyword,
  212. "<span style='color: #9f9f9f;'>" + keyword + "</span>"
  213. );
  214. } else if (this.deptDisplay == 2) {
  215. // 科室别称
  216. html = row.deptalias.replace(
  217. keyword,
  218. "<span style='color: #9f9f9f;'>" + keyword + "</span>"
  219. );
  220. }
  221. html = "<div>" + html + "</div>";
  222. var tmpObj = {
  223. keyword: this.deptDisplay == 1 ? row.dept : row.deptalias,
  224. htmlStr: html,
  225. };
  226. keywordArr.push(tmpObj);
  227. }
  228. return keywordArr;
  229. },
  230. //清除历史搜索
  231. oldDelete() {
  232. uni.showModal({
  233. content: "确定清除历史搜索记录?",
  234. success: (res) => {
  235. if (res.confirm) {
  236. console.log("用户点击确定");
  237. this.oldKeywordList = [];
  238. uni.removeStorage({
  239. key: "OldKeys",
  240. });
  241. } else if (res.cancel) {
  242. console.log("用户点击取消");
  243. }
  244. },
  245. });
  246. },
  247. //执行搜索
  248. doSearch(keyword) {
  249. keyword = keyword === false ? this.keyword : keyword;
  250. this.keyword = keyword;
  251. this.saveKeyword(keyword); //保存为历史
  252. let arr = this.deptList.filter((item) => {
  253. return this.deptDisplay == 1 ?
  254. item.dept === keyword :
  255. item.deptalias === keyword;
  256. });
  257. if (arr.length) {
  258. this.changeSearchDeptResult(arr[0]);
  259. if (this.searchDeptParams.type === "changeDept_index") {
  260. //首页切换科室
  261. this.changeDept_index_handler();
  262. } else if (
  263. this.searchDeptParams.type === "selectDept_start_qucikCreateOrder"
  264. ) {
  265. //快捷建单选择起点科室
  266. this.searchDeptResultList.start = arr[0];
  267. this.changeSearchDeptResultList(this.searchDeptResultList);
  268. this.selectDept_startOrEnd_qucikCreateOrder();
  269. } else if (
  270. this.searchDeptParams.type === "selectDept_end_qucikCreateOrder"
  271. ) {
  272. //快捷建单选择终点科室
  273. this.searchDeptResultList.end = arr[0];
  274. this.changeSearchDeptResultList(this.searchDeptResultList);
  275. this.selectDept_startOrEnd_qucikCreateOrder();
  276. }
  277. }
  278. },
  279. //保存关键字到历史记录
  280. saveKeyword(keyword) {
  281. uni.getStorage({
  282. key: "OldKeys",
  283. success: (res) => {
  284. var OldKeys = JSON.parse(res.data);
  285. var findIndex = OldKeys.indexOf(keyword);
  286. if (findIndex == -1) {
  287. OldKeys.unshift(keyword);
  288. } else {
  289. OldKeys.splice(findIndex, 1);
  290. OldKeys.unshift(keyword);
  291. }
  292. //最多10个纪录
  293. OldKeys.length > 10 && OldKeys.pop();
  294. uni.setStorage({
  295. key: "OldKeys",
  296. data: JSON.stringify(OldKeys),
  297. });
  298. this.oldKeywordList = OldKeys; //更新历史搜索
  299. },
  300. fail: (e) => {
  301. var OldKeys = [keyword];
  302. uni.setStorage({
  303. key: "OldKeys",
  304. data: JSON.stringify(OldKeys),
  305. });
  306. this.oldKeywordList = OldKeys; //更新历史搜索
  307. },
  308. });
  309. },
  310. //首页切换科室
  311. changeDept_index_handler() {
  312. if (this.searchDeptParams.type === "changeDept_index") {
  313. // 首页切换科室
  314. uni.showLoading({
  315. title: "加载中",
  316. mask: true,
  317. });
  318. if (
  319. this.loginInfo.user.currentHospital.id ==
  320. this.searchDeptParams.hospital.value
  321. ) {
  322. //没有切换院区
  323. this.changeDept_index_handler_common();
  324. } else {
  325. //切换院区
  326. reqChangeHospital({
  327. currentHosId: this.searchDeptParams.hospital.value,
  328. loginType: "PC",
  329. }).then((res) => {
  330. if (res.status == 200) {
  331. this.changeDept_index_handler_common();
  332. } else {
  333. uni.hideLoading();
  334. this.$refs.seiminModel.showChangeDept({
  335. skin: "toast",
  336. icon: "error",
  337. content: res.msg || "操作失败",
  338. });
  339. throw new Error(res.msg || "操作失败");
  340. }
  341. });
  342. }
  343. }
  344. },
  345. // 首页切换科室,公共方法
  346. changeDept_index_handler_common() {
  347. // 更新用户
  348. reqUpdData("data", "user", {
  349. user: {
  350. dept: {
  351. id: this.searchDeptResult.id,
  352. },
  353. id: this.loginInfo.user.id,
  354. },
  355. }).then((res) => {
  356. if (res.status == 200) {
  357. // 获取当前登录用户信息
  358. reqGetCurrentUser().then((res) => {
  359. uni.hideLoading();
  360. if (res.status == 200) {
  361. this.$refs.seiminModel.showChangeDept({
  362. skin: "toast",
  363. icon: "success",
  364. content: "切换科室成功",
  365. btns: [{
  366. click: (e) => {
  367. this.changeLoginInfo(res.data);
  368. this.changeSeiminModel(true);
  369. uni.navigateTo({
  370. url: this.searchDeptParams.backUrl,
  371. });
  372. },
  373. }, ],
  374. });
  375. } else {
  376. this.$refs.seiminModel.showChangeDept({
  377. skin: "toast",
  378. icon: "error",
  379. content: res.msg || "操作失败",
  380. });
  381. throw new Error(res.msg || "操作失败");
  382. }
  383. });
  384. } else {
  385. uni.hideLoading();
  386. this.$refs.seiminModel.showChangeDept({
  387. skin: "toast",
  388. icon: "error",
  389. content: res.msg || "操作失败",
  390. });
  391. throw new Error(res.msg || "操作失败");
  392. }
  393. });
  394. },
  395. // 快捷建单选择起点科室或选择终点科室
  396. selectDept_startOrEnd_qucikCreateOrder() {
  397. uni.navigateTo({
  398. url: this.searchDeptParams.backUrl,
  399. });
  400. },
  401. },
  402. };
  403. </script>
  404. <style lang="scss" scoped>
  405. view {
  406. display: block;
  407. }
  408. .search-box {
  409. background-color: rgb(242, 242, 242);
  410. padding: 15upx 2.5%;
  411. display: flex;
  412. justify-content: space-between;
  413. position: sticky;
  414. top: 0;
  415. }
  416. .search-box .mSearch-input-box {
  417. width: 100%;
  418. }
  419. .search-box .input-box {
  420. width: 85%;
  421. flex-shrink: 1;
  422. display: flex;
  423. justify-content: center;
  424. align-items: center;
  425. }
  426. .search-box .search-btn {
  427. width: 15%;
  428. margin: 0 0 0 2%;
  429. display: flex;
  430. justify-content: center;
  431. align-items: center;
  432. flex-shrink: 0;
  433. font-size: 28upx;
  434. color: #fff;
  435. background: linear-gradient(to right, #ff9801, #ff570a);
  436. border-radius: 60upx;
  437. }
  438. .search-box .input-box>input {
  439. width: 100%;
  440. height: 60upx;
  441. font-size: 32upx;
  442. border: 0;
  443. border-radius: 60upx;
  444. -webkit-appearance: none;
  445. -moz-appearance: none;
  446. appearance: none;
  447. padding: 0 3%;
  448. margin: 0;
  449. background-color: #ffffff;
  450. }
  451. .placeholder-class {
  452. color: #9e9e9e;
  453. }
  454. .search-keyword {
  455. width: 100%;
  456. background-color: rgb(242, 242, 242);
  457. }
  458. .keyword-list-box {
  459. height: calc(100vh - 110upx);
  460. padding-top: 10upx;
  461. border-radius: 20upx 20upx 0 0;
  462. background-color: #fff;
  463. }
  464. .keyword-entry-tap {
  465. background-color: #eee;
  466. }
  467. .keyword-entry {
  468. width: 94%;
  469. height: 80upx;
  470. margin: 0 3%;
  471. font-size: 30upx;
  472. color: #333;
  473. display: flex;
  474. justify-content: space-between;
  475. align-items: center;
  476. border-bottom: solid 1upx #e7e7e7;
  477. }
  478. .keyword-entry image {
  479. width: 60upx;
  480. height: 60upx;
  481. }
  482. .keyword-entry .keyword-text,
  483. .keyword-entry .keyword-img {
  484. height: 80upx;
  485. display: flex;
  486. align-items: center;
  487. }
  488. .keyword-entry .keyword-text {
  489. width: 90%;
  490. }
  491. .keyword-entry .keyword-img {
  492. width: 10%;
  493. justify-content: center;
  494. }
  495. .keyword-box {
  496. height: calc(100vh - 110upx);
  497. border-radius: 20upx 20upx 0 0;
  498. background-color: #fff;
  499. }
  500. .keyword-box .keyword-block {
  501. padding: 10upx 0;
  502. }
  503. .keyword-box .keyword-block .keyword-list-header {
  504. width: 94%;
  505. padding: 10upx 3%;
  506. font-size: 27upx;
  507. color: #333;
  508. display: flex;
  509. justify-content: space-between;
  510. }
  511. .keyword-box .keyword-block .keyword-list-header image {
  512. width: 40upx;
  513. height: 40upx;
  514. }
  515. .keyword-box .keyword-block .keyword {
  516. width: 94%;
  517. padding: 3px 3%;
  518. display: flex;
  519. flex-flow: wrap;
  520. justify-content: flex-start;
  521. }
  522. .keyword-box .keyword-block .hide-hot-tis {
  523. display: flex;
  524. justify-content: center;
  525. font-size: 28upx;
  526. color: #6b6b6b;
  527. }
  528. .keyword-box .keyword-block .keyword>view {
  529. display: flex;
  530. justify-content: center;
  531. align-items: center;
  532. border-radius: 60upx;
  533. padding: 0 20upx;
  534. margin: 10upx 20upx 10upx 0;
  535. height: 60upx;
  536. font-size: 28upx;
  537. background-color: rgb(242, 242, 242);
  538. color: #6b6b6b;
  539. }
  540. </style>