DlMockController.java 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. package org.jeecg.modules.dlglong.controller;
  2. import com.alibaba.fastjson.JSON;
  3. import com.alibaba.fastjson.JSONArray;
  4. import com.alibaba.fastjson.JSONObject;
  5. import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
  6. import com.baomidou.mybatisplus.core.metadata.IPage;
  7. import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
  8. import lombok.extern.slf4j.Slf4j;
  9. import org.apache.commons.io.IOUtils;
  10. import org.apache.commons.lang3.StringUtils;
  11. import org.jeecg.common.api.vo.Result;
  12. import org.jeecg.common.system.query.MatchTypeEnum;
  13. import org.jeecg.common.system.query.QueryCondition;
  14. import org.jeecg.common.system.query.QueryGenerator;
  15. import org.jeecg.common.constant.VxeSocketConst;
  16. import org.jeecg.modules.demo.mock.vxe.websocket.VxeSocket;
  17. import org.jeecg.modules.dlglong.entity.MockEntity;
  18. import org.springframework.web.bind.annotation.*;
  19. import javax.servlet.http.HttpServletRequest;
  20. import java.io.IOException;
  21. import java.io.InputStream;
  22. import java.net.URLDecoder;
  23. import java.util.*;
  24. /**
  25. * @Description: DlMockController
  26. * @author: jeecg-boot
  27. */
  28. @Slf4j
  29. @RestController
  30. @RequestMapping("/mock/dlglong")
  31. public class DlMockController {
  32. /**
  33. * 模拟更改状态
  34. *
  35. * @param id
  36. * @param status
  37. * @return
  38. */
  39. @GetMapping("/change1")
  40. public Result mockChange1(@RequestParam("id") String id, @RequestParam("status") String status) {
  41. /* id 为 行的id(rowId),只要获取到rowId,那么只需要调用 VXESocket.sendMessageToAll() 即可 */
  42. // 封装行数据
  43. JSONObject rowData = new JSONObject();
  44. // 这个字段就是要更改的行数据ID
  45. rowData.put("id", id);
  46. // 这个字段就是要更改的列的key和具体的值
  47. rowData.put("status", status);
  48. // 模拟更改数据
  49. this.mockChange(rowData);
  50. return Result.ok();
  51. }
  52. /**
  53. * 模拟更改拖轮状态
  54. *
  55. * @param id
  56. * @param tugStatus
  57. * @return
  58. */
  59. @GetMapping("/change2")
  60. public Result mockChange2(@RequestParam("id") String id, @RequestParam("tug_status") String tugStatus) {
  61. /* id 为 行的id(rowId),只要获取到rowId,那么只需要调用 VXESocket.sendMessageToAll() 即可 */
  62. // 封装行数据
  63. JSONObject rowData = new JSONObject();
  64. // 这个字段就是要更改的行数据ID
  65. rowData.put("id", id);
  66. // 这个字段就是要更改的列的key和具体的值
  67. JSONObject status = JSON.parseObject(tugStatus);
  68. rowData.put("tug_status", status);
  69. // 模拟更改数据
  70. this.mockChange(rowData);
  71. return Result.ok();
  72. }
  73. /**
  74. * 模拟更改进度条状态
  75. *
  76. * @param id
  77. * @param progress
  78. * @return
  79. */
  80. @GetMapping("/change3")
  81. public Result mockChange3(@RequestParam("id") String id, @RequestParam("progress") String progress) {
  82. /* id 为 行的id(rowId),只要获取到rowId,那么只需要调用 VXESocket.sendMessageToAll() 即可 */
  83. // 封装行数据
  84. JSONObject rowData = new JSONObject();
  85. // 这个字段就是要更改的行数据ID
  86. rowData.put("id", id);
  87. // 这个字段就是要更改的列的key和具体的值
  88. rowData.put("progress", progress);
  89. // 模拟更改数据
  90. this.mockChange(rowData);
  91. return Result.ok();
  92. }
  93. private void mockChange(JSONObject rowData) {
  94. // 封装socket数据
  95. JSONObject socketData = new JSONObject();
  96. // 这里的 socketKey 必须要和调度计划页面上写的 socketKey 属性保持一致
  97. socketData.put("socketKey", "page-dispatch");
  98. // 这里的 args 必须得是一个数组,下标0是行数据,下标1是caseId,一般不用传
  99. socketData.put("args", new Object[]{rowData, ""});
  100. // 封装消息字符串,这里的 type 必须是 VXESocketConst.TYPE_UVT
  101. String message = VxeSocket.packageMessage(VxeSocketConst.TYPE_UVT, socketData);
  102. // 调用 sendMessageToAll 发送给所有在线的用户
  103. VxeSocket.sendMessageToAll(message);
  104. }
  105. /**
  106. * 模拟更改【大船待审】状态
  107. *
  108. * @param status
  109. * @return
  110. */
  111. @GetMapping("/change4")
  112. public Result mockChange4(@RequestParam("status") String status) {
  113. // 封装socket数据
  114. JSONObject socketData = new JSONObject();
  115. // 这里的 key 是前端注册时使用的key,必须保持一致
  116. socketData.put("key", "dispatch-dcds-status");
  117. // 这里的 args 必须得是一个数组,每一位都是注册方法的参数,按顺序传递
  118. socketData.put("args", new Object[]{status});
  119. // 封装消息字符串,这里的 type 必须是 VXESocketConst.TYPE_UVT
  120. String message = VxeSocket.packageMessage(VxeSocketConst.TYPE_CSD, socketData);
  121. // 调用 sendMessageToAll 发送给所有在线的用户
  122. VxeSocket.sendMessageToAll(message);
  123. return Result.ok();
  124. }
  125. /**
  126. * 【模拟】即时保存单行数据
  127. *
  128. * @param rowData 行数据,实际使用时可以替换成一个实体类
  129. */
  130. @PutMapping("/immediateSaveRow")
  131. public Result mockImmediateSaveRow(@RequestBody JSONObject rowData) throws Exception {
  132. System.out.println("即时保存.rowData:" + rowData.toJSONString());
  133. // 延时1.5秒,模拟网慢堵塞真实感
  134. Thread.sleep(500);
  135. return Result.ok();
  136. }
  137. /**
  138. * 【模拟】即时保存整个表格的数据
  139. *
  140. * @param tableData 表格数据(实际使用时可以替换成一个List实体类)
  141. */
  142. @PostMapping("/immediateSaveAll")
  143. public Result mockImmediateSaveAll(@RequestBody JSONArray tableData) throws Exception {
  144. // 【注】:
  145. // 1、tableData里包含该页所有的数据
  146. // 2、如果你实现了“即时保存”,那么除了新增的数据,其他的都是已经保存过的了,
  147. // 不需要再进行一次update操作了,所以可以在前端传数据的时候就遍历判断一下,
  148. // 只传新增的数据给后台insert即可,否者将会造成性能上的浪费。
  149. // 3、新增的行是没有id的,通过这一点,就可以判断是否是新增的数据
  150. System.out.println("即时保存.tableData:" + tableData.toJSONString());
  151. // 延时1.5秒,模拟网慢堵塞真实感
  152. Thread.sleep(1000);
  153. return Result.ok();
  154. }
  155. /**
  156. * 获取模拟数据
  157. *
  158. * @param pageNo 页码
  159. * @param pageSize 页大小
  160. * @param parentId 父ID,不传则查询顶级
  161. * @return
  162. */
  163. @GetMapping("/getData")
  164. public Result getMockData(
  165. @RequestParam(name = "pageNo", defaultValue = "1") Integer pageNo,
  166. @RequestParam(name = "pageSize", defaultValue = "10") Integer pageSize,
  167. // 父级id,根据父级id查询子级,如果为空则查询顶级
  168. @RequestParam(name = "parentId", required = false) String parentId
  169. ) {
  170. // 模拟JSON数据路径
  171. String path = "classpath:org/jeecg/modules/dlglong/json/dlglong.json";
  172. // 读取JSON数据
  173. JSONArray dataList = readJsonData(path);
  174. if (dataList == null) {
  175. return Result.error("读取数据失败!");
  176. }
  177. IPage<JSONObject> page = this.queryDataPage(dataList, parentId, pageNo, pageSize);
  178. return Result.ok(page);
  179. }
  180. /**
  181. * 获取模拟“调度计划”页面的数据
  182. *
  183. * @param pageNo 页码
  184. * @param pageSize 页大小
  185. * @param parentId 父ID,不传则查询顶级
  186. * @return
  187. */
  188. @GetMapping("/getDdjhData")
  189. public Result getMockDdjhData(
  190. // SpringMVC 会自动将参数注入到实体里
  191. MockEntity mockEntity,
  192. @RequestParam(name = "pageNo", defaultValue = "1") Integer pageNo,
  193. @RequestParam(name = "pageSize", defaultValue = "10") Integer pageSize,
  194. // 父级id,根据父级id查询子级,如果为空则查询顶级
  195. @RequestParam(name = "parentId", required = false) String parentId,
  196. @RequestParam(name = "status", required = false) String status,
  197. // 高级查询条件
  198. @RequestParam(name = "superQueryParams", required = false) String superQueryParams,
  199. // 高级查询模式
  200. @RequestParam(name = "superQueryMatchType", required = false) String superQueryMatchType,
  201. HttpServletRequest request
  202. ) {
  203. // 获取查询条件(前台传递的查询参数)
  204. Map<String, String[]> parameterMap = request.getParameterMap();
  205. // 遍历输出到控制台
  206. System.out.println("\ngetDdjhData - 普通查询条件:");
  207. for (String key : parameterMap.keySet()) {
  208. System.out.println("-- " + key + ": " + JSON.toJSONString(parameterMap.get(key)));
  209. }
  210. // 输出高级查询
  211. try {
  212. System.out.println("\ngetDdjhData - 高级查询条件:");
  213. // 高级查询模式
  214. MatchTypeEnum matchType = MatchTypeEnum.getByValue(superQueryMatchType);
  215. if (matchType == null) {
  216. System.out.println("-- 高级查询模式:不识别(" + superQueryMatchType + ")");
  217. } else {
  218. System.out.println("-- 高级查询模式:" + matchType.getValue());
  219. }
  220. superQueryParams = URLDecoder.decode(superQueryParams, "UTF-8");
  221. List<QueryCondition> conditions = JSON.parseArray(superQueryParams, QueryCondition.class);
  222. if (conditions != null) {
  223. for (QueryCondition condition : conditions) {
  224. System.out.println("-- " + JSON.toJSONString(condition));
  225. }
  226. } else {
  227. System.out.println("-- 没有传递任何高级查询条件");
  228. }
  229. System.out.println();
  230. } catch (Exception e) {
  231. log.error("-- 高级查询操作失败:" + superQueryParams, e);
  232. e.printStackTrace();
  233. }
  234. /* 注:实际使用中不用写上面那种繁琐的代码,这里只是为了直观的输出到控制台里而写的示例,
  235. 使用下面这种写法更简洁方便 */
  236. // 封装成 MyBatisPlus 能识别的 QueryWrapper,可以直接使用这个对象进行SQL筛选条件拼接
  237. // 这个方法也会自动封装高级查询条件,但是高级查询参数名必须是superQueryParams和superQueryMatchType
  238. QueryWrapper<MockEntity> queryWrapper = QueryGenerator.initQueryWrapper(mockEntity, parameterMap);
  239. System.out.println("queryWrapper: " + queryWrapper.getCustomSqlSegment());
  240. // 模拟JSON数据路径
  241. String path = "classpath:org/jeecg/modules/dlglong/json/ddjh.json";
  242. String statusValue = "8";
  243. if (statusValue.equals(status)) {
  244. path = "classpath:org/jeecg/modules/dlglong/json/ddjh_s8.json";
  245. }
  246. // 读取JSON数据
  247. JSONArray dataList = readJsonData(path);
  248. if (dataList == null) {
  249. return Result.error("读取数据失败!");
  250. }
  251. IPage<JSONObject> page = this.queryDataPage(dataList, parentId, pageNo, pageSize);
  252. // 逐行查询子表数据,用于计算拖轮状态
  253. List<JSONObject> records = page.getRecords();
  254. for (JSONObject record : records) {
  255. Map<String, Integer> tugStatusMap = new HashMap<>(5);
  256. String id = record.getString("id");
  257. // 查询出主表的拖轮
  258. String tugMain = record.getString("tug");
  259. // 判断是否有值
  260. if (StringUtils.isNotBlank(tugMain)) {
  261. // 拖轮根据分号分割
  262. String[] tugs = tugMain.split(";");
  263. // 查询子表数据
  264. List<JSONObject> subRecords = this.queryDataPage(dataList, id, null, null).getRecords();
  265. // 遍历子表和拖轮数据,找出进行计算反推拖轮状态
  266. for (JSONObject subData : subRecords) {
  267. String subTug = subData.getString("tug");
  268. if (StringUtils.isNotBlank(subTug)) {
  269. for (String tug : tugs) {
  270. if (tug.equals(subTug)) {
  271. // 计算拖轮状态逻辑
  272. int statusCode = 0;
  273. /* 如果有发船时间、作业开始时间、作业结束时间、回船时间,则主表中的拖轮列中的每个拖轮背景色要即时变色 */
  274. // 有发船时间,状态 +1
  275. String departureTime = subData.getString("departure_time");
  276. if (StringUtils.isNotBlank(departureTime)) {
  277. statusCode += 1;
  278. }
  279. // 有作业开始时间,状态 +1
  280. String workBeginTime = subData.getString("work_begin_time");
  281. if (StringUtils.isNotBlank(workBeginTime)) {
  282. statusCode += 1;
  283. }
  284. // 有作业结束时间,状态 +1
  285. String workEndTime = subData.getString("work_end_time");
  286. if (StringUtils.isNotBlank(workEndTime)) {
  287. statusCode += 1;
  288. }
  289. // 有回船时间,状态 +1
  290. String returnTime = subData.getString("return_time");
  291. if (StringUtils.isNotBlank(returnTime)) {
  292. statusCode += 1;
  293. }
  294. // 保存拖轮状态,key是拖轮的值,value是状态,前端根据不同的状态码,显示不同的颜色,这个颜色也可以后台计算完之后返回给前端直接使用
  295. tugStatusMap.put(tug, statusCode);
  296. break;
  297. }
  298. }
  299. }
  300. }
  301. }
  302. // 新加一个字段用于保存拖轮状态,不要直接覆盖原来的,这个字段可以不保存到数据库里
  303. record.put("tug_status", tugStatusMap);
  304. }
  305. page.setRecords(records);
  306. return Result.ok(page);
  307. }
  308. /**
  309. * 模拟查询数据,可以根据父ID查询,可以分页
  310. *
  311. * @param dataList 数据列表
  312. * @param parentId 父ID
  313. * @param pageNo 页码
  314. * @param pageSize 页大小
  315. * @return
  316. */
  317. private IPage<JSONObject> queryDataPage(JSONArray dataList, String parentId, Integer pageNo, Integer pageSize) {
  318. // 根据父级id查询子级
  319. JSONArray dataDb = dataList;
  320. if (StringUtils.isNotBlank(parentId)) {
  321. JSONArray results = new JSONArray();
  322. List<String> parentIds = Arrays.asList(parentId.split(","));
  323. this.queryByParentId(dataDb, parentIds, results);
  324. dataDb = results;
  325. }
  326. // 模拟分页(实际中应用SQL自带的分页)
  327. List<JSONObject> records = new ArrayList<>();
  328. IPage<JSONObject> page;
  329. long beginIndex, endIndex;
  330. // 如果任意一个参数为null,则不分页
  331. if (pageNo == null || pageSize == null) {
  332. page = new Page<>(0, dataDb.size());
  333. beginIndex = 0;
  334. endIndex = dataDb.size();
  335. } else {
  336. page = new Page<>(pageNo, pageSize);
  337. beginIndex = page.offset();
  338. endIndex = page.offset() + page.getSize();
  339. }
  340. for (long i = beginIndex; (i < endIndex && i < dataDb.size()); i++) {
  341. JSONObject data = dataDb.getJSONObject((int) i);
  342. data = JSON.parseObject(data.toJSONString());
  343. // 不返回 children
  344. data.remove("children");
  345. records.add(data);
  346. }
  347. page.setRecords(records);
  348. page.setTotal(dataDb.size());
  349. return page;
  350. }
  351. private void queryByParentId(JSONArray dataList, List<String> parentIds, JSONArray results) {
  352. for (int i = 0; i < dataList.size(); i++) {
  353. JSONObject data = dataList.getJSONObject(i);
  354. JSONArray children = data.getJSONArray("children");
  355. // 找到了该父级
  356. if (parentIds.contains(data.getString("id"))) {
  357. if (children != null) {
  358. // addAll 的目的是将多个子表的数据合并在一起
  359. results.addAll(children);
  360. }
  361. } else {
  362. if (children != null) {
  363. queryByParentId(children, parentIds, results);
  364. }
  365. }
  366. }
  367. results.addAll(new JSONArray());
  368. }
  369. private JSONArray readJsonData(String path) {
  370. try {
  371. InputStream stream = getClass().getClassLoader().getResourceAsStream(path.replace("classpath:", ""));
  372. if (stream != null) {
  373. String json = IOUtils.toString(stream, "UTF-8");
  374. return JSON.parseArray(json);
  375. }
  376. } catch (IOException e) {
  377. log.error(e.getMessage(), e);
  378. }
  379. return null;
  380. }
  381. }