computedDialog.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735
  1. <template>
  2. <el-dialog
  3. :visible.sync="isOpenComputed"
  4. :close-on-click-modal="false"
  5. :modal-append-to-body="false"
  6. @close="cancelHandle"
  7. custom-class="computed-dialog"
  8. center
  9. width="1090px"
  10. top="8vh"
  11. v-dialogDrag
  12. >
  13. <div slot="title" style="display: flex; align-items: center">
  14. <img
  15. :src="$icons.computed"
  16. style="color: #fff; width: 16px; height: 16px; margin-right: 5px"
  17. />
  18. <span style="font-size: 16px">{{ title }}</span>
  19. </div>
  20. <div class="dialog-main">
  21. <ul class="add-cont">
  22. <li class="add-li" v-for="(list, index) in addList" :key="index">
  23. <span class="li-tag">{{ list.tag }}</span>
  24. <el-select
  25. v-model="list.target"
  26. v-loadMore="searchLoad"
  27. :filterable="!list.target"
  28. clearable
  29. placeholder="请输入指标名称"
  30. style="width: 400px"
  31. @change="chooseTarget"
  32. @clear="clearHandle(index)"
  33. :disabled="calulateForm.view"
  34. remote
  35. :remote-method="searchTarget"
  36. @click.native="inputFocusHandle"
  37. >
  38. <i slot="prefix" class="el-input__icon el-icon-search"></i>
  39. <el-option
  40. v-for="item in searchOptions"
  41. :key="item.EdbInfoId"
  42. :label="$parent.currentLang==='en'?(item.EdbNameEn||item.EdbName):item.EdbName"
  43. :value="item.EdbInfoId"
  44. >
  45. <edbDetailPopover :info="item">
  46. <div slot="reference">{{$parent.currentLang==='en'?(item.EdbNameEn||item.EdbName):item.EdbName}}</div>
  47. </edbDetailPopover>
  48. </el-option>
  49. </el-select>
  50. <i class="el-icon-tickets" style="color:#409EFF;font-size:18px" @click="$emit('lookHistory',list.target)" v-if="list.target"/>
  51. <i
  52. class="el-icon-error del-tag"
  53. v-if="index > 3 && !calulateForm.view"
  54. @click="delTarget(index)"
  55. />
  56. <span :class="{
  57. 'target-date': true,
  58. 'newest': list.end_date===newestDate
  59. }" v-if="list.start_date">{{
  60. `${list.start_date}至${list.end_date}`
  61. }}</span>
  62. </li>
  63. </ul>
  64. <span class="add-icon" @click="addTargetHandle" v-if="!calulateForm.view">
  65. <i
  66. class="el-icon-circle-plus-outline"
  67. style="color: #5882ef; font-size: 16px"
  68. />
  69. 添加更多参数
  70. </span>
  71. <div class="computed-min">
  72. <div class="computed-section">
  73. <div>
  74. <label class="label">空值处理
  75. <el-tooltip placement="top">
  76. <div slot="content" v-html="formTips['null-val']" style="width:300px;line-height:20px;"/>
  77. <i class="el-icon-question"/>
  78. </el-tooltip>
  79. </label>
  80. <el-select
  81. v-model="nullValueForm.nullValueWay"
  82. placeholder="请选择"
  83. :disabled="calulateForm.view"
  84. >
  85. <el-option
  86. v-for="item in nullWayOptions"
  87. :key="item.value"
  88. :label="item.label"
  89. :value="item.value"
  90. >
  91. </el-option>
  92. </el-select>
  93. </div>
  94. <div style="margin-left: 120px" v-if="showMaxNullDeal">
  95. <label class="label">MAX、MIN空值处理
  96. <el-tooltip placement="top">
  97. <div slot="content" v-html="formTips['max-null-val']" style="width:300px;line-height:20px;"/>
  98. <i class="el-icon-question"/>
  99. </el-tooltip>
  100. </label>
  101. <el-select
  102. v-model="nullValueForm.maxNullWay"
  103. placeholder="请选择"
  104. :disabled="calulateForm.view"
  105. >
  106. <el-option label="等于0" :value="1" />
  107. <el-option label="跳过空值" :value="2" />
  108. </el-select>
  109. </div>
  110. </div>
  111. <div class="computed-section">
  112. <label class="label">计算公式
  113. <el-tooltip placement="top">
  114. <div slot="content" v-html="formTips['formula']" style="width:300px;line-height:20px;"/>
  115. <i class="el-icon-question"/>
  116. </el-tooltip>
  117. </label>
  118. <!-- <el-input placeholder="请输入公式" v-model="formula" clearable :disabled="calulateForm.view">
  119. </el-input> -->
  120. <ul class="formula-list">
  121. <li style="margin-bottom: 15px;">
  122. <el-input placeholder="请输入公式" v-model="formulaList[0].formula" clearable :disabled="calulateForm.view" style="width: 220px"/>
  123. <span v-if="formulaDateArr.length" class="date-section-text">{{formulaDateArr[formulaDateArr.length-1]}}(含)之后</span>
  124. <span class="example-txt">公式示例:A*0.5+B*C*1.2+120-MAX(A,B,C) &nbsp;函数支持:MAX(),MIN(),ln(A),log(a,A),abs(),exp(),pow(),round()</span>
  125. </li>
  126. <li class="formula-item" v-for="(item,index) in formulaList.slice(1)" :key="index+1">
  127. <el-input
  128. placeholder="请输入公式"
  129. v-model="item.formula"
  130. clearable
  131. :disabled="calulateForm.view"
  132. style="width: 220px"
  133. />
  134. <el-date-picker
  135. v-model="item.date"
  136. type="date"
  137. value-format="yyyy-MM-dd"
  138. style="margin: 0 10px;width: 220px"
  139. placeholder="选择日期"
  140. :disabled="calulateForm.view"
  141. @change="selectFormulaDate($event,item)"
  142. />
  143. <i class="el-icon-circle-close" style="font-size:20px;" v-if="!calulateForm.view" @click="removeFormulaItem(index+1)"/>
  144. <template v-if="formulaDateArr.length&&item.date">
  145. <span v-if="item.date===formulaDateArr[0]" class="date-section-text">{{formulaDateArr[0]}}之前</span>
  146. <span v-else class="date-section-text">{{formulaDateArr[formulaDateArr.findIndex(_ =>_ ===item.date)-1]}}(含)——{{item.date}}</span>
  147. </template>
  148. </li>
  149. </ul>
  150. </div>
  151. <el-button
  152. v-if="!calulateForm.view"
  153. icon="el-icon-plus"
  154. style="margin-left:70px;"
  155. @click="addFormulaHandle"
  156. >新增分段</el-button>
  157. </div>
  158. <el-form
  159. ref="diaForm"
  160. label-position="right"
  161. inline
  162. label-width="80px"
  163. :model="formData"
  164. :rules="formRules"
  165. :disabled="calulateForm.view"
  166. >
  167. <el-form-item label="指标名称" prop="targetName">
  168. <el-input
  169. v-model="formData.targetName"
  170. style="width: 340px"
  171. placeholder="请输入指标名称"
  172. />
  173. </el-form-item>
  174. <el-form-item label="单位" prop="unit">
  175. <selectUnit v-model="formData.unit" style="width: 340px" />
  176. </el-form-item>
  177. <el-form-item label="指标目录" prop="menu">
  178. <el-cascader
  179. v-model="formData.menu"
  180. :options="options"
  181. :props="levelProps"
  182. clearable
  183. placeholder="请选择指标目录"
  184. />
  185. </el-form-item>
  186. <el-form-item label="频度" prop="frequency">
  187. <el-select
  188. v-model="formData.frequency"
  189. placeholder="请选择频率"
  190. style="width: 340px"
  191. clearable
  192. >
  193. <el-option
  194. v-for="item in frequencyArr"
  195. :key="item"
  196. :label="item"
  197. :value="item"
  198. >
  199. </el-option>
  200. </el-select>
  201. </el-form-item>
  202. </el-form>
  203. </div>
  204. <div class="dia-bot" v-if="!calulateForm.view">
  205. <el-button
  206. type="primary"
  207. style="margin-right: 20px"
  208. @click="saveHandle"
  209. :loading="dataloading"
  210. v-if="title==='计算指标'"
  211. >生成计算指标</el-button
  212. >
  213. <el-button
  214. type="primary"
  215. style="margin-right: 20px"
  216. @click="saveHandle"
  217. v-else
  218. >保存</el-button
  219. >
  220. <el-button type="primary" plain @click="cancelHandle('cancel')">取消</el-button>
  221. </div>
  222. <el-popover
  223. placement="top-start"
  224. width="360"
  225. trigger="click">
  226. <p style="padding:30px;line-height:25px;" v-html="$parent.tips.get(type)"/>
  227. <span slot="reference" class="tip-label">公式说明</span>
  228. </el-popover>
  229. </el-dialog>
  230. </template>
  231. <script>
  232. import { dataBaseInterface } from '@/api/api.js';
  233. import * as preDictEdbInterface from '@/api/modules/predictEdbApi.js';
  234. import { formRules } from '../databaseComponents/util';
  235. import { unitArr } from '@/utils/defaultOptions';
  236. const tag_arr = [];
  237. for(let i=0;i<26;i++) tag_arr.push(String.fromCharCode(65+i));
  238. export default {
  239. name: '',
  240. props: {
  241. isOpenComputed: {
  242. type: Boolean,
  243. },
  244. title: {
  245. type: String,
  246. default: '计算指标',
  247. },
  248. calulateForm: {
  249. type: Object,
  250. },
  251. calulateList: {
  252. type: Array,
  253. },
  254. edbSource: { // ''||'predict'
  255. type: String,
  256. default: ''
  257. }
  258. },
  259. computed: {
  260. type() {
  261. return this.edbSource ? 31 : 4;
  262. },
  263. /* max空值处理显示 当输入的公式包含MAX、MIN且空值处理为0时,输入公式失焦后出现右侧选项; */
  264. showMaxNullDeal() {
  265. let haveMaxOrMin = this.formulaList.some(_ => _.formula.toUpperCase().includes('MAX') || _.formula.toUpperCase().includes('MIN'))
  266. return haveMaxOrMin && this.nullValueForm.nullValueWay===4
  267. },
  268. formulaDateArr() {
  269. return this.formulaList.map(_ => _.date).filter(_ => _).sort((a,b) => new Date(a)-new Date(b))
  270. }
  271. },
  272. watch: {
  273. isOpenComputed(newval) {
  274. newval && this.getMenu();
  275. // newval && this.searchTarget('');
  276. /* 回显 */
  277. if (this.calulateList.length && newval) {
  278. this.addList = _.cloneDeep(this.calulateList);
  279. this.formulaList = JSON.parse(this.calulateForm.formula).map(_ => ({
  280. formula: _.f,
  281. date: _.d||''
  282. }));
  283. this.formData = {
  284. targetName: this.calulateForm.targetName,
  285. unit: this.calulateForm.unit,
  286. menu: this.calulateForm.menu,
  287. frequency: this.calulateForm.frequency,
  288. };
  289. this.nullValueForm = {
  290. nullValueWay: this.calulateForm.emptyType,
  291. maxNullWay: this.calulateForm.maxEmptyType,
  292. }
  293. this.searchOptions = this.calulateList.map(item => ({
  294. EdbInfoId: item.target,
  295. EdbName: item.edb_name,
  296. }))
  297. this.getNewestDate();
  298. }
  299. },
  300. },
  301. data() {
  302. return {
  303. options: [],
  304. levelProps: {
  305. label: 'ClassifyName',
  306. value: 'ClassifyId',
  307. children: 'Children',
  308. checkStrictly: true
  309. },
  310. frequencyArr: ['日度', '周度','旬度','月度', '季度', '年度'],
  311. formRules,
  312. unitArr,
  313. addList: [
  314. {
  315. tag: tag_arr[0],
  316. target: '',
  317. start_date: '',
  318. end_date: '',
  319. },
  320. {
  321. tag: tag_arr[1],
  322. target: '',
  323. start_date: '',
  324. end_date: '',
  325. },
  326. {
  327. tag: tag_arr[2],
  328. target: '',
  329. start_date: '',
  330. end_date: '',
  331. },
  332. {
  333. tag: tag_arr[3],
  334. target: '',
  335. start_date: '',
  336. end_date: '',
  337. },
  338. ],
  339. searchOptions: [],
  340. formula: '', //计算公式
  341. formulaList: [ //公式数组
  342. { formula: '', }
  343. ],
  344. dataloading: false,
  345. formData: {
  346. targetName: '',
  347. unit: '',
  348. menu: '',
  349. frequency: '',
  350. },
  351. search_have_more: false,
  352. search_page: 1,
  353. current_search:'',
  354. newestDate: '',
  355. nullValueForm: {
  356. nullValueWay: 0,
  357. maxNullWay: 1
  358. },//空值处理
  359. nullWayOptions: [
  360. { label: '查找前后35天最近值',value: 0 },
  361. { label: '不计算',value: 1 },
  362. { label: '前值填充',value: 2 },
  363. { label: '后值填充',value: 3 },
  364. { label: '等于0',value: 4 },
  365. ],
  366. formTips: {
  367. 'null-val': `1、查找前后35天最近值:在参与计算的日期序列上某指标无值时,该指标往前/往后找距离最近的值作为当天的值进行计算,遍历允许跨年,往前最多35天,往后最多35天<br>
  368. 2、不计算:只要有一个指标在某个日期没有值(即空值),则计算指标在该日期没有值 <br>
  369. 3、前值填充:空值优先以最近的前值填充,没有前值时,用后值填充 <br>
  370. 4、后值填充:空值优先以最近的后值填充,没有前值时,用后值填充 <br>
  371. 5、等于0:空值以0值参与计算 <br>
  372. 注意:此处缺失值的处理,作用于数据全部时间段`,
  373. 'max-null-val': `MAX、MIN公式中指标存在空值时按如下规则处理:<br>
  374. 1、等于0,空值用0参与计算;<br>
  375. 2、跳过空值,去除空值指标,剩余指标进行计算,若该日期所有指标均为空值,则该日期无值;`,
  376. 'formula':`1、支持新增分段,实现不同分段使用不同的计算公式,若未新增分段,则所有日期序列用统一公式计算<br>
  377. 2、新增分段需配置新公式和时间节点,在时间节点之前(不含)使用新公式,在时间节点之后(含)使用已配置公式,每个分段公式支持修改<br>
  378. 3、分段时间节点不允许重复,不允许超出第一个指标的日期区间`
  379. }
  380. };
  381. },
  382. methods: {
  383. /* 获取目录结构 */
  384. async getMenu() {
  385. const res = this.edbSource === 'predict'
  386. ? await preDictEdbInterface.classifyListV2()
  387. : await dataBaseInterface.menuListV3()
  388. if (res.Ret !== 200) return
  389. this.filterNodes(res.Data.AllNodes||[]);
  390. this.options = res.Data.AllNodes || [];
  391. },
  392. // 递归改变第三级目录结构
  393. filterNodes(arr) {
  394. arr.length &&
  395. arr.forEach((item) => {
  396. item.Children.length && this.filterNodes(item.Children);
  397. if (!item.Children.length) {
  398. delete item.Children;
  399. }
  400. });
  401. },
  402. /* 添加额外的指标列 */
  403. addTargetHandle() {
  404. if(this.addList.length >= 26) return this.$message.warning('添加指标个数已达上限')
  405. let tag = this.addList[this.addList.length-1].tag;
  406. let index = tag_arr.findIndex(item => item === tag);
  407. const item = {
  408. tag: tag_arr[index+1],
  409. target: '',
  410. start_date: '',
  411. end_date: ''
  412. };
  413. this.addList.push(item);
  414. },
  415. /* 搜索指标 */
  416. searchTarget(query) {
  417. this.search_page = 1;
  418. this.current_search = query;
  419. this.searchApi(this.current_search);
  420. },
  421. /* 聚焦获取当前检索 */
  422. inputFocusHandle(e) {
  423. this.search_page = 1;
  424. this.current_search = e.target.value;
  425. this.searchApi(this.current_search);
  426. },
  427. async searchApi(query,page=1) {
  428. const res = this.edbSource === 'predict'
  429. ? await preDictEdbInterface.edbSearch({
  430. Keyword: query,
  431. CurrentIndex: page,
  432. FilterSource: this.type === 45 ? 3 : this.type === 66 ? 6 : 1,
  433. Frequency: this.type===64?'季度': ''
  434. })
  435. : await dataBaseInterface.targetSearchByPage({
  436. KeyWord:query,
  437. CurrentIndex: page,
  438. FilterSource: this.type === 14 ? 3 : this.type === 63 ? 6 : 1,
  439. Frequency: this.type===61?'季度': ''
  440. })
  441. if(res.Ret !== 200) return
  442. const { List,Paging } = res.Data;
  443. this.search_have_more = page < Paging.Pages;
  444. this.searchOptions = page === 1 ? List : this.searchOptions.concat(List);
  445. },
  446. searchLoad() {
  447. if(!this.search_have_more) return;
  448. this.searchApi(this.current_search,++this.search_page)
  449. },
  450. /* 选中指标 显示开始日期结束日期 */
  451. chooseTarget(val) {
  452. if (val) {
  453. const choose_obj = this.searchOptions.find(
  454. (item) => item.EdbInfoId === val
  455. );
  456. this.addList.forEach((list) => {
  457. if (list.target === val) {
  458. list.start_date = choose_obj.StartDate;
  459. list.end_date = choose_obj.EndDate;
  460. this.getNewestDate();
  461. }
  462. });
  463. }
  464. },
  465. /* 获取所选日期最早日期 */
  466. getNewestDate() {
  467. let dateArr = this.addList.filter(_ => _.end_date).map(_ => new Date(_.end_date).getTime())
  468. let earliestDate = Math.min(...dateArr);
  469. let sameDateArr = dateArr.filter(_ => _===earliestDate)
  470. if(dateArr.length === 1 || sameDateArr.length===dateArr.length) {
  471. this.newestDate = '';
  472. return
  473. }
  474. this.newestDate = this.$moment(Math.min(...dateArr)).format('YYYY-MM-DD');
  475. },
  476. /* 清空指标和关联日期 */
  477. clearHandle(index) {
  478. this.addList[index].start_date = '';
  479. this.addList[index].end_date = '';
  480. this.getNewestDate();
  481. },
  482. // 删除指标
  483. delTarget(index) {
  484. this.addList.splice(index, 1);
  485. this.getNewestDate();
  486. },
  487. async saveHandle() {
  488. if (!this.formulaList[0].formula) return this.$message.warning('计算公式不能为空');
  489. await this.$refs.diaForm.validate();
  490. // 指标id数组
  491. let EdbInfoIdArr = this.addList.filter((item) => item.target).map((item) => ({
  492. EdbInfoId: item.target,
  493. FromTag: item.tag,
  494. }));
  495. let formulaArr = this.formulaList.map(_ => ({f: _.formula,d: _.date}))
  496. const { nullValueWay,maxNullWay } = this.nullValueForm;
  497. let params = {
  498. CalculateFormula: JSON.stringify(formulaArr),
  499. ClassifyId: this.formData.menu[this.formData.menu.length - 1] || 0,
  500. EdbName: this.formData.targetName,
  501. Frequency: this.formData.frequency,
  502. Unit: this.formData.unit,
  503. EdbInfoIdArr,
  504. EmptyType: nullValueWay,
  505. MaxEmptyType: maxNullWay
  506. };
  507. this.dataloading = true;
  508. let res;
  509. if(this.edbSource === 'predict') {
  510. res = await preDictEdbInterface.calculateEdbSave(this.calulateForm.edb_id ? {...params,EdbInfoId:this.calulateForm.edb_id } : params)
  511. }else {
  512. res = this.calulateForm.edb_id
  513. ? await dataBaseInterface.calculateEdit({...params,EdbInfoId: this.calulateForm.edb_id})
  514. : await dataBaseInterface.calculateAdd(params)
  515. }
  516. this.dataloading = false;
  517. if (res.Ret !== 200) return
  518. this.$message.success(res.Msg);
  519. this.calulateForm.edb_id
  520. ? this.$emit('addCallBack','edit')
  521. : this.$emit('addCallBack','add',{ code:res.Data.UniqueCode,id:res.Data.EdbInfoId,classifyId:params.ClassifyId });
  522. this.init();
  523. },
  524. /* 新增公式分段 */
  525. addFormulaHandle() {
  526. let addItem = {
  527. formula: this.formulaList[this.formulaList.length-1].formula,
  528. date: ''
  529. }
  530. this.formulaList.push(addItem)
  531. },
  532. /* 移除公式 */
  533. removeFormulaItem(index) {
  534. this.formulaList.splice(index,1)
  535. },
  536. /* 选择日期 检验排 */
  537. selectFormulaDate(val,item) {
  538. console.log(val,item)
  539. const { start_date,end_date } = this.addList[0];
  540. if(!start_date) return
  541. let dateStamp = new Date(val).getTime(),
  542. startStamp = new Date(start_date).getTime(),
  543. endStamp = new Date(end_date).getTime();
  544. if (dateStamp > endStamp || dateStamp < startStamp) {
  545. item.date = '';
  546. return this.$message.warning('分段日期必须在第一个指标日期区间')
  547. }
  548. else if(this.formulaList.filter(_ => _.date===val).length>1) {
  549. item.date = '';
  550. return this.$message.warning('分段日期不可重复')
  551. }
  552. },
  553. init() {
  554. this.$refs.diaForm.resetFields();
  555. this.addList = [
  556. {
  557. tag: tag_arr[0],
  558. target: '',
  559. start_date: '',
  560. end_date: '',
  561. },
  562. {
  563. tag: tag_arr[1],
  564. target: '',
  565. start_date: '',
  566. end_date: '',
  567. },
  568. {
  569. tag: tag_arr[2],
  570. target: '',
  571. start_date: '',
  572. end_date: '',
  573. },
  574. {
  575. tag: tag_arr[3],
  576. target: '',
  577. start_date: '',
  578. end_date: '',
  579. },
  580. ];
  581. this.searchOptions = [];
  582. this.formula = '';
  583. this.formData = {
  584. targetName: '',
  585. unit: '',
  586. menu: '',
  587. frequency: '',
  588. };
  589. this.formulaList = [
  590. { formula: '' }
  591. ]
  592. this.nullValueForm = {
  593. nullValueWay: 0,
  594. maxNullWay: 1
  595. }
  596. },
  597. cancelHandle(type) {
  598. this.init();
  599. this.$emit('cancel');
  600. type==='cancel' && !this.calulateForm.edb_id && this.$emit('openPrev');
  601. },
  602. },
  603. mounted() {},
  604. };
  605. </script>
  606. <style lang="scss">
  607. .computed-dialog {
  608. overflow: hidden;
  609. div::-webkit-scrollbar {
  610. width: 6px !important;
  611. }
  612. .el-dialog__body {
  613. max-height: 700px;
  614. overflow: auto;
  615. }
  616. .dialog-main {
  617. padding: 25px 42px 25px 25px;
  618. .el-cascader .el-input {
  619. width: 340px;
  620. }
  621. .add-cont {
  622. display: flex;
  623. flex-wrap: wrap;
  624. justify-content: space-between;
  625. .add-li {
  626. position: relative;
  627. margin-bottom: 48px;
  628. .li-tag {
  629. font-size: 16px;
  630. margin-right: 8px;
  631. }
  632. .del-tag {
  633. position: absolute;
  634. right: -30px;
  635. top: 12px;
  636. font-size: 16px;
  637. cursor: pointer;
  638. }
  639. .target-date {
  640. color: #5882ef;
  641. position: absolute;
  642. bottom: -25px;
  643. left: 24px;
  644. &.newest {
  645. color: #f00;
  646. }
  647. }
  648. }
  649. }
  650. .add-icon {
  651. font-size: 16px;
  652. color: #5882ef;
  653. cursor: pointer;
  654. }
  655. .computed-min {
  656. margin: 50px 0;
  657. padding-bottom: 40px;
  658. border-bottom: 1px dashed #aab4cc;
  659. .computed-section {
  660. display: flex;
  661. margin-top: 20px;
  662. }
  663. .label {
  664. padding:10px 10px 10px 0;
  665. }
  666. .example-txt {
  667. display: block;
  668. margin-top: 10px;
  669. }
  670. .formula-item {
  671. display: flex;
  672. align-items: center;
  673. margin-bottom: 15px;
  674. }
  675. .date-section-text {
  676. margin-left: 15px;
  677. }
  678. }
  679. }
  680. .dia-bot {
  681. padding-bottom: 40px;
  682. display: flex;
  683. justify-content: center;
  684. }
  685. .tip-label {
  686. position: absolute;
  687. bottom: 30px;
  688. right: 30px;
  689. color: #409EFF;
  690. cursor: pointer;
  691. }
  692. }
  693. </style>