index.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425
  1. <template>
  2. <div>
  3. <el-tabs type="border-card">
  4. <el-tab-pane label="秒" v-if="shouldHide('second')">
  5. <CrontabSecond @update="updateContabValue" :check="checkNumber" ref="cronsecond" />
  6. </el-tab-pane>
  7. <el-tab-pane label="分钟" v-if="shouldHide('min')">
  8. <CrontabMin
  9. @update="updateContabValue"
  10. :check="checkNumber"
  11. :cron="contabValueObj"
  12. ref="cronmin"
  13. />
  14. </el-tab-pane>
  15. <el-tab-pane label="小时" v-if="shouldHide('hour')">
  16. <CrontabHour
  17. @update="updateContabValue"
  18. :check="checkNumber"
  19. :cron="contabValueObj"
  20. ref="cronhour"
  21. />
  22. </el-tab-pane>
  23. <el-tab-pane label="日" v-if="shouldHide('day')">
  24. <CrontabDay
  25. @update="updateContabValue"
  26. :check="checkNumber"
  27. :cron="contabValueObj"
  28. ref="cronday"
  29. />
  30. </el-tab-pane>
  31. <el-tab-pane label="月" v-if="shouldHide('mouth')">
  32. <CrontabMouth
  33. @update="updateContabValue"
  34. :check="checkNumber"
  35. :cron="contabValueObj"
  36. ref="cronmouth"
  37. />
  38. </el-tab-pane>
  39. <el-tab-pane label="周" v-if="shouldHide('week')">
  40. <CrontabWeek
  41. @update="updateContabValue"
  42. :check="checkNumber"
  43. :cron="contabValueObj"
  44. ref="cronweek"
  45. />
  46. </el-tab-pane>
  47. <el-tab-pane label="年" v-if="shouldHide('year')">
  48. <CrontabYear
  49. @update="updateContabValue"
  50. :check="checkNumber"
  51. :cron="contabValueObj"
  52. ref="cronyear"
  53. />
  54. </el-tab-pane>
  55. </el-tabs>
  56. <div class="popup-main">
  57. <div class="popup-result">
  58. <p class="title">时间表达式</p>
  59. <table>
  60. <thead>
  61. <th v-for="item of tabTitles" width="40" :key="item">{{item}}</th>
  62. <th>Cron 表达式</th>
  63. </thead>
  64. <tbody>
  65. <td>
  66. <span>{{contabValueObj.second}}</span>
  67. </td>
  68. <td>
  69. <span>{{contabValueObj.min}}</span>
  70. </td>
  71. <td>
  72. <span>{{contabValueObj.hour}}</span>
  73. </td>
  74. <td>
  75. <span>{{contabValueObj.day}}</span>
  76. </td>
  77. <td>
  78. <span>{{contabValueObj.mouth}}</span>
  79. </td>
  80. <td>
  81. <span>{{contabValueObj.week}}</span>
  82. </td>
  83. <td>
  84. <span>{{contabValueObj.year}}</span>
  85. </td>
  86. <td>
  87. <span>{{contabValueString}}</span>
  88. </td>
  89. </tbody>
  90. </table>
  91. </div>
  92. <CrontabResult :ex="contabValueString"></CrontabResult>
  93. <div class="pop_btn">
  94. <el-button size="small" type="primary" @click="submitFill">确定</el-button>
  95. <el-button size="small" type="warning" @click="clearCron">重置</el-button>
  96. <el-button size="small" @click="hidePopup">取消</el-button>
  97. </div>
  98. </div>
  99. </div>
  100. </template>
  101. <script>
  102. import CrontabSecond from "./second.vue";
  103. import CrontabMin from "./min.vue";
  104. import CrontabHour from "./hour.vue";
  105. import CrontabDay from "./day.vue";
  106. import CrontabMouth from "./mouth.vue";
  107. import CrontabWeek from "./week.vue";
  108. import CrontabYear from "./year.vue";
  109. import CrontabResult from "./result.vue";
  110. export default {
  111. data() {
  112. return {
  113. tabTitles: ["秒", "分钟", "小时", "日", "月", "周", "年"],
  114. tabActive: 0,
  115. myindex: 0,
  116. contabValueObj: {
  117. second: "*",
  118. min: "*",
  119. hour: "*",
  120. day: "*",
  121. mouth: "*",
  122. week: "?",
  123. year: "",
  124. },
  125. };
  126. },
  127. name: "vcrontab",
  128. props: ["expression", "hideComponent"],
  129. methods: {
  130. shouldHide(key) {
  131. if (this.hideComponent && this.hideComponent.includes(key)) return false;
  132. return true;
  133. },
  134. resolveExp() {
  135. //反解析 表达式
  136. if (this.expression) {
  137. let arr = this.expression.split(" ");
  138. if (arr.length >= 6) {
  139. //6 位以上是合法表达式
  140. let obj = {
  141. second: arr[0],
  142. min: arr[1],
  143. hour: arr[2],
  144. day: arr[3],
  145. mouth: arr[4],
  146. week: arr[5],
  147. year: arr[6] ? arr[6] : "",
  148. };
  149. this.contabValueObj = {
  150. ...obj,
  151. };
  152. for (let i in obj) {
  153. if (obj[i]) this.changeRadio(i, obj[i]);
  154. }
  155. }
  156. } else {
  157. //没有传入的表达式 则还原
  158. this.clearCron();
  159. }
  160. },
  161. // tab切换值
  162. tabCheck(index) {
  163. this.tabActive = index;
  164. },
  165. // 由子组件触发,更改表达式组成的字段值
  166. updateContabValue(name, value, from) {
  167. "updateContabValue", name, value, from;
  168. this.contabValueObj[name] = value;
  169. if (from && from !== name) {
  170. console.log(`来自组件 ${from} 改变了 ${name} ${value}`);
  171. this.changeRadio(name, value);
  172. }
  173. },
  174. //赋值到组件
  175. changeRadio(name, value) {
  176. let arr = ["second", "min", "hour", "mouth"],
  177. refName = "cron" + name,
  178. insVlaue;
  179. if (!this.$refs[refName]) return;
  180. if (arr.includes(name)) {
  181. if (value === "*") {
  182. insVlaue = 1;
  183. } else if (value.indexOf("-") > -1) {
  184. let indexArr = value.split("-");
  185. isNaN(indexArr[0])
  186. ? (this.$refs[refName].cycle01 = 0)
  187. : (this.$refs[refName].cycle01 = indexArr[0]);
  188. this.$refs[refName].cycle02 = indexArr[1];
  189. insVlaue = 2;
  190. } else if (value.indexOf("/") > -1) {
  191. let indexArr = value.split("/");
  192. isNaN(indexArr[0])
  193. ? (this.$refs[refName].average01 = 0)
  194. : (this.$refs[refName].average01 = indexArr[0]);
  195. this.$refs[refName].average02 = indexArr[1];
  196. insVlaue = 3;
  197. } else {
  198. insVlaue = 4;
  199. this.$refs[refName].checkboxList = value.split(",");
  200. }
  201. } else if (name == "day") {
  202. if (value === "*") {
  203. insVlaue = 1;
  204. } else if (value == "?") {
  205. insVlaue = 2;
  206. } else if (value.indexOf("-") > -1) {
  207. let indexArr = value.split("-");
  208. isNaN(indexArr[0])
  209. ? (this.$refs[refName].cycle01 = 0)
  210. : (this.$refs[refName].cycle01 = indexArr[0]);
  211. this.$refs[refName].cycle02 = indexArr[1];
  212. insVlaue = 3;
  213. } else if (value.indexOf("/") > -1) {
  214. let indexArr = value.split("/");
  215. isNaN(indexArr[0])
  216. ? (this.$refs[refName].average01 = 0)
  217. : (this.$refs[refName].average01 = indexArr[0]);
  218. this.$refs[refName].average02 = indexArr[1];
  219. insVlaue = 4;
  220. } else if (value.indexOf("W") > -1) {
  221. let indexArr = value.split("W");
  222. isNaN(indexArr[0])
  223. ? (this.$refs[refName].workday = 0)
  224. : (this.$refs[refName].workday = indexArr[0]);
  225. insVlaue = 5;
  226. } else if (value === "L") {
  227. insVlaue = 6;
  228. } else {
  229. this.$refs[refName].checkboxList = value.split(",");
  230. insVlaue = 7;
  231. }
  232. } else if (name == "week") {
  233. if (value === "*") {
  234. insVlaue = 1;
  235. } else if (value == "?") {
  236. insVlaue = 2;
  237. } else if (value.indexOf("-") > -1) {
  238. let indexArr = value.split("-");
  239. isNaN(indexArr[0])
  240. ? (this.$refs[refName].cycle01 = 0)
  241. : (this.$refs[refName].cycle01 = indexArr[0]);
  242. this.$refs[refName].cycle02 = indexArr[1];
  243. insVlaue = 3;
  244. } else if (value.indexOf("#") > -1) {
  245. let indexArr = value.split("#");
  246. isNaN(indexArr[0])
  247. ? (this.$refs[refName].average01 = 1)
  248. : (this.$refs[refName].average01 = indexArr[0]);
  249. this.$refs[refName].average02 = indexArr[1];
  250. insVlaue = 4;
  251. } else if (value.indexOf("L") > -1) {
  252. let indexArr = value.split("L");
  253. isNaN(indexArr[0])
  254. ? (this.$refs[refName].weekday = 1)
  255. : (this.$refs[refName].weekday = indexArr[0]);
  256. insVlaue = 5;
  257. } else {
  258. this.$refs[refName].checkboxList = value.split(",");
  259. insVlaue = 7;
  260. }
  261. } else if (name == "year") {
  262. if (value == "") {
  263. insVlaue = 1;
  264. } else if (value == "*") {
  265. insVlaue = 2;
  266. } else if (value.indexOf("-") > -1) {
  267. insVlaue = 3;
  268. } else if (value.indexOf("/") > -1) {
  269. insVlaue = 4;
  270. } else {
  271. this.$refs[refName].checkboxList = value.split(",");
  272. insVlaue = 5;
  273. }
  274. }
  275. this.$refs[refName].radioValue = insVlaue;
  276. },
  277. // 表单选项的子组件校验数字格式(通过-props传递)
  278. checkNumber(value, minLimit, maxLimit) {
  279. //检查必须为整数
  280. value = Math.floor(value);
  281. if (value < minLimit) {
  282. value = minLimit;
  283. } else if (value > maxLimit) {
  284. value = maxLimit;
  285. }
  286. return value;
  287. },
  288. // 隐藏弹窗
  289. hidePopup() {
  290. this.$emit("hide");
  291. },
  292. // 填充表达式
  293. submitFill() {
  294. this.$emit("fill", this.contabValueString);
  295. this.hidePopup();
  296. },
  297. clearCron() {
  298. // 还原选择项
  299. ("准备还原");
  300. this.contabValueObj = {
  301. second: "*",
  302. min: "*",
  303. hour: "*",
  304. day: "*",
  305. mouth: "*",
  306. week: "?",
  307. year: "",
  308. };
  309. for (let j in this.contabValueObj) {
  310. this.changeRadio(j, this.contabValueObj[j]);
  311. }
  312. },
  313. },
  314. computed: {
  315. contabValueString: function() {
  316. let obj = this.contabValueObj;
  317. let str =
  318. obj.second +
  319. " " +
  320. obj.min +
  321. " " +
  322. obj.hour +
  323. " " +
  324. obj.day +
  325. " " +
  326. obj.mouth +
  327. " " +
  328. obj.week +
  329. (obj.year == "" ? "" : " " + obj.year);
  330. return str;
  331. },
  332. },
  333. components: {
  334. CrontabSecond,
  335. CrontabMin,
  336. CrontabHour,
  337. CrontabDay,
  338. CrontabMouth,
  339. CrontabWeek,
  340. CrontabYear,
  341. CrontabResult,
  342. },
  343. watch: {
  344. expression: "resolveExp",
  345. hideComponent(value) {
  346. // 隐藏部分组件
  347. },
  348. },
  349. mounted: function() {
  350. this.resolveExp();
  351. },
  352. };
  353. </script>
  354. <style scoped>
  355. .pop_btn {
  356. text-align: center;
  357. margin-top: 20px;
  358. }
  359. .popup-main {
  360. position: relative;
  361. margin: 10px auto;
  362. background: #fff;
  363. border-radius: 5px;
  364. font-size: 12px;
  365. overflow: hidden;
  366. }
  367. .popup-title {
  368. overflow: hidden;
  369. line-height: 34px;
  370. padding-top: 6px;
  371. background: #f2f2f2;
  372. }
  373. .popup-result {
  374. box-sizing: border-box;
  375. line-height: 24px;
  376. margin: 25px auto;
  377. padding: 15px 10px 10px;
  378. border: 1px solid #ccc;
  379. position: relative;
  380. }
  381. .popup-result .title {
  382. position: absolute;
  383. top: -28px;
  384. left: 50%;
  385. width: 140px;
  386. font-size: 14px;
  387. margin-left: -70px;
  388. text-align: center;
  389. line-height: 30px;
  390. background: #fff;
  391. }
  392. .popup-result table {
  393. text-align: center;
  394. width: 100%;
  395. margin: 0 auto;
  396. }
  397. .popup-result table span {
  398. display: block;
  399. width: 100%;
  400. font-family: arial;
  401. line-height: 30px;
  402. height: 30px;
  403. white-space: nowrap;
  404. overflow: hidden;
  405. border: 1px solid #e8e8e8;
  406. }
  407. .popup-result-scroll {
  408. font-size: 12px;
  409. line-height: 24px;
  410. height: 10em;
  411. overflow-y: auto;
  412. }
  413. </style>