searchDept.vue 16 KB

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