a.html 52 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport"
  6. content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no" />
  7. <link rel="stylesheet" href="./resource/vantIndex.css" />
  8. <link rel="stylesheet" href="./vue-form-making/index.css">
  9. </head>
  10. <body>
  11. <style>
  12. :root:root {
  13. --van-nav-bar-background: rgb(21, 122, 44);
  14. --van-nav-bar-title-text-color: rgb(255, 255, 255);
  15. --van-nav-bar-icon-color: rgb(255, 255, 255);
  16. }
  17. .blank_adopzrdd {
  18. display: flex;
  19. align-items: center;
  20. margin-bottom: 10px;
  21. }
  22. .blank_adopzrdd>span {
  23. display: inline-block;
  24. width: 80px;
  25. }
  26. .seal-form-row {
  27. display: flex; align-items: center; padding: 10px 16px;
  28. border-bottom: 1px solid #ebedf0; background: #fff;
  29. }
  30. .seal-form-label { width: 100px; font-size: 14px; color: #646566; flex-shrink: 0; }
  31. .seal-form-val { flex: 1; font-size: 14px; color: #333; text-align: right; }
  32. .seal-form-arrow { font-size: 20px; color: #c8c9cc; margin-left: 8px; }
  33. .seal-form-input { flex: 1; border: none; font-size: 14px; text-align: right; outline: none; background: transparent; }
  34. .seal-card { background: #fff; border-radius: 8px; margin-bottom: 10px; overflow: hidden; box-shadow: 0 1px 4px rgba(0,0,0,.08); }
  35. .seal-card-header { display: flex; align-items: center; padding: 10px 12px; background: #f7f8fa; border-bottom: 1px solid #ebedf0; }
  36. .seal-card-index { font-size: 14px; font-weight: bold; color: #157A2C; margin-right: 8px; }
  37. .seal-card-type { flex: 1; font-size: 15px; font-weight: bold; color: #323233; }
  38. .seal-card-actions { display: flex; gap: 6px; flex-shrink: 0; }
  39. .seal-card-body { padding: 8px 12px; }
  40. .seal-card-row { display: flex; padding: 6px 0; font-size: 13px; }
  41. .seal-card-label { width: 42px; color: #969799; flex-shrink: 0; }
  42. .seal-card-value { flex: 1; color: #323233; word-break: break-all; }
  43. :root#app {
  44. background: #4298fd0d;
  45. }
  46. ::v-deep.fm-form-item {
  47. margin-bottom: 10px !important;
  48. }
  49. </style>
  50. <div id="app">
  51. <van-nav-bar height='100rpx;margin-top:100rpx' @click-left="onClickLeft"
  52. style="background-color: rgb(21, 122, 44);color: rgb(255, 255, 255);" :title="title" left-arrow
  53. :safe-area-inset-top='true' :placeholder='true' :fixed='true'></van-nav-bar>
  54. <fm-generate-vant-form style='height: 81vh !important;overflow: auto;background: #a3a6ad1c;margin-top:44px'
  55. v-if='isFlag' :data="jsonData" :value="form.valueJson" ref="generateForm" :edit='isEdit' @on-upload-local="handleUploadLocal">
  56. <!-- 费用报销 -->
  57. <template v-slot:blank_adopzrdd="scope">
  58. <div style="width: 100%;">
  59. <van-button type="primary" size="small" @click="addNewRow('blank_adopzrdd')"
  60. style="margin-bottom: 10px">添加报销事项</van-button>
  61. <div v-for="(item, index) in scope.model.blank_adopzrdd" :key="index" style="width: 100%;">
  62. <div class="blank_adopzrdd">
  63. <span>{{ index + 1 }}报销事项:</span>
  64. <van-field rows="1" v-model="scope.model.blank_adopzrdd[index].remark" type="textarea"
  65. autosize style="width: calc(100% - 80px)"></van-field>
  66. </div>
  67. <div class="blank_adopzrdd">
  68. <span>
  69. <!-- <van-icon @click="delNewRow('blank_adopzrdd', index)" name="delete"
  70. style="color: #f56c6c;"></van-icon> -->
  71. 金额:</span>
  72. <van-field v-model="scope.model.blank_adopzrdd[index].price" type="number"
  73. style="width: calc(100% - 80px)"
  74. @change="changePrice(scope.model.blank_adopzrdd)"></van-field>
  75. </div>
  76. <div class="blank_adopzrdd">
  77. <span>
  78. <van-icon @click="delNewRow('blank_adopzrdd', index, true)" name="delete"
  79. style="color: #f56c6c;"></van-icon>
  80. 附件:</span>
  81. <template v-if="item.files && item.files.length">
  82. <van-badge :content="item.files.length">
  83. <van-button type="primary" size="mini" @click="handleUploadLocal('blank_adopzrdd', index)">上传附件</van-button>
  84. </van-badge>
  85. </template>
  86. <van-button v-else type="primary" size="mini" @click="handleUploadLocal('blank_adopzrdd', index)">上传附件</van-button>
  87. </div>
  88. </div>
  89. </div>
  90. </template>
  91. <template v-slot:manage_workorder="scope">
  92. <van-field v-model="scope.model.manage_workorder.code" readonly @click="addWorkOrder()"></van-field>
  93. </template>
  94. <template v-slot:blank_use_seal="scope">
  95. <div style="width: 100%;">
  96. <!-- 列表项 -->
  97. <view v-for="(item, index) in scope.model.blank_use_seal" :key="index" class="seal-card">
  98. <div class="seal-card-header">
  99. <span class="seal-card-index">{{ index + 1 }}.</span>
  100. <span class="seal-card-type">{{ item.useSealType || '未选择类型' }}</span>
  101. <div class="seal-card-actions">
  102. <van-button size="mini" type="primary" plain @click="openSealEditDialog(scope, 'edit', index)">编辑</van-button>
  103. <van-button size="mini" type="danger" plain @click="delNewRow('blank_use_seal', index)">删除</van-button>
  104. </div>
  105. </div>
  106. <div class="seal-card-body">
  107. <div class="seal-card-row">
  108. <span class="seal-card-label">事由</span>
  109. <span class="seal-card-value">{{ item.useSealRemark || '未填写' }}</span>
  110. </div>
  111. <div class="seal-card-row">
  112. <span class="seal-card-label">印章</span>
  113. <span class="seal-card-value">{{ item.useSealName || '未选择' }}</span>
  114. </div>
  115. </div>
  116. </view>
  117. <van-button type="primary" size="small" block @click="openSealEditDialog(scope, 'add', scope.model.blank_use_seal?scope.model.blank_use_seal.length:0)" style="margin-bottom:10px">+ 添加印章使用</van-button>
  118. </div>
  119. </template>
  120. <template v-slot:blank_use_qualification="scope">
  121. <div style="width: 100%;">
  122. <view v-for="(item, index) in scope.model.blank_use_qualification" :key="index" class="seal-card">
  123. <div class="seal-card-header">
  124. <span class="seal-card-index">{{ index + 1 }}.</span>
  125. <span class="seal-card-type">{{ item.useQualificationType || '未选择类型' }}</span>
  126. <div class="seal-card-actions">
  127. <van-button size="mini" type="primary" plain @click="openQualEditDialog(scope, 'edit', index)">编辑</van-button>
  128. <van-button size="mini" type="danger" plain @click="delNewRow('blank_use_qualification', index)">删除</van-button>
  129. </div>
  130. </div>
  131. <div class="seal-card-body">
  132. <div class="seal-card-row">
  133. <span class="seal-card-label">事由</span>
  134. <span class="seal-card-value">{{ item.useQualificationRemark || '未填写' }}</span>
  135. </div>
  136. <div class="seal-card-row">
  137. <span class="seal-card-label">名称</span>
  138. <span class="seal-card-value">{{ item.useQualificationName || '未选择' }}</span>
  139. </div>
  140. </div>
  141. </view>
  142. <van-button type="primary" size="small" block @click="openQualEditDialog(scope, 'add', scope.model.blank_use_qualification?scope.model.blank_use_qualification.length:0)" style="margin-bottom:10px">+ 添加资质使用</van-button>
  143. </div>
  144. </template>
  145. <template v-slot:blank_application_component="scope">
  146. <div style="width: 100%;">
  147. <view v-for="(item, index) in scope.model.blank_application_component" :key="index" class="seal-card">
  148. <div class="seal-card-header">
  149. <span class="seal-card-index">{{ index + 1 }}.</span>
  150. <span class="seal-card-type">{{ item.productName || '未选择产品' }}</span>
  151. <div class="seal-card-actions">
  152. <van-button size="mini" type="primary" plain @click="openAppEditDialog(scope, 'edit', index)">编辑</van-button>
  153. <van-button size="mini" type="danger" plain @click="delNewRow('blank_application_component', index)">删除</van-button>
  154. </div>
  155. </div>
  156. <div class="seal-card-body">
  157. <div class="seal-card-row">
  158. <span class="seal-card-label">数量</span>
  159. <span class="seal-card-value">{{ item.quantity || '未填写' }}</span>
  160. </div>
  161. <div class="seal-card-row">
  162. <span class="seal-card-label">规格</span>
  163. <span class="seal-card-value">{{ item.specification || '未填写' }}</span>
  164. </div>
  165. </div>
  166. </view>
  167. <van-button type="primary" size="small" block @click="openAppEditDialog(scope, 'add', scope.model.blank_application_component?scope.model.blank_application_component.length:0)" style="margin-bottom:10px">+ 添加产品</van-button>
  168. </div>
  169. </template>
  170. <template v-slot:blank_business_trip="scope">
  171. <div style="width: 100%;">
  172. <view v-for="(item, index) in scope.model.blank_business_trip" :key="index" class="seal-card">
  173. <div class="seal-card-header">
  174. <span class="seal-card-index">{{ index + 1 }}.</span>
  175. <span class="seal-card-type">{{ item.transportation || '未知交通' }} {{ item.departureCity || '?' }}→{{ item.destinationCity || '?' }}</span>
  176. <div class="seal-card-actions">
  177. <van-button size="mini" type="primary" plain @click="openTripEditDialog(scope, 'edit', index)">编辑</van-button>
  178. <van-button size="mini" type="danger" plain @click="delNewRow('blank_business_trip', index).then(() => calcTripTotal())">删除</van-button>
  179. </div>
  180. </div>
  181. <div class="seal-card-body">
  182. <div class="seal-card-row">
  183. <span class="seal-card-label">往返</span>
  184. <span class="seal-card-value">{{ item.isRoundTrip || '未填写' }}</span>
  185. </div>
  186. <div class="seal-card-row">
  187. <span class="seal-card-label">时间</span>
  188. <span class="seal-card-value">{{ item.startTime || '-' }} ~ {{ item.endTime || '-' }}</span>
  189. </div>
  190. <div class="seal-card-row">
  191. <span class="seal-card-label">时长</span>
  192. <span class="seal-card-value">{{ item.duration || '-' }}</span>
  193. </div>
  194. </div>
  195. </view>
  196. <van-button type="primary" size="small" block @click="openTripEditDialog(scope, 'add', scope.model.blank_business_trip?scope.model.blank_business_trip.length:0)" style="margin-bottom:10px">+ 添加行程</van-button>
  197. </div>
  198. </template>
  199. <template v-slot:blank_purchase_requisition="scope">
  200. <div style="width: 100%;">
  201. <view v-for="(item, index) in scope.model.blank_purchase_requisition" :key="index" class="seal-card">
  202. <div class="seal-card-header">
  203. <span class="seal-card-index">{{ index + 1 }}.</span>
  204. <span class="seal-card-type">{{ item.materialName || '未填写物料' }} ×{{ item.quantity||0 }}{{ item.unit||'' }}</span>
  205. <div class="seal-card-actions">
  206. <van-button size="mini" type="primary" plain @click="openPrEditDialog(scope, 'edit', index)">编辑</van-button>
  207. <van-button size="mini" type="danger" plain @click="delNewRow('blank_purchase_requisition', index)">删除</van-button>
  208. </div>
  209. </div>
  210. <div class="seal-card-body">
  211. <div class="seal-card-row">
  212. <span class="seal-card-label">规格</span>
  213. <span class="seal-card-value">{{ item.specification || '-' }}</span>
  214. </div>
  215. <div class="seal-card-row">
  216. <span class="seal-card-label">用途</span>
  217. <span class="seal-card-value">{{ item.purpose || '-' }}</span>
  218. </div>
  219. <div class="seal-card-row">
  220. <span class="seal-card-label">交付</span>
  221. <span class="seal-card-value">{{ item.expectedDeliveryDate || '-' }}</span>
  222. </div>
  223. <div class="seal-card-row" v-if="item.remark">
  224. <span class="seal-card-label">备注</span>
  225. <span class="seal-card-value">{{ item.remark }}</span>
  226. </div>
  227. </div>
  228. </view>
  229. <van-button type="primary" size="small" block @click="openPrEditDialog(scope, 'add', scope.model.blank_purchase_requisition?scope.model.blank_purchase_requisition.length:0)" style="margin-bottom:10px">+ 添加物料</van-button>
  230. </div>
  231. </template>
  232. <template v-slot:blank_business_component="scope">
  233. <div style="width: 100%;">
  234. <view v-for="(item, index) in scope.model.blank_business_component" :key="index" class="seal-card">
  235. <div class="seal-card-header">
  236. <span class="seal-card-index">{{ index + 1 }}.</span>
  237. <span class="seal-card-type">{{ item.transportationWay || '未知交通' }} {{ item.startPlace || '?' }}→{{ item.endPlace || '?' }}</span>
  238. <div class="seal-card-actions">
  239. <van-button size="mini" type="primary" plain @click="openBusinessEditDialog(scope, 'edit', index)">编辑</van-button>
  240. <van-button size="mini" type="danger" plain @click="delNewRow('blank_business_component', index).then(() => calcBizTotal())">删除</van-button>
  241. </div>
  242. </div>
  243. <div class="seal-card-body">
  244. <div class="seal-card-row">
  245. <span class="seal-card-label">出发</span>
  246. <span class="seal-card-value">{{ item.startDate || '-' }}</span>
  247. </div>
  248. <div class="seal-card-row">
  249. <span class="seal-card-label">到达</span>
  250. <span class="seal-card-value">{{ item.endDate || '-' }}</span>
  251. </div>
  252. <div class="seal-card-row">
  253. <span class="seal-card-label">出发地</span>
  254. <span class="seal-card-value">{{ item.startPlace || '-' }}</span>
  255. </div>
  256. <div class="seal-card-row">
  257. <span class="seal-card-label">到达地</span>
  258. <span class="seal-card-value">{{ item.endPlace || '-' }}</span>
  259. </div>
  260. <div class="seal-card-row">
  261. <span class="seal-card-label">交通</span>
  262. <span class="seal-card-value">{{ item.transportationWay || '-' }}</span>
  263. </div>
  264. <div class="seal-card-row">
  265. <span class="seal-card-label">费用</span>
  266. <span class="seal-card-value">¥{{ item.price || '0' }}</span>
  267. </div>
  268. </div>
  269. </view>
  270. <van-button type="primary" size="small" block @click="openBusinessEditDialog(scope, 'add', scope.model.blank_business_component?scope.model.blank_business_component.length:0)" style="margin-bottom:10px">+ 添加交通费用</van-button>
  271. </div>
  272. </template>
  273. </fm-generate-vant-form>
  274. <!-- 印章选择弹窗(根层级,避免被 form overflow 裁剪) -->
  275. <van-popup v-model:show="sealPopVisible" position="bottom" :style="{ height: '30%' }" teleport="body">
  276. <van-picker
  277. show-toolbar
  278. :title="sealPopType === 'type' ? '印章使用类型' : '印章名称'"
  279. :columns="sealPopType === 'type' ? useSealTypeList : useSealNameList"
  280. @confirm="onSealPopConfirm"
  281. @cancel="sealPopVisible = false"
  282. />
  283. </van-popup>
  284. <!-- 印章编辑弹窗 -->
  285. <van-popup v-model:show="sealEditVisible" position="bottom" :style="{ height: '55%' }" round>
  286. <div style="padding:16px">
  287. <h4 style="text-align:center;margin-bottom:12px">{{ sealEditTitle }}</h4>
  288. <!-- 印章使用类型 -->
  289. <div class="seal-form-row" @click="openSealPickerInEdit('type')">
  290. <span class="seal-form-label">印章使用类型</span>
  291. <span class="seal-form-val">{{ sealEditForm.useSealType || '请选择' }}</span>
  292. <span class="seal-form-arrow">›</span>
  293. </div>
  294. <!-- 印章使用事由 -->
  295. <div class="seal-form-row">
  296. <span class="seal-form-label">印章使用事由</span>
  297. <input class="seal-form-input" v-model="sealEditForm.useSealRemark" placeholder="请输入" />
  298. </div>
  299. <!-- 印章名称 -->
  300. <div class="seal-form-row" @click="openSealPickerInEdit('name')">
  301. <span class="seal-form-label">印章名称</span>
  302. <span class="seal-form-val">{{ sealEditForm.useSealName || '请选择' }}</span>
  303. <span class="seal-form-arrow">›</span>
  304. </div>
  305. <div style="display:flex;justify-content:space-between;padding:20px 0">
  306. <van-button style="width:48%" @click="sealEditVisible = false">取消</van-button>
  307. <van-button style="width:48%" type="primary" @click="onSealEditSave">保存</van-button>
  308. </div>
  309. </div>
  310. </van-popup>
  311. <!-- 资质选择弹窗 -->
  312. <van-popup v-model:show="qualPopVisible" position="bottom" :style="{ height: '30%' }">
  313. <van-picker
  314. show-toolbar
  315. :title="qualPopType === 'type' ? '资质类型' : '使用资质名称'"
  316. :columns="qualPopType === 'type' ? useQualificationTypeList : useQualificationNameList"
  317. @confirm="onQualPopConfirm"
  318. @cancel="qualPopVisible = false"
  319. />
  320. </van-popup>
  321. <!-- 资质编辑弹窗 -->
  322. <van-popup v-model:show="qualEditVisible" position="bottom" :style="{ height: '55%' }" round>
  323. <div style="padding:16px">
  324. <h4 style="text-align:center;margin-bottom:12px">{{ qualEditTitle }}</h4>
  325. <div class="seal-form-row" @click="openQualPickerInEdit('type')">
  326. <span class="seal-form-label">资质类型</span>
  327. <span class="seal-form-val">{{ qualEditForm.useQualificationType || '请选择' }}</span>
  328. <span class="seal-form-arrow">›</span>
  329. </div>
  330. <div class="seal-form-row">
  331. <span class="seal-form-label">资质使用事由</span>
  332. <input class="seal-form-input" v-model="qualEditForm.useQualificationRemark" placeholder="请输入" />
  333. </div>
  334. <div class="seal-form-row" @click="openQualPickerInEdit('name')">
  335. <span class="seal-form-label">使用资质名称</span>
  336. <span class="seal-form-val">{{ qualEditForm.useQualificationName || '请选择' }}</span>
  337. <span class="seal-form-arrow">›</span>
  338. </div>
  339. <div style="display:flex;justify-content:space-between;padding:20px 0">
  340. <van-button style="width:48%" @click="qualEditVisible = false">取消</van-button>
  341. <van-button style="width:48%" type="primary" @click="onQualEditSave">保存</van-button>
  342. </div>
  343. </div>
  344. </van-popup>
  345. <!-- 产品选择弹窗 -->
  346. <van-popup v-model:show="appPopVisible" position="bottom" :style="{ height: '45%' }">
  347. <van-picker
  348. show-toolbar
  349. title="选择产品名称"
  350. :columns="productNameList"
  351. @confirm="onAppPopConfirm"
  352. @cancel="appPopVisible = false"
  353. />
  354. </van-popup>
  355. <!-- 产品编辑弹窗 -->
  356. <van-popup v-model:show="appEditVisible" position="bottom" :style="{ height: '55%' }" round>
  357. <div style="padding:16px">
  358. <h4 style="text-align:center;margin-bottom:12px">{{ appEditTitle }}</h4>
  359. <div class="seal-form-row" @click="openAppPickerInEdit">
  360. <span class="seal-form-label">产品名称</span>
  361. <span class="seal-form-val">{{ appEditForm.productName || '请选择' }}</span>
  362. <span class="seal-form-arrow">›</span>
  363. </div>
  364. <div class="seal-form-row">
  365. <span class="seal-form-label">数量</span>
  366. <input class="seal-form-input" v-model="appEditForm.quantity" type="number" placeholder="请输入" />
  367. </div>
  368. <div class="seal-form-row">
  369. <span class="seal-form-label">规格型号</span>
  370. <input class="seal-form-input" v-model="appEditForm.specification" placeholder="请输入" />
  371. </div>
  372. <div style="display:flex;justify-content:space-between;padding:20px 0">
  373. <van-button style="width:48%" @click="appEditVisible = false">取消</van-button>
  374. <van-button style="width:48%" type="primary" @click="onAppEditSave">保存</van-button>
  375. </div>
  376. </div>
  377. </van-popup>
  378. <!-- 行程编辑弹窗 -->
  379. <van-popup v-model:show="tripEditVisible" position="bottom" :style="{ height: '70%' }" round>
  380. <div style="padding:16px">
  381. <h4 style="text-align:center;margin-bottom:12px">{{ tripEditTitle }}</h4>
  382. <div class="seal-form-row">
  383. <span class="seal-form-label">交通工具</span>
  384. <input class="seal-form-input" v-model="tripEditForm.transportation" placeholder="请输入" />
  385. </div>
  386. <div class="seal-form-row">
  387. <span class="seal-form-label">单程往返</span>
  388. <input class="seal-form-input" v-model="tripEditForm.isRoundTrip" placeholder="请输入" />
  389. </div>
  390. <div class="seal-form-row">
  391. <span class="seal-form-label">出发城市</span>
  392. <input class="seal-form-input" v-model="tripEditForm.departureCity" placeholder="请输入" />
  393. </div>
  394. <div class="seal-form-row">
  395. <span class="seal-form-label">目的城市</span>
  396. <input class="seal-form-input" v-model="tripEditForm.destinationCity" placeholder="请输入" />
  397. </div>
  398. <div class="seal-form-row">
  399. <span class="seal-form-label">开始时间</span>
  400. <input class="seal-form-input" type="datetime-local" v-model="tripEditForm.startTime" @change="calcTripDuration" />
  401. </div>
  402. <div class="seal-form-row">
  403. <span class="seal-form-label">结束时间</span>
  404. <input class="seal-form-input" type="datetime-local" v-model="tripEditForm.endTime" @change="calcTripDuration" />
  405. </div>
  406. <div class="seal-form-row">
  407. <span class="seal-form-label">时长</span>
  408. <span class="seal-form-val">{{ tripEditForm.duration || '自动计算' }}</span>
  409. </div>
  410. <div style="display:flex;justify-content:space-between;padding:20px 0">
  411. <van-button style="width:48%" @click="tripEditVisible = false">取消</van-button>
  412. <van-button style="width:48%" type="primary" @click="onTripEditSave">保存</van-button>
  413. </div>
  414. </div>
  415. </van-popup>
  416. <!-- 采购申请编辑弹窗 -->
  417. <van-popup v-model:show="prEditVisible" position="bottom" :style="{ height: '70%' }" round>
  418. <div style="padding:16px">
  419. <h4 style="text-align:center;margin-bottom:12px">{{ prEditTitle }}</h4>
  420. <div class="seal-form-row"><span class="seal-form-label">物料名称</span><input class="seal-form-input" v-model="prEditForm.materialName" placeholder="请输入" /></div>
  421. <div class="seal-form-row"><span class="seal-form-label">规格型号</span><input class="seal-form-input" v-model="prEditForm.specification" placeholder="请输入" /></div>
  422. <div class="seal-form-row"><span class="seal-form-label">单位</span><input class="seal-form-input" v-model="prEditForm.unit" placeholder="个/台/套" /></div>
  423. <div class="seal-form-row"><span class="seal-form-label">数量</span><input class="seal-form-input" type="number" v-model="prEditForm.quantity" placeholder="请输入" /></div>
  424. <div class="seal-form-row"><span class="seal-form-label">用途</span><input class="seal-form-input" v-model="prEditForm.purpose" placeholder="请输入" /></div>
  425. <div class="seal-form-row"><span class="seal-form-label">交付日期</span><input class="seal-form-input" type="datetime-local" v-model="prEditForm.expectedDeliveryDate" /></div>
  426. <div class="seal-form-row"><span class="seal-form-label">备注</span><input class="seal-form-input" v-model="prEditForm.remark" placeholder="请输入" /></div>
  427. <div style="display:flex;justify-content:space-between;padding:20px 0">
  428. <van-button style="width:48%" @click="prEditVisible = false">取消</van-button>
  429. <van-button style="width:48%" type="primary" @click="onPrEditSave">保存</van-button>
  430. </div>
  431. </div>
  432. </van-popup>
  433. <!-- 交通费用编辑弹窗 -->
  434. <van-popup v-model:show="businessEditVisible" position="bottom" :style="{ height: '75%' }" round>
  435. <div style="padding:16px">
  436. <h4 style="text-align:center;margin-bottom:12px">{{ businessEditTitle }}</h4>
  437. <div class="seal-form-row">
  438. <span class="seal-form-label">出发日期</span>
  439. <input class="seal-form-input" type="datetime-local" v-model="businessEditForm.startDate" />
  440. </div>
  441. <div class="seal-form-row">
  442. <span class="seal-form-label">到达日期</span>
  443. <input class="seal-form-input" type="datetime-local" v-model="businessEditForm.endDate" />
  444. </div>
  445. <div class="seal-form-row">
  446. <span class="seal-form-label">出发地</span>
  447. <input class="seal-form-input" v-model="businessEditForm.startPlace" placeholder="请输入" />
  448. </div>
  449. <div class="seal-form-row">
  450. <span class="seal-form-label">到达地</span>
  451. <input class="seal-form-input" v-model="businessEditForm.endPlace" placeholder="请输入" />
  452. </div>
  453. <div class="seal-form-row" @click="businessTransportPopVisible = true">
  454. <span class="seal-form-label">交通方式</span>
  455. <span class="seal-form-val">{{ businessEditForm.transportationWay || '请选择' }}</span>
  456. <span class="seal-form-arrow">›</span>
  457. </div>
  458. <div class="seal-form-row">
  459. <span class="seal-form-label">交通费用</span>
  460. <input class="seal-form-input" type="number" v-model="businessEditForm.price" placeholder="请输入" />
  461. </div>
  462. <div style="display:flex;justify-content:space-between;padding:20px 0">
  463. <van-button style="width:48%" @click="businessEditVisible = false">取消</van-button>
  464. <van-button style="width:48%" type="primary" @click="onBusinessEditSave">保存</van-button>
  465. </div>
  466. </div>
  467. </van-popup>
  468. <!-- 交通方式选择弹窗 -->
  469. <van-popup v-model:show="businessTransportPopVisible" position="bottom" :style="{ height: '30%' }">
  470. <van-picker
  471. show-toolbar
  472. title="选择交通方式"
  473. :columns="transportationWayList"
  474. @confirm="onBusinessTransportConfirm"
  475. @cancel="businessTransportPopVisible = false"
  476. />
  477. </van-popup>
  478. <div>
  479. <div v-if='isEdit'
  480. style="display: flex;justify-content: space-between; padding: 30px 30px; position: fixed; bottom: 0; width: 100%; height: 80px;">
  481. <van-button style="width:48%" type="primary" :loading="loading" @click="handleAudit(1)">提交
  482. </van-button>
  483. <van-button style="width:48%" type="danger" :loading="loading" @click="onClickLeft">取消
  484. </van-button>
  485. </div>
  486. </div>
  487. </div>
  488. <script src="./resource/vue.global.prod.js"></script>
  489. <script src="./resource/vant.min.js"></script>
  490. <script src="./resource/axios.min.js"></script>
  491. <script src="./resource/uni.webview.js"></script>
  492. <script src="./vue-form-making/form-making-v3.umd.js"></script>
  493. <script>
  494. let EnvObj = {}
  495. uni.getEnv(function(res) {
  496. EnvObj = res;
  497. console.log('当前环境:' + JSON.stringify(res));
  498. });
  499. Vue.createApp({
  500. data() {
  501. return {
  502. loading: false,
  503. isFlag: false,
  504. isEdit: true,
  505. title: '',
  506. jsonData: {},
  507. editData: {},
  508. form: {},
  509. headers: {},
  510. APIUrl: '',
  511. type: 'add',
  512. selectFilesField: null,
  513. selectFilesRowIndex: null,
  514. // useSealTypeList: ['杭州', '宁波', '温州', '绍兴', '湖州', '嘉兴', '金华', '衢州'],
  515. // useSealNameList: ['公章', '合同章', '财务章', '法人章', '发票章', '部门章'],
  516. sealPopVisible: false, // 印章选择弹窗
  517. sealPopType: 'type', // 'type' | 'name'
  518. sealPopScope: null, // 弹窗对应的 scope(编辑模式为空)
  519. sealPopIndex: null, // 弹窗对应的行索引(编辑模式为当前行)
  520. sealPopIsEdit: false, // 是否来自编辑弹窗
  521. sealEditVisible: false, // 印章编辑弹窗
  522. sealEditTitle: '添加印章使用', // 编辑弹窗标题
  523. sealEditForm: { useSealType: '', useSealRemark: '', useSealName: '' },
  524. sealEditMode: 'add', // 'add' | 'edit'
  525. sealEditScope: null,
  526. sealEditIndex: null,
  527. useSealTypeList: [
  528. {
  529. value: '加盖',
  530. text: '加盖'
  531. },
  532. {
  533. value: '借用',
  534. text: '借用'
  535. },
  536. {
  537. value: '无',
  538. text: '无'
  539. },
  540. ],
  541. useSealNameList: [
  542. {
  543. value: '总公司公章',
  544. text: '总公司公章'
  545. },
  546. {
  547. value: '合同专用章',
  548. text: '合同专用章'
  549. },
  550. {
  551. value: '法人章',
  552. text: '法人章'
  553. },
  554. {
  555. value: '财务专用章',
  556. text: '财务专用章'
  557. },
  558. {
  559. value: '发票专用章',
  560. text: '发票专用章'
  561. },
  562. {
  563. value: '质检专用章',
  564. text: '质检专用章'
  565. },
  566. {
  567. value: '业务专用章',
  568. text: '业务专用章'
  569. },
  570. {
  571. value: '出库专用章',
  572. text: '出库专用章'
  573. },
  574. {
  575. value: '无',
  576. text: '无'
  577. },
  578. ],
  579. // 资质使用
  580. useQualificationTypeList: [
  581. { value: '原件', text: '原件' },
  582. { value: '复印件', text: '复印件' },
  583. { value: '扫描件', text: '扫描件' },
  584. { value: '无', text: '无' },
  585. ],
  586. useQualificationNameList: [
  587. {
  588. value: '营业执照正本(行政人事部)',
  589. text: '营业执照正本(行政人事部)'
  590. },
  591. {
  592. value: '营业执照副本(行政人事部)',
  593. text: '营业执照副本(行政人事部)'
  594. },
  595. {
  596. value: '第一类医疗器械生产备案凭证(行政人事部)',
  597. text: '第一类医疗器械生产备案凭证(行政人事部)'
  598. },
  599. {
  600. value: '开户许可证(行政人事部)',
  601. text: '开户许可证(行政人事部)'
  602. },
  603. {
  604. value: '医疗器械生产许可证(行政人事部)',
  605. text: '医疗器械生产许可证(行政人事部)'
  606. },
  607. {
  608. value: '医疗器械生产产品登记表(行政人事部)',
  609. text: '医疗器械生产产品登记表(行政人事部)'
  610. },
  611. {
  612. value: '专利证书(行政人事部)',
  613. text: '专利证书(行政人事部)'
  614. },
  615. {
  616. value: '二类注册证(注册法务部)',
  617. text: '二类注册证(注册法务部)'
  618. },
  619. {
  620. value: '无',
  621. text: '无'
  622. },
  623. ],
  624. qualPopVisible: false,
  625. qualPopType: 'type',
  626. qualEditVisible: false,
  627. qualEditTitle: '添加资质使用',
  628. qualEditForm: { useQualificationType: '', useQualificationRemark: '', useQualificationName: '' },
  629. qualEditMode: 'add',
  630. qualEditScope: null,
  631. qualEditIndex: null,
  632. qualPopIsEdit: true,
  633. // 产品申请
  634. productNameList: [
  635. { value: '一次性使用刀头', text: '一次性使用刀头' },
  636. { value: '一次性使用多功能引流管', text: '一次性使用多功能引流管' },
  637. { value: '强光手电筒+变径接头', text: '强光手电筒+变径接头' },
  638. { value: '手电筒充电器', text: '手电筒充电器' },
  639. { value: '医用冷敷贴疼痛型', text: '医用冷敷贴疼痛型' },
  640. { value: '医用冷敷凝胶防脱育发型', text: '医用冷敷凝胶防脱育发型' },
  641. { value: '医用冷敷凝胶痔疮型', text: '医用冷敷凝胶痔疮型' },
  642. { value: '医用冷敷凝胶皮肤瘙痒型', text: '医用冷敷凝胶皮肤瘙痒型' },
  643. { value: '医用透明质酸钠修复贴', text: '医用透明质酸钠修复贴' },
  644. { value: '一次性射频等离子手术电极', text: '一次性射频等离子手术电极' },
  645. { value: '一次性冲冼吸引器', text: '一次性冲冼吸引器' },
  646. { value: '一次性使用胃管', text: '一次性使用胃管' },
  647. { value: '笔记本套装', text: '笔记本套装' },
  648. ],
  649. appPopVisible: false,
  650. appEditVisible: false,
  651. appEditTitle: '添加产品',
  652. appEditForm: { productName: '', quantity: '', specification: '' },
  653. appEditMode: 'add',
  654. appEditScope: null,
  655. appEditIndex: null,
  656. // 行程出差
  657. tripEditVisible: false,
  658. tripEditTitle: '添加行程',
  659. tripEditForm: { transportation: '', isRoundTrip: '', departureCity: '', destinationCity: '', startTime: '', endTime: '', duration: '' },
  660. tripEditMode: 'add',
  661. tripEditScope: null,
  662. tripEditIndex: null,
  663. // 采购申请
  664. prEditVisible: false,
  665. prEditTitle: '添加物料',
  666. prEditForm: { materialName: '', specification: '', unit: '', quantity: '', purpose: '', expectedDeliveryDate: '', remark: '' },
  667. prEditMode: 'add',
  668. prEditScope: null,
  669. prEditIndex: null,
  670. // 交通费用
  671. transportationWayList: [
  672. {
  673. value: '火车',
  674. text: '火车'
  675. },
  676. {
  677. value: '飞机',
  678. text: '飞机'
  679. },
  680. {
  681. value: '高铁',
  682. text: '高铁'
  683. },
  684. {
  685. value: '公车',
  686. text: '公车'
  687. },
  688. {
  689. value: '私车',
  690. text: '私车'
  691. },
  692. {
  693. value: '大巴',
  694. text: '大巴'
  695. },
  696. {
  697. value: '顺风车',
  698. text: '顺风车'
  699. },
  700. {
  701. value: '滴滴',
  702. text: '滴滴'
  703. }
  704. ],
  705. businessEditVisible: false,
  706. businessEditTitle: '添加交通费用',
  707. businessEditForm: { startDate: '', endDate: '', startPlace: '', endPlace: '', transportationWay: '', price: '' },
  708. businessEditMode: 'add',
  709. businessEditScope: null,
  710. businessEditIndex: null,
  711. businessTransportPopVisible: false,
  712. }
  713. },
  714. created() {
  715. let _this = this
  716. window.x_sun = (e) => {
  717. _this.setSalesServiceWork(JSON.parse(e))
  718. }
  719. // 接收文件选择后的回调
  720. window.x_selectFiles = async (e) => {
  721. let data = JSON.parse(e)
  722. console.log('收到选中的文件:', data)
  723. if (_this.$refs.generateForm) {
  724. if (_this.selectFilesRowIndex != null && _this.selectFilesField) {
  725. // 数组行附件:更新 blank_adopzrdd[index].files
  726. let formData = await _this.$refs.generateForm.getData(false);
  727. var arr = formData[_this.selectFilesField] || []
  728. var row = arr[_this.selectFilesRowIndex]
  729. if (row) {
  730. row.files = data.files || []
  731. _this.$refs.generateForm.setData({
  732. [_this.selectFilesField]: arr
  733. })
  734. }
  735. _this.selectFilesRowIndex = null
  736. } else {
  737. // 普通表单字段
  738. _this.$refs.generateForm.setData({
  739. [data.field]: data.files
  740. })
  741. }
  742. }
  743. }
  744. this.headers = this.getQueryParams('headers');
  745. let params = this.getQueryParams('params');
  746. this.isEdit = params.isEdit || true
  747. this.type = params.type || 'add'
  748. this.APIUrl = this.headers.serverInfo || 'http://192.168.1.105:18086'
  749. axios({
  750. method: 'get',
  751. url: this.APIUrl + `/flowable/bpmcustomform/getById/${params.id}`,
  752. headers: this.headers,
  753. }).then((res) => {
  754. if (res.data.code != '-1') {
  755. this.form = res.data.data
  756. this.form.formId = res.data.data.id
  757. this.form.id = ''
  758. this.form.valueJson = {}
  759. this.title = this.form.name
  760. this.jsonData = JSON.parse(res.data.data.formJson.makingJson);
  761. this.jsonData.list.forEach(item => {
  762. item.options.headers = [{
  763. key: 'Authorization',
  764. value: this.headers.Authorization
  765. }]
  766. item.options.action = item.options.action && item.options.action.replace(
  767. '/api', this.APIUrl)
  768. if (item.type == "deptAndUserCascader") {
  769. item.type = 'cascader'
  770. }
  771. if (item.type == "deptCascader") {
  772. item.type = 'cascader'
  773. /* item.isPathValue = false */
  774. if (item.options.isDefaultLoginUser) {
  775. /* this.form.valueJson[item.model] = [1,1765997946953797633]; */
  776. this.form.valueJson[item.model] = params.userInfo.groupIdList;
  777. }
  778. }
  779. if (item.type == "userSelect") {
  780. item.type = 'select'
  781. if (item.options.isDefaultLoginUser) {
  782. this.form.valueJson[item.model] = params.userInfo.userId;
  783. }
  784. }
  785. })
  786. this.jsonData.config.dataSource && this.jsonData.config.dataSource.forEach(item => {
  787. item.headers = {
  788. Authorization: this.headers.Authorization
  789. }
  790. item.url = item.url && item.url.replace('/api', this.APIUrl)
  791. })
  792. this.isFlag = true
  793. this.$nextTick(() => {
  794. if (params.manage_workorder) {
  795. this.setSalesServiceWork(params.manage_workorder)
  796. }
  797. })
  798. }
  799. });
  800. // axios({
  801. // method: 'get',
  802. // url: this.APIUrl + `/bpm/task/list-by-process-instance-id?processInstanceId=${params.id}`,
  803. // headers: this.headers,
  804. // }).then((res) => {
  805. // console.log(res)
  806. // if (res.data.code != '-1') {
  807. // }
  808. // });
  809. },
  810. methods: {
  811. // ===== 印章编辑弹窗 =====
  812. openSealEditDialog(scope, mode, index) {
  813. this.sealEditVisible = true
  814. this.sealEditScope = scope
  815. this.sealEditIndex = index
  816. this.sealEditMode = mode
  817. if (mode === 'edit' && scope.model.blank_use_seal[index]) {
  818. var row = scope.model.blank_use_seal[index]
  819. this.sealEditForm = { useSealType: row.useSealType||'', useSealRemark: row.useSealRemark||'', useSealName: row.useSealName||'' }
  820. this.sealEditTitle = '编辑印章使用'
  821. } else {
  822. this.sealEditForm = { useSealType: '', useSealRemark: '', useSealName: '' }
  823. this.sealEditTitle = '添加印章使用'
  824. }
  825. },
  826. async onSealEditSave() {
  827. var scope = this.sealEditScope
  828. var idx = this.sealEditIndex
  829. var key = 'blank_use_seal'
  830. var data = await this.$refs.generateForm.getData(false);
  831. if (!data[key]) data[key] = []
  832. if (this.sealEditMode === 'edit') {
  833. data[key][idx] = JSON.parse(JSON.stringify(this.sealEditForm))
  834. } else {
  835. data[key].push(JSON.parse(JSON.stringify(this.sealEditForm)))
  836. }
  837. this.$refs.generateForm.setData({ [key]: data[key] })
  838. this.sealEditVisible = false
  839. },
  840. openSealPickerInEdit(fieldType) {
  841. this.sealPopType = fieldType
  842. this.sealPopScope = null
  843. this.sealPopIndex = null
  844. this.sealPopIsEdit = true
  845. this.sealPopVisible = true
  846. },
  847. openSealPicker(scope, index, fieldType) {
  848. this.sealPopScope = scope
  849. this.sealPopIndex = index
  850. this.sealPopType = fieldType || 'type'
  851. this.sealPopIsEdit = false
  852. this.sealPopVisible = true
  853. },
  854. onSealPopConfirm(picker) {
  855. var sel = (picker && picker.selectedOptions && picker.selectedOptions[0]) ? picker.selectedOptions[0].text : picker
  856. var field = this.sealPopType === 'name' ? 'useSealName' : 'useSealType'
  857. if (this.sealPopIsEdit) {
  858. // 编辑弹窗模式:回写到 sealEditForm
  859. this.sealEditForm[field] = sel
  860. } else {
  861. // 列表内联模式
  862. var scope = this.sealPopScope
  863. var idx = this.sealPopIndex
  864. if (scope && scope.model && scope.model.blank_use_seal && scope.model.blank_use_seal[idx]) {
  865. scope.model.blank_use_seal[idx][field] = sel
  866. }
  867. }
  868. this.sealPopVisible = false
  869. },
  870. // ===== 资质使用 =====
  871. openQualEditDialog(scope, mode, index) {
  872. this.qualEditScope = scope
  873. this.qualEditIndex = index
  874. this.qualEditMode = mode
  875. if (mode === 'edit' && scope.model.blank_use_qualification[index]) {
  876. var r = scope.model.blank_use_qualification[index]
  877. this.qualEditForm = { useQualificationType: r.useQualificationType||'', useQualificationRemark: r.useQualificationRemark||'', useQualificationName: r.useQualificationName||'' }
  878. this.qualEditTitle = '编辑资质使用'
  879. } else {
  880. this.qualEditForm = { useQualificationType: '', useQualificationRemark: '', useQualificationName: '' }
  881. this.qualEditTitle = '添加资质使用'
  882. }
  883. this.qualEditVisible = true
  884. },
  885. async onQualEditSave() {
  886. var scope = this.qualEditScope
  887. var idx = this.qualEditIndex
  888. var key = 'blank_use_qualification'
  889. var data = await this.$refs.generateForm.getData(false);
  890. if (!data[key]) data[key] = []
  891. if (this.qualEditMode === 'edit') {
  892. data[key][idx] = JSON.parse(JSON.stringify(this.qualEditForm))
  893. } else {
  894. data[key].push(JSON.parse(JSON.stringify(this.qualEditForm)))
  895. }
  896. this.$refs.generateForm.setData({ [key]: data[key] })
  897. this.qualEditVisible = false
  898. },
  899. openQualPickerInEdit(fieldType) {
  900. this.qualPopType = fieldType
  901. this.qualPopIsEdit = true
  902. this.qualPopVisible = true
  903. },
  904. onQualPopConfirm(picker) {
  905. var sel = (picker && picker.selectedOptions && picker.selectedOptions[0]) ? picker.selectedOptions[0].text : picker
  906. var field = this.qualPopType === 'name' ? 'useQualificationName' : 'useQualificationType'
  907. this.qualEditForm[field] = sel
  908. this.qualPopVisible = false
  909. },
  910. // ===== 产品申请 =====
  911. openAppEditDialog(scope, mode, index) {
  912. this.appEditScope = scope
  913. this.appEditIndex = index
  914. this.appEditMode = mode
  915. if (mode === 'edit' && scope.model.blank_application_component[index]) {
  916. var r = scope.model.blank_application_component[index]
  917. this.appEditForm = { productName: r.productName||'', quantity: r.quantity||'', specification: r.specification||'' }
  918. this.appEditTitle = '编辑产品'
  919. } else {
  920. this.appEditForm = { productName: '', quantity: '', specification: '' }
  921. this.appEditTitle = '添加产品'
  922. }
  923. this.appEditVisible = true
  924. },
  925. async onAppEditSave() {
  926. var scope = this.appEditScope
  927. var idx = this.appEditIndex
  928. var key = 'blank_application_component'
  929. var data = await this.$refs.generateForm.getData(false);
  930. if (!data[key]) data[key] = []
  931. if (this.appEditMode === 'edit') {
  932. data[key][idx] = JSON.parse(JSON.stringify(this.appEditForm))
  933. } else {
  934. data[key].push(JSON.parse(JSON.stringify(this.appEditForm)))
  935. }
  936. this.$refs.generateForm.setData({ [key]: data[key] })
  937. this.appEditVisible = false
  938. },
  939. openAppPickerInEdit() {
  940. this.appPopVisible = true
  941. },
  942. onAppPopConfirm(picker) {
  943. var sel = (picker && picker.selectedOptions && picker.selectedOptions[0]) ? picker.selectedOptions[0].text : picker
  944. this.appEditForm.productName = sel
  945. this.appPopVisible = false
  946. },
  947. // ===== 行程出差 =====
  948. openTripEditDialog(scope, mode, index) {
  949. this.tripEditScope = scope
  950. this.tripEditIndex = index
  951. this.tripEditMode = mode
  952. if (mode === 'edit' && scope.model.blank_business_trip[index]) {
  953. var r = scope.model.blank_business_trip[index]
  954. this.tripEditForm = {
  955. transportation: r.transportation||'', isRoundTrip: r.isRoundTrip||'',
  956. departureCity: r.departureCity||'', destinationCity: r.destinationCity||'',
  957. startTime: r.startTime||'', endTime: r.endTime||'', duration: r.duration||''
  958. }
  959. this.tripEditTitle = '编辑行程'
  960. } else {
  961. this.tripEditForm = { transportation: '', isRoundTrip: '', departureCity: '', destinationCity: '', startTime: '', endTime: '', duration: '' }
  962. this.tripEditTitle = '添加行程'
  963. }
  964. this.tripEditVisible = true
  965. },
  966. calcTripDuration() {
  967. var s = this.tripEditForm.startTime, e = this.tripEditForm.endTime
  968. if (!s || !e) { this.tripEditForm.duration = ''; return }
  969. var ds = new Date(s).getTime(), de = new Date(e).getTime()
  970. if (isNaN(ds) || isNaN(de) || de <= ds) { this.tripEditForm.duration = ''; return }
  971. var diff = Math.floor((de - ds) / 1000)
  972. var d = Math.floor(diff / 86400), h = Math.floor((diff % 86400) / 3600), m = Math.floor((diff % 3600) / 60), sec = diff % 60
  973. var parts = []
  974. if (d) parts.push(d + '天'); if (h) parts.push(h + '时'); if (m) parts.push(m + '分'); if (sec || !parts.length) parts.push(sec + '秒')
  975. this.tripEditForm.duration = parts.join('')
  976. },
  977. async onTripEditSave() {
  978. var scope = this.tripEditScope
  979. var idx = this.tripEditIndex
  980. var key = 'blank_business_trip'
  981. var data = await this.$refs.generateForm.getData(false);
  982. if (!data[key]) data[key] = []
  983. if (this.tripEditMode === 'edit') {
  984. data[key][idx] = JSON.parse(JSON.stringify(this.tripEditForm))
  985. } else {
  986. data[key].push(JSON.parse(JSON.stringify(this.tripEditForm)))
  987. }
  988. await this.$refs.generateForm.setData({ [key]: data[key] })
  989. this.tripEditVisible = false
  990. await this.calcTripTotal()
  991. },
  992. async calcTripTotal() {
  993. var data = await this.$refs.generateForm.getData(false);
  994. var list = data.blank_business_trip || []
  995. var totalSec = 0
  996. console.log('calcTripTotal', data, list)
  997. list.forEach(function(it) {
  998. var d = it.duration
  999. if (!d) return
  1000. var match = { days: d.match(/(\d+)天/), hours: d.match(/(\d+)时/), minutes: d.match(/(\d+)分/), seconds: d.match(/(\d+)秒/) }
  1001. totalSec += (parseInt(match.days&&match.days[1])||0)*86400 + (parseInt(match.hours&&match.hours[1])||0)*3600 + (parseInt(match.minutes&&match.minutes[1])||0)*60 + (parseInt(match.seconds&&match.seconds[1])||0)
  1002. })
  1003. if (totalSec > 0) {
  1004. var dy = Math.floor(totalSec/86400), hr = Math.floor((totalSec%86400)/3600), mn = Math.floor((totalSec%3600)/60), sc = totalSec%60
  1005. var parts = []
  1006. if (dy) parts.push(dy+'天'); if (hr) parts.push(hr+'时'); if (mn) parts.push(mn+'分'); if (sc||!parts.length) parts.push(sc+'秒')
  1007. this.$refs.generateForm.setData({ input_if1cav3p: parts.join('') })
  1008. }
  1009. },
  1010. // ===== 采购申请 =====
  1011. openPrEditDialog(scope, mode, index) {
  1012. this.prEditScope = scope; this.prEditIndex = index; this.prEditMode = mode
  1013. if (mode === 'edit' && scope.model.blank_purchase_requisition[index]) {
  1014. var r = scope.model.blank_purchase_requisition[index]
  1015. this.prEditForm = { materialName: r.materialName||'', specification: r.specification||'', unit: r.unit||'', quantity: r.quantity||'', purpose: r.purpose||'', expectedDeliveryDate: r.expectedDeliveryDate||'', remark: r.remark||'' }
  1016. this.prEditTitle = '编辑物料'
  1017. } else {
  1018. this.prEditForm = { materialName: '', specification: '', unit: '', quantity: '', purpose: '', expectedDeliveryDate: '', remark: '' }
  1019. this.prEditTitle = '添加物料'
  1020. }
  1021. this.prEditVisible = true
  1022. },
  1023. async onPrEditSave() {
  1024. var scope = this.prEditScope, idx = this.prEditIndex, key = 'blank_purchase_requisition'
  1025. var data = await this.$refs.generateForm.getData(false)
  1026. if (!data[key]) data[key] = []
  1027. if (this.prEditMode === 'edit') data[key][idx] = JSON.parse(JSON.stringify(this.prEditForm))
  1028. else data[key].push(JSON.parse(JSON.stringify(this.prEditForm)))
  1029. this.$refs.generateForm.setData({ [key]: data[key] })
  1030. this.prEditVisible = false
  1031. },
  1032. // ===== 交通费用 =====
  1033. openBusinessEditDialog(scope, mode, index) {
  1034. this.businessEditScope = scope
  1035. this.businessEditIndex = index
  1036. this.businessEditMode = mode
  1037. if (mode === 'edit' && scope.model.blank_business_component[index]) {
  1038. var r = scope.model.blank_business_component[index]
  1039. this.businessEditForm = { startDate: r.startDate||'', endDate: r.endDate||'', startPlace: r.startPlace||'', endPlace: r.endPlace||'', transportationWay: r.transportationWay||'', price: r.price||'' }
  1040. this.businessEditTitle = '编辑交通费用'
  1041. } else {
  1042. this.businessEditForm = { startDate: '', endDate: '', startPlace: '', endPlace: '', transportationWay: '', price: '' }
  1043. this.businessEditTitle = '添加交通费用'
  1044. }
  1045. this.businessEditVisible = true
  1046. },
  1047. async onBusinessEditSave() {
  1048. var scope = this.businessEditScope, idx = this.businessEditIndex, key = 'blank_business_component'
  1049. var data = await this.$refs.generateForm.getData(false)
  1050. if (!data[key]) data[key] = []
  1051. if (this.businessEditMode === 'edit') data[key][idx] = JSON.parse(JSON.stringify(this.businessEditForm))
  1052. else data[key].push(JSON.parse(JSON.stringify(this.businessEditForm)))
  1053. await this.$refs.generateForm.setData({ [key]: data[key] })
  1054. this.businessEditVisible = false
  1055. await this.calcBizTotal()
  1056. },
  1057. onBusinessTransportConfirm(picker) {
  1058. var sel = (picker && picker.selectedOptions && picker.selectedOptions[0]) ? picker.selectedOptions[0].text : picker
  1059. this.businessEditForm.transportationWay = sel
  1060. this.businessTransportPopVisible = false
  1061. },
  1062. async calcBizTotal() {
  1063. var data = await this.$refs.generateForm.getData(false)
  1064. var list = data.blank_business_component || []
  1065. var totalPrice = 0
  1066. list.forEach(function(it) {
  1067. totalPrice += Number(it.price) || 0
  1068. })
  1069. // 其他费用字段(住宿费、市内交通平台、餐费补贴、市内交通费、招待费、其他费用)
  1070. var zsf = Number(data.number_uyuxcowe) || 0
  1071. var snjtpt = Number(data.number_0zw5q8y9) || 0
  1072. var cfbt = Number(data.number_fyl8n1lm) || 0
  1073. var snjtf = Number(data.number_p093shsw) || 0
  1074. var zdf = Number(data.number_uiv62o0r) || 0
  1075. var qtfy = Number(data.number_i5y1dk9x) || 0
  1076. this.$refs.generateForm.setData({
  1077. number_06fgz5gs: totalPrice + zsf + snjtpt + cfbt + snjtf + zdf + qtfy
  1078. })
  1079. },
  1080. async handleUploadLocal (model, rowIndex) {
  1081. console.log('上传本地文件按钮被点击,字段模型:', model, '行索引:', rowIndex)
  1082. let data = await this.$refs.generateForm.getData(false);
  1083. let files = rowIndex != null && data[model] && data[model][rowIndex] ? (data[model][rowIndex].files || []) : (data[model] || [])
  1084. this.selectFilesField = model
  1085. this.selectFilesRowIndex = rowIndex != null ? rowIndex : null
  1086. uni.postMessage({
  1087. data: {
  1088. type: 'selectFiles',
  1089. field: model,
  1090. files: files,
  1091. isEdit: this.type || 'add'
  1092. }
  1093. });
  1094. },
  1095. addWorkOrder() {
  1096. uni.postMessage({
  1097. data: {
  1098. type: 'selectWork',
  1099. }
  1100. });
  1101. },
  1102. async addNewRow(key) {
  1103. console.log(key);
  1104. let data = await this.$refs.generateForm.getData(false);
  1105. data[key].push({
  1106. price: '',
  1107. remark: ''
  1108. });
  1109. this.$refs.generateForm.setData({
  1110. key: data[key]
  1111. });
  1112. },
  1113. async delNewRow(key, index, isChange) {
  1114. let data = await this.$refs.generateForm.getData(false);
  1115. data[key] = data[key].filter((item, index1) => index1 != index);
  1116. await this.$refs.generateForm.setData({
  1117. [key]: data[key]
  1118. });
  1119. if (isChange) {
  1120. this.changePrice(data[key])
  1121. }
  1122. },
  1123. changePrice(data) {
  1124. let num = 0;
  1125. data.forEach((item) => {
  1126. num += Number(item.price);
  1127. });
  1128. this.$refs.generateForm.setData({
  1129. input_jd9ouzyh: num
  1130. });
  1131. },
  1132. setSalesServiceWork(data) {
  1133. this.$refs.generateForm.setData({
  1134. manage_workorder: data
  1135. });
  1136. this.$refs.generateForm.setRules('manage_workorder', [{
  1137. required: false,
  1138. message: '必须填写'
  1139. }])
  1140. setTimeout(() => {
  1141. this.$refs.generateForm.validate(['manage_workorder'])
  1142. })
  1143. },
  1144. getQueryParams(queryName) {
  1145. const urlSearchParams = new URLSearchParams(window.location.search);
  1146. const query = urlSearchParams.get(queryName);
  1147. console.log(query, 'query')
  1148. return JSON.parse(query);
  1149. },
  1150. generateFormValid(validate = true) {
  1151. return this.$refs.generateForm.getData(validate).then((data) => { //清空content
  1152. for (key in data) {
  1153. if (key.includes('fileupload')) {
  1154. data[key].forEach(item => {
  1155. item['content'] = ''
  1156. })
  1157. }
  1158. }
  1159. return data;
  1160. });
  1161. },
  1162. async handleAudit(status) {
  1163. console.log(1111111, status)
  1164. let formData = await this.$refs.generateForm.getData(false);
  1165. console.log('formData~~~~', formData)
  1166. this.loading = true;
  1167. console.log(this.loading)
  1168. try {
  1169. this.form.valueJson = await this.generateFormValid();
  1170. } catch (error) {
  1171. console.log(22222, error)
  1172. this.loading = false;
  1173. return;
  1174. }
  1175. console.log(this.form.valueJson)
  1176. this.form.processType = '1';
  1177. let API = this.APIUrl + '/bpm/process-instance/create'
  1178. axios({
  1179. method: 'post',
  1180. url: API,
  1181. headers: this.headers,
  1182. data: {
  1183. ...this.form,
  1184. }
  1185. }).then((res) => {
  1186. this.loading = false;
  1187. if (res.data.code != '-1') {
  1188. vant.showNotify({
  1189. type: 'success',
  1190. message: `提交成功!`,
  1191. duration: 1000,
  1192. });
  1193. setTimeout(() => {
  1194. this.onClickLeft()
  1195. }, 1000)
  1196. } else {
  1197. vant.showNotify({
  1198. type: 'danger',
  1199. message: `提交失败!${res.data.message}`,
  1200. duration: 1000,
  1201. });
  1202. }
  1203. }).catch((err) => {
  1204. vant.showNotify({
  1205. type: 'danger',
  1206. message: `提交失败!${err.message}`,
  1207. duration: 1000,
  1208. });
  1209. this.loading = false;
  1210. });
  1211. },
  1212. onClickLeft() {
  1213. uni.navigateBack({
  1214. delta: 1
  1215. });
  1216. }
  1217. }
  1218. }).use(vant).use(FormMakingV3).mount('#app')
  1219. </script>
  1220. </body>
  1221. </html>