From f3d4db769ba9b51b799d981511a710fd443d0e08 Mon Sep 17 00:00:00 2001
From: king <18310653075@163.com>
Date: 星期一, 21 四月 2025 12:18:03 +0800
Subject: [PATCH] Merge branch 'master' into positec

---
 src/templates/zshare/verifycard/customscript/index.jsx |  302 ++++++++++++++++----------------------------------
 1 files changed, 97 insertions(+), 205 deletions(-)

diff --git a/src/templates/zshare/verifycard/customscript/index.jsx b/src/templates/zshare/verifycard/customscript/index.jsx
index 6150cb5..3a13b0e 100644
--- a/src/templates/zshare/verifycard/customscript/index.jsx
+++ b/src/templates/zshare/verifycard/customscript/index.jsx
@@ -1,9 +1,8 @@
 import React, {Component} from 'react'
 import PropTypes from 'prop-types'
-import { Form, Row, Col, Button, notification, Modal, Tooltip, Radio, Select, Switch } from 'antd'
+import { Form, Row, Col, Button, notification, Tooltip, Radio, Select, Switch } from 'antd'
 import { QuestionCircleOutlined } from '@ant-design/icons'
 
-import Api from '@/api'
 import { checkSQL } from '@/utils/utils-custom.js'
 import CodeMirror from '@/templates/zshare/codemirror'
 import './index.scss'
@@ -14,11 +13,8 @@
     btn: PropTypes.object,
     formfields: PropTypes.string,
     colfields: PropTypes.string,
-    initsql: PropTypes.string,
-    defaultsql: PropTypes.string,
-    useDefaultSql: PropTypes.any,
+    getSysDefSql: PropTypes.func,
     systemScripts: PropTypes.array,
-    customScripts: PropTypes.array,
     scriptsChange: PropTypes.func
   }
 
@@ -46,11 +42,13 @@
   }
 
   handleConfirm = () => {
-    const { type, workFlow, flowType, flowSql, useDefaultSql, defaultsql } = this.props
+    const { type } = this.props
     const { editItem, skip } = this.state
     // 琛ㄥ崟鎻愪氦鏃舵鏌ヨ緭鍏ュ�兼槸鍚︽纭�
     this.props.form.validateFieldsAndScroll((err, values) => {
-      if (type === 'fullscreen' && err) {
+      if (err) return
+
+      if (!values.sql || /^[\s\n]+$/.test(values.sql)) {
         notification.warning({
           top: 92,
           message: '璇疯緭鍏ql!',
@@ -58,195 +56,43 @@
         })
         return
       }
-      if (!err) {
-        if (/^[\s\n]+$/.test(values.sql)) {
-          notification.warning({
-            top: 92,
-            message: '璇疯緭鍏ql!',
-            duration: 5
-          })
-          return
-        }
-        values.uuid = editItem ? editItem.uuid : ''
-        values.position = values.position || (editItem ? editItem.position : 'front')
 
-        if (type === 'fullscreen' && editItem) {
-          values.status = editItem.status || 'true'
-        }
+      values.uuid = editItem ? editItem.uuid : ''
+      values.position = values.position || (editItem ? editItem.position : 'front')
 
-        let pass = checkSQL(values.sql, 'customscript')
+      if (type === 'fullscreen' && editItem) {
+        values.status = editItem.status || 'true'
+      }
 
-        if (!pass) return
+      let pass = checkSQL(values.sql, 'customscript')
 
-        let tail = `
-          aaa:
-        `
-
-        let _initCustomScript = '' // 鍒濆鍖栬剼鏈�
-        let _prevCustomScript = '' // 榛樿sql鍓嶆墽琛岃剼鏈�
-        let _backCustomScript = '' // 榛樿sql鍚庢墽琛岃剼鏈�
-
-        this.props.customScripts.forEach(item => {
-          let _item = values.uuid === item.uuid ? values : item
-
-          if (_item.status === 'false') return
-
-          if (_item.position === 'init') {
-            _initCustomScript += `
-            /* 鍒濆鍖栬剼鏈� */
-            ${_item.sql}
-            `
-          } else if (_item.position === 'front') {
-            _prevCustomScript += `
-            /* 榛樿sql鍓嶈剼鏈� */
-            ${_item.sql}
-            `
-          } else {
-            _backCustomScript += `
-            /* 榛樿sql鍚庤剼鏈� */
-            ${_item.sql}
-            `
-          }
+      if (!pass && !skip) return
+      
+      if (skip) {
+        this.setState({
+          skip: false,
+          editItem: null
         })
+        this.props.form.setFieldsValue({
+          sql: ''
+        })
+        this.props.scriptsChange(values)
+      } else {
+        this.setState({loading: true})
 
-        if (!values.uuid) {
-          if (values.position === 'init') {
-            _initCustomScript += `
-            /* 鍒濆鍖栬剼鏈� */
-            ${values.sql}
-            `
-          } else if (values.position === 'front') {
-            _prevCustomScript += `
-            /* 榛樿sql鍓嶈剼鏈� */
-            ${values.sql}
-            `
-          } else {
-            _backCustomScript += `
-            /* 榛樿sql鍚庤剼鏈� */
-            ${values.sql}
-            `
-          }
-        }
-
-        let sql = this.props.initsql + _initCustomScript + _prevCustomScript
-
-        if (useDefaultSql) {
-          sql += `\n/* 榛樿sql */
-            ${defaultsql}
-          `
-        }
-
-        if (window.GLOB.process && workFlow === 'true' && flowSql === 'true') {
-          if (flowType === 'start') {
-            sql += `
-              /* 宸ヤ綔娴侀粯璁ql */
-              insert into s_my_works_flow (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,work_group,works_flow_detail_id,work_grade,bid,createuserid,CreateUser,CreateStaff,upid)
-              select @ID@,@works_flow_code@,@works_flow_name@,@works_flow_param@,@status@,@statusname@,@work_group@,@works_flow_detail_id@,@work_grade@,@bid@,@UserID@,@UserName,@FullName,@time_id@
-              insert into s_my_works_flow_log (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,works_flow_detail_id,work_group,work_grade,upid)
-              select @ID@,@works_flow_code@,@works_flow_name@ ,@works_flow_param@,@status@,@statusname@,@works_flow_detail_id@,@work_group@,@work_grade@,@time_id@
-              insert into s_my_works_flow_notice (works_flow_id,works_flow_code,works_flow_detail_id,userid,notice_type,createuserid,CreateUser,CreateStaff,upid)
-              select @ID@,@works_flow_code@,@works_flow_detail_id@,@userid@,@start_type@,@userid@,@UserName,@FullName,@time_id@
-            `
-          } else {
-            sql += `
-              /* 宸ヤ綔娴侀粯璁ql */
-              update s_my_works_flow set status=@status@,statusname=@statusname@,works_flow_param=@works_flow_param@,works_flow_detail_id=@works_flow_detail_id@,modifydate=getdate(),upid=@time_id@,modifyuserid=@userid@,modifyuser=@username,modifystaff=@fullname
-              where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0
-              insert into s_my_works_flow_log (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,works_flow_detail_id,work_group,work_grade,upid)
-              select @ID@,@works_flow_code@,@works_flow_name@ ,@works_flow_param@,@status@,@statusname@,@works_flow_detail_id@,@work_group@,@work_grade@,@time_id@
-              
-              update s_my_works_flow_role set deleted=10,modifydate=getdate(),upid=@time_id@,modifyuserid=@userid@,modifyuser=@username,modifystaff=@fullname 
-              where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0
-
-              if @check_userids@ != ''
-              begin
-                    insert into s_my_works_flow_role (works_flow_id,works_flow_code,userid,works_flow_detail_id,createuserid,CreateUser,CreateStaff,upid)
-                    select @ID@,@works_flow_code@,ID,@works_flow_detail_id@,@userid@,@UserName,@FullName,@time_id@ from dbo.SplitComma(@check_userids@)
-                    insert into s_my_works_flow_notice (works_flow_id,works_flow_code,works_flow_detail_id,userid,notice_type,createuserid,CreateUser,CreateStaff,upid)
-                    select @ID@,@works_flow_code@,@works_flow_detail_id@,ID,@check_type@,@userid@,@UserName,@FullName,@time_id@ from dbo.SplitComma(@check_userids@)
-              end
-              if @notice_userids@ != ''
-              begin
-                    update n
-                    set deleted=10,modifydate=getdate(),upid=@time_id@,modifyuserid=@userid@,modifyuser=@username,modifystaff=@fullname 
-                    from (select * from s_my_works_flow_notice where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0) n
-                    inner join (select ID from dbo.SplitComma(@notice_userids@)) s
-                    on n.userid = s.id
-                    insert into s_my_works_flow_notice (works_flow_id,works_flow_code,works_flow_detail_id,userid,notice_type,createuserid,CreateUser,CreateStaff,upid)
-                    select @ID@,@works_flow_code@,@works_flow_detail_id@,ID,@notice_type@,@userid@,@UserName,@FullName,@time_id@ from dbo.SplitComma(@notice_userids@)
-              end
-            `
-          }
-        }
-
-        sql += _backCustomScript + tail
-
-        if (window.GLOB.funcs && window.GLOB.funcs.length > 0) {
-          window.GLOB.funcs.forEach(item => {
-            let reg = new RegExp('\\$ex@' + item.func_code + '@ex\\$', 'ig')
-            sql = sql.replace(reg, `/*$ex@${item.func_code}-begin*/\n${item.key_sql}\n/*@ex$-end*/`)
-          })
-        }
-
-        // 鏁版嵁鏉冮檺
-        sql = sql.replace(/@\$|\$@/ig, '').replace(/\$check@|@check\$/ig, '')
-        sql = sql.replace(/@datam@/ig, `''`)
-        sql = sql.replace(/@typename@/ig, `'debug'`)
-
-        if (window.GLOB.process && workFlow === 'true') {
-          sql = sql.replace(/@works_flow_code@/ig, `'works_flow_code'`)
-          sql = sql.replace(/@works_flow_name@/ig, `'works_flow_name'`)
-          sql = sql.replace(/@works_flow_param@/ig, `'works_flow_param'`)
-          sql = sql.replace(/@works_flow_detail_id@/ig, `'works_flow_detail_id'`)
-          sql = sql.replace(/@status@/ig, `0`)
-          sql = sql.replace(/@statusname@/ig, `'寮�濮�'`)
-          sql = sql.replace(/@work_group@/ig, `'work_group'`)
-          sql = sql.replace(/@work_grade@/ig, '0')
-          sql = sql.replace(/@start_type@/ig, `'寮�濮�'`)
-          sql = sql.replace(/@check_type@/ig, `'瀹℃牳'`)
-          sql = sql.replace(/@notice_type@/ig, `'鎶勯��'`)
-
-          if (flowType !== 'start') {
-            sql = sql.replace(/@check_userids@/ig, `'checkuserids'`)
-            sql = sql.replace(/@notice_userids@/ig, `'noticeuserids'`)
-          } else {
-            sql = sql.replace(/@check_userids@/ig, `''`)
-            sql = sql.replace(/@notice_userids@/ig, `''`)
-          }
-        }
-        
-        if (skip) {
+        this.props.scriptsChange(values, () => {
           this.setState({
-            skip: false,
+            loading: false,
             editItem: null
-          }, () => {
-            this.props.scriptsChange(values)
           })
           this.props.form.setFieldsValue({
-            sql: ' '
+            sql: ''
           })
-        } else {
-          this.setState({loading: true})
-          Api.sDebug(sql).then(res => {
-            if (res.status || res.ErrCode === '-2') {
-              this.setState({
-                loading: false,
-                editItem: null
-              }, () => {
-                this.props.scriptsChange(values)
-              })
-              this.props.form.setFieldsValue({
-                sql: ' '
-              })
-            } else {
-              this.setState({loading: false})
-  
-              Modal.error({
-                title: res.message
-              })
-            }
+        }, () => {
+          this.setState({
+            loading: false
           })
-        }
+        })
       }
     })
   }
@@ -257,12 +103,12 @@
     })
 
     this.props.form.setFieldsValue({
-      sql: ' '
+      sql: ''
     })
   }
 
   selectScript = (value, option) => {
-    const { flowType } = this.props
+    const { flowType, flowRemark, getSysDefSql } = this.props
 
     if (!value || !option) return
 
@@ -280,15 +126,62 @@
       if (flowType === 'start') {
         value = `insert into s_my_works_flow (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,work_group,works_flow_detail_id,work_grade,bid,createuserid,CreateUser,CreateStaff,upid)
         select @ID@,@works_flow_code@,@works_flow_name@,@works_flow_param@,@status@,@statusname@,@work_group@,@works_flow_detail_id@,@work_grade@,@bid@,@UserID@,@UserName,@FullName,@time_id@
-        insert into s_my_works_flow_log (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,works_flow_detail_id,work_group,work_grade,upid)
-        select @ID@,@works_flow_code@,@works_flow_name@ ,@works_flow_param@,@status@,@statusname@,@works_flow_detail_id@,@work_group@,@work_grade@,@time_id@
+        insert into s_my_works_flow_log (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,works_flow_detail_id,work_group,work_grade,bid,createuserid,CreateUser,CreateStaff,upid)
+        select @ID@,@works_flow_code@,@works_flow_name@ ,@works_flow_param@,@status@,@statusname@,@works_flow_detail_id@,@work_group@,@work_grade@,@bid@,@UserID@,@UserName,@FullName,@time_id@
         insert into s_my_works_flow_notice (works_flow_id,works_flow_code,works_flow_detail_id,userid,notice_type,createuserid,CreateUser,CreateStaff,upid)
-        select @ID@,@works_flow_code@,@works_flow_detail_id@,@userid@,@start_type@,@userid@,@UserName,@FullName,@time_id@`
+        select @ID@,@works_flow_code@,@works_flow_detail_id@,@userid@,@start_type@,@userid@,@UserName,@FullName,@time_id@
+        insert into s_my_works_flow_role (works_flow_id,works_flow_code,userid,works_flow_detail_id,createuserid,CreateUser,CreateStaff,upid,typecharone)
+        select @ID@,@works_flow_code@,@userid@,@works_flow_detail_id@,@userid@,@UserName,@FullName,@time_id@,'begin'`
       } else {
-        value = `update s_my_works_flow set status=@status@,statusname=@statusname@,works_flow_param=@works_flow_param@,works_flow_detail_id=@works_flow_detail_id@,modifydate=getdate(),upid=@time_id@,modifyuserid=@userid@,modifyuser=@username,modifystaff=@fullname
+        value = `set @retmsg =''
+        select @retmsg='X' from s_my_works_flow_role where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0 and userid =@userid@ and works_flow_detail_id =@works_flow_detail_id
+
+        if @retmsg =''
+        begin
+            select @retmsg='X' from s_my_works_flow_role where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and userid=@userid@ and works_flow_detail_id =@works_flow_detail_id
+
+            if @retmsg !=''
+            begin
+                select @ErrorCode='E', @retmsg='褰撳墠鍗曟嵁宸插鏍革紝璇峰埛鏂板悗閲嶈瘯'
+                goto aaa
+            end
+
+            if @dataM@ !='' 
+            begin
+              set @retmsg =''
+              select @retmsg='X' from s_my_works_flow_role where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0 and works_flow_detail_id =@works_flow_detail_id
+
+              if @retmsg != ''
+              begin
+                  goto goto_mk
+              end
+            end
+            else
+            begin
+                set @retmsg =''
+                select @retmsg=userid from s_my_works_flow_role where works_flow_id=@id@ and works_flow_code=@works_flow_code@ and deleted=0 and works_flow_detail_id =@works_flow_detail_id
+                
+                if @retmsg !=''
+                begin
+                    select @retmsg=workerCode+workerName from BD_workers where id=@retmsg 
+                    
+                    select @retmsg='椤甸潰鏁版嵁宸叉洿鏂帮紝鎴栨病鏈夊綋鍓嶅崟鎹殑瀹℃壒鏉冮檺锛岃鑱旂郴'+@retmsg+'鎿嶄綔'
+                    goto aaa 
+                end
+            end
+
+            select @retmsg='椤甸潰鏁版嵁宸叉洿鏂帮紝鎴栨病鏈夊綋鍓嶅崟鎹殑瀹℃壒鏉冮檺'
+            goto aaa 
+        end
+
+        goto_mk:
+
+        set @retmsg=''
+
+        update s_my_works_flow set status=@status@,statusname=@statusname@,works_flow_param=@works_flow_param@,works_flow_detail_id=@works_flow_detail_id@,modifydate=getdate(),upid=@time_id@,modifyuserid=@userid@,modifyuser=@username,modifystaff=@fullname${flowRemark ? ',remark=@' + flowRemark : ''}
         where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0
-        insert into s_my_works_flow_log (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,works_flow_detail_id,work_group,work_grade,upid)
-        select @ID@,@works_flow_code@,@works_flow_name@ ,@works_flow_param@,@status@,@statusname@,@works_flow_detail_id@,@work_group@,@work_grade@,@time_id@
+        insert into s_my_works_flow_log (works_flow_id,works_flow_code,works_flow_name,works_flow_param,status,statusname,works_flow_detail_id,work_group,work_grade,bid,createuserid,CreateUser,CreateStaff,upid${flowRemark ? ',remark' : ''})
+        select @ID@,@works_flow_code@,@works_flow_name@ ,@works_flow_param@,@status@,@statusname@,@works_flow_detail_id@,@work_group@,@work_grade@,@bid@,@UserID@,@UserName,@FullName,@time_id@${flowRemark ? ',@' + flowRemark : ''}
         
         update s_my_works_flow_role set deleted=10,modifydate=getdate(),upid=@time_id@,modifyuserid=@userid@,modifyuser=@username,modifystaff=@fullname 
         where works_flow_id=@ID@ and works_flow_code=@works_flow_code@ and deleted=0
@@ -313,6 +206,11 @@
       }
 
       value = value.replace(/\n\s{8}/g, '\n')
+    } else if (value === 'defaultsql') {
+      value = ''
+      if (getSysDefSql) {
+        value = getSysDefSql()
+      }
     }
 
     _sql = _sql.replace(/\s{6}$/, '')
@@ -327,7 +225,7 @@
   }
 
   render() {
-    const { formfields, colfields, systemScripts, btn, type, workFlow } = this.props
+    const { formfields, colfields, systemScripts, btn, type, workFlow, flowType } = this.props
     const { getFieldDecorator } = this.props.form
     const { editItem, skip } = this.state
     const formItemLayout = {
@@ -358,9 +256,9 @@
           </Col> : null}
           {!_type ? <Col span={24} className="sqlfield">
             <Form.Item label={'鍙敤瀛楁'}>
-              <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="鍏叡鍊硷紝璇锋寜鐓xxx@鏍煎紡浣跨敤銆�"><span style={{color: '#1890ff'}}>BID, ID, LoginUID, SessionUid, UserID, Appkey, time_id, typename, datam</span></Tooltip>,&nbsp;
-              {window.GLOB.process && workFlow === 'true' ? <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="宸ヤ綔娴佸彉閲忥紝璇锋寜鐓xxx@鏍煎紡浣跨敤銆�"><span style={{color: '#26C281'}}>works_flow_code, works_flow_name, works_flow_param, works_flow_detail_id, status, statusname, work_group, work_grade, start_type, check_type, notice_type, check_userids, notice_userids, </span></Tooltip> : null}
-              <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="绯荤粺鍙橀噺锛岀郴缁熶細瀹氫箟鍙橀噺骞惰祴鍊笺��"><span style={{color: '#fa8c16'}}>UserName, FullName, RoleID, mk_departmentcode, mk_organization, mk_user_type, mk_nation, mk_province, mk_city, mk_district, mk_address, mk_deleted</span></Tooltip>,&nbsp;
+              <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="鍏叡鍊硷紝璇锋寜鐓xxx@鏍煎紡浣跨敤銆�"><span style={{color: '#1890ff'}}>BID, ID, LoginUID, SessionUid, UserID, Appkey, lang, time_id, typename, datam</span></Tooltip>,&nbsp;
+              {window.GLOB.process && workFlow === 'true' ? <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title={`宸ヤ綔娴佸彉閲忥紝璇锋寜鐓xxx@鏍煎紡浣跨敤${flowType !== 'start' ? '锛寃orks_flow_sign 涓哄垎鏀惎鐢ㄩ珮绾ц缃椂鐨勬爣璁板��' : ''}銆俙}><span style={{color: '#26C281'}}>works_flow_code, works_flow_name, works_flow_param, works_flow_detail_id, status, statusname, work_group, work_grade, {flowType === 'start' ? 'start_type,' : 'check_type, notice_type, check_userids, notice_userids, works_flow_sign,'} </span></Tooltip> : null}
+              <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="绯荤粺鍙橀噺锛岀郴缁熶細瀹氫箟鍙橀噺骞惰祴鍊笺��"><span style={{color: '#fa8c16'}}>UserName, FullName, RoleID, mk_departmentcode, mk_organization, mk_user_type, mk_nation, mk_province, mk_city, mk_district, mk_address, mk_deleted, mk_submit_type</span></Tooltip>,&nbsp;
               <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="绯荤粺鍙橀噺锛岀郴缁熶細瀹氫箟鍙橀噺骞跺湪鍗曞彿鐢熸垚鎴栧垱寤哄嚟璇佹椂浣跨敤銆�"><span style={{color: '#13c2c2'}}>BillCode, BVoucher, FIBVoucherDate, FiYear, ModularDetailCode</span></Tooltip>
               {formfields ? <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="琛ㄥ崟鍙橀噺锛岀郴缁熶細瀹氫箟鍙橀噺骞惰祴鍊笺��">, <span style={{color: '#8E44AD'}}>{formfields}</span></Tooltip> : ''}
               {colfields ? <Tooltip mouseLeaveDelay={0.3} mouseEnterDelay={0.3} placement="top" title="瀛楁闆嗗彉閲忥紝绯荤粺浼氬畾涔夊彉閲忓苟璧嬪�笺��">, {colfields}</Tooltip> : ''}
@@ -392,7 +290,7 @@
                 onSelect={this.selectScript}
                 getPopupContainer={() => document.getElementById('verify-custom-scripts' + _type)}
               >
-                <Select.Option key="default" value={this.props.defaultsql}>榛樿sql</Select.Option>
+                <Select.Option key="default" value="defaultsql">榛樿sql</Select.Option>
                 {window.GLOB.process && workFlow === 'true' ? <Select.Option key="flow" value="flowSql">榛樿sql锛堝伐浣滄祦锛�</Select.Option> : null}
                 <Select.Option key="debugger" value={`z_debug: select @ErrorCode='E',@retmsg='娴嬭瘯鏂偣' goto aaa`}>
                   娴嬭瘯鏂偣
@@ -421,15 +319,9 @@
                 <QuestionCircleOutlined className="mk-form-tip" />
                 sql
               </Tooltip>
-            }>
+            } required>
               {getFieldDecorator('sql', {
                 initialValue: '',
-                rules: [
-                  {
-                    required: true,
-                    message: '璇疯緭鍏ql!'
-                  }
-                ]
               })(<CodeMirror />)}
             </Form.Item>
           </Col>

--
Gitblit v1.8.0