customsData.vue 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221
  1. <template>
  2. <div class="search-form">
  3. <!-- 原有的查询条件区域 -->
  4. <div class="query-container">
  5. <a-row :gutter="16">
  6. <a-col :span="6">
  7. <a-form-item label="产品" labelAlign="left">
  8. <a-input v-model:value="form.product" placeholder="产品" />
  9. </a-form-item>
  10. </a-col>
  11. <a-col :span="6">
  12. <a-form-item label="HS编码" labelAlign="left">
  13. <a-input v-model:value="form.hsCode" placeholder="HS编码" />
  14. </a-form-item>
  15. </a-col>
  16. <a-col :span="6">
  17. <a-form-item label="起始日期" labelAlign="left">
  18. <a-date-picker v-model:value="form.startDate" placeholder="起始日期" />
  19. </a-form-item>
  20. </a-col>
  21. <a-col :span="6">
  22. <a-form-item label="截止日期" labelAlign="left">
  23. <a-date-picker v-model:value="form.endDate" placeholder="截止日期" />
  24. </a-form-item>
  25. </a-col>
  26. </a-row>
  27. <a-row :gutter="16">
  28. <a-col :span="6">
  29. <a-form-item label="供应商" labelAlign="left">
  30. <a-input v-model:value="form.supplier" placeholder="供应商" />
  31. </a-form-item>
  32. </a-col>
  33. <a-col :span="6">
  34. <a-form-item label="供应商注册号" labelAlign="left">
  35. <a-input v-model:value="form.supplierReg" placeholder="供应商注册号" />
  36. </a-form-item>
  37. </a-col>
  38. <a-col :span="6">
  39. <a-form-item label="供应商地址" labelAlign="left">
  40. <a-input v-model:value="form.supplierAddress" placeholder="供应商地址" />
  41. </a-form-item>
  42. </a-col>
  43. <a-col :span="6">
  44. <div class="checkbox-group">
  45. <a-checkbox-group v-model:value="form.options">
  46. <a-checkbox value="excludeLogistics">去除物流</a-checkbox>
  47. <a-checkbox value="excludeNVL">去除NVL</a-checkbox>
  48. </a-checkbox-group>
  49. </div>
  50. </a-col>
  51. </a-row>
  52. <a-row :gutter="16">
  53. <a-col :span="6">
  54. <a-form-item label="采购商" labelAlign="left">
  55. <a-input v-model:value="form.buyer" placeholder="采购商" />
  56. </a-form-item>
  57. </a-col>
  58. <a-col :span="6">
  59. <a-form-item label="采购商注册号" labelAlign="left">
  60. <a-input v-model:value="form.buyerReg" placeholder="采购商注册号" />
  61. </a-form-item>
  62. </a-col>
  63. <a-col :span="6">
  64. <a-form-item label="采购商地址" labelAlign="left">
  65. <a-input v-model:value="form.buyerAddress" placeholder="采购商地址" />
  66. </a-form-item>
  67. </a-col>
  68. <a-col :span="6">
  69. <div class="checkbox-group">
  70. <a-checkbox-group v-model:value="form.buyerOptions">
  71. <a-checkbox value="excludeLogistics">去除物流</a-checkbox>
  72. <a-checkbox value="excludeNVL">去除NVL</a-checkbox>
  73. </a-checkbox-group>
  74. </div>
  75. </a-col>
  76. </a-row>
  77. <!-- 额外搜索框 -->
  78. <div v-if="showAdvancedSearch">
  79. <a-row :gutter="16">
  80. <a-col :span="6">
  81. <a-form-item label="重量范围" labelAlign="left">
  82. <a-input-group compact>
  83. <a-input style="width: 50%" v-model="form.weightMin" placeholder="最小值" />
  84. <a-input style="width: 50%" v-model="form.weightMax" placeholder="最大值" />
  85. </a-input-group>
  86. </a-form-item>
  87. </a-col>
  88. <a-col :span="6">
  89. <a-form-item label="数量范围" labelAlign="left">
  90. <a-input-group compact>
  91. <a-input style="width: 50%" v-model="form.quantityMin" placeholder="最小值" />
  92. <a-input style="width: 50%" v-model="form.quantityMax" placeholder="最大值" />
  93. </a-input-group>
  94. </a-form-item>
  95. </a-col>
  96. <a-col :span="6">
  97. <a-form-item label="金额范围" labelAlign="left">
  98. <a-input-group compact>
  99. <a-input style="width: 50%" v-model="form.amountMin" placeholder="最小值" />
  100. <a-input style="width: 50%" v-model="form.amountMax" placeholder="最大值" />
  101. </a-input-group>
  102. </a-form-item>
  103. </a-col>
  104. <a-col :span="6">
  105. <a-form-item label="TEU范围" labelAlign="left">
  106. <a-input-group compact>
  107. <a-input style="width: 50%" v-model="form.teuMin" placeholder="最小值" />
  108. <a-input style="width: 50%" v-model="form.teuMax" placeholder="最大值" />
  109. </a-input-group>
  110. </a-form-item>
  111. </a-col>
  112. </a-row>
  113. <a-row :gutter="16">
  114. <a-col :span="6">
  115. <a-form-item label="原产国" labelAlign="left">
  116. <a-input v-model:value="form.originCountry" placeholder="原产国" />
  117. </a-form-item>
  118. </a-col>
  119. <a-col :span="6">
  120. <a-form-item label="启运港" labelAlign="left">
  121. <a-input v-model:value="form.origPort" placeholder="启运港" />
  122. </a-form-item>
  123. </a-col>
  124. <a-col :span="6">
  125. <a-form-item label="目的国" labelAlign="left">
  126. <a-input v-model:value="form.destinationCountry" placeholder="目的国" />
  127. </a-form-item>
  128. </a-col>
  129. <a-col :span="6">
  130. <a-form-item label="目的港" labelAlign="left">
  131. <a-input v-model:value="form.destPort" placeholder="目的港" />
  132. </a-form-item>
  133. </a-col>
  134. </a-row>
  135. <a-row :gutter="16">
  136. <a-col :span="6">
  137. <a-form-item label="海关" labelAlign="left">
  138. <a-input v-model:value="form.customs" placeholder="海关" />
  139. </a-form-item>
  140. </a-col>
  141. <a-col :span="6">
  142. <a-form-item label="运输方式" labelAlign="left">
  143. <a-input v-model:value="form.transportMode" placeholder="运输方式" />
  144. </a-form-item>
  145. </a-col>
  146. <a-col :span="6">
  147. <a-form-item label="成交方式" labelAlign="left">
  148. <a-input v-model:value="form.incoterms" placeholder="成交方式" />
  149. </a-form-item>
  150. </a-col>
  151. <a-col :span="6">
  152. <a-form-item label="主单号" labelAlign="left">
  153. <a-input v-model:value="form.mainOrderNo" placeholder="主单号" />
  154. </a-form-item>
  155. </a-col>
  156. </a-row>
  157. <a-row :gutter="16">
  158. <a-col :span="6">
  159. <a-form-item label="分单号" labelAlign="left">
  160. <a-input v-model:value="form.subOrderNo" placeholder="分单号" />
  161. </a-form-item>
  162. </a-col>
  163. <a-col :span="6">
  164. <a-form-item label="集装箱号" labelAlign="left">
  165. <a-input v-model:value="form.containerNo" placeholder="集装箱号" />
  166. </a-form-item>
  167. </a-col>
  168. <a-col :span="6">
  169. <a-form-item label="承运人" labelAlign="left">
  170. <a-input v-model:value="form.carrier" placeholder="承运人" />
  171. </a-form-item>
  172. </a-col>
  173. <a-col :span="6">
  174. <a-form-item label="船名" labelAlign="left">
  175. <a-input v-model:value="form.shipName" placeholder="船名" />
  176. </a-form-item>
  177. </a-col>
  178. </a-row>
  179. <a-row :gutter="16">
  180. <a-col :span="6">
  181. <a-form-item label="商标" labelAlign="left">
  182. <a-input v-model:value="form.brand" placeholder="商标" />
  183. </a-form-item>
  184. </a-col>
  185. <a-col :span="18">
  186. <a-form-item label="其他" labelAlign="left">
  187. <a-input v-model:value="form.other" placeholder="其他" />
  188. </a-form-item>
  189. </a-col>
  190. </a-row>
  191. </div>
  192. <!-- 更多搜索按钮 -->
  193. <a-row :gutter="16" style="margin-top: 16px">
  194. <a-col :span="6">
  195. <a-button @click="toggleAdvancedSearch">
  196. {{ showAdvancedSearch ? '收起' : '更多搜索' }}
  197. </a-button>
  198. </a-col>
  199. </a-row>
  200. <!-- 按钮行 -->
  201. <a-row class="button-row" justify="end">
  202. <a-button type="primary" @click="handleSearch">搜索</a-button>
  203. </a-row>
  204. </div>
  205. </div>
  206. <!-- 使用 v-show 控制数据展示区域的显示/隐藏 -->
  207. <div v-show="showDataContent" class="table-container">
  208. <a-tabs v-model:activeKey="activeTabKey" @change="handleTabChange" class="custom-tabs" type="card">
  209. <a-tab-pane key="transaction" tab="交易信息">
  210. <a-card title="交易信息">
  211. <template #extra>
  212. <span>排序:</span>
  213. <a-radio-group v-model:value="sortMode" button-style="solid" style="margin-left: 16px;"
  214. @change="handleSortChange">
  215. <a-radio-button value="date">交易日期</a-radio-button>
  216. <a-radio-button value="weight">交易重量</a-radio-button>
  217. </a-radio-group>
  218. </template>
  219. <a-table :columns="columns" :data-source="tableData" :loading="loading" :pagination="{
  220. current: pagination.current,
  221. pageSize: pagination.pageSize,
  222. total: pagination.total,
  223. showSizeChanger: true,
  224. showQuickJumper: true,
  225. showTotal: (total) => `共 ${total} 条`,
  226. onChange: handleTableChange,
  227. onShowSizeChange: handleTableChange,
  228. }" @change="handleTableChange">
  229. <template #bodyCell="{ column, record, text }">
  230. <template v-if="column.key === 'supplier_t'">
  231. <a @click="handleSupplierClick(record.supplier_id)">{{ text }}</a>
  232. </template>
  233. <template v-if="column.key === 'buyer_t'">
  234. <a @click="handleBuyerClick(record.buyer_id)">{{ text }}</a>
  235. </template>
  236. <template v-if="column.key === 'freight_intro'">
  237. <a @click="handleFreightIntroClick()">详细信息</a>
  238. </template>
  239. </template>
  240. </a-table>
  241. </a-card>
  242. <div :inert="isModalVisible">
  243. <!-- 其他内容 -->
  244. <a-modal v-model:open="isModalVisible" title="企业详情" :width="1200" @close="closeModal">
  245. <div>
  246. <div v-if="supplier">
  247. <h1>{{ supplier.name }}</h1> <!-- 添加企业名称 -->
  248. <p><strong>地址:</strong> {{ supplier.address }}</p> <!-- 添加企业地址 -->
  249. <p v-if="supplier.postal_code"><strong>联系方式:</strong> postal code:{{ supplier.postal_code }}</p>
  250. <!-- 添加联系方式 -->
  251. </div>
  252. </div>
  253. <!-- 选项卡 -->
  254. <a-tabs v-model:activeKey="activeTabKey1" @change="handleTabChange1" type="card">
  255. <a-tab-pane key="tradeOverview" tab="贸易总览">
  256. <!-- 贸易总览内容 -->
  257. <div class="analysis-content">
  258. <div class="analysis-item">
  259. <FreightHistory :data="freightHistoryData" />
  260. </div>
  261. <div class="analysis-item">
  262. <CompanyProduct :hsCodeData="companyHsCodeData" />
  263. </div>
  264. <div class="analysis-item">
  265. <TradePartners :data="tradePartnersData" />
  266. </div>
  267. <div class="analysis-item">
  268. <RegionDistribution :data="regionDistributionData" />
  269. </div>
  270. </div>
  271. </a-tab-pane>
  272. <a-tab-pane key="importDetails" tab="进口详单">
  273. <!-- 新增表格展示进口详单 -->
  274. <a-table :columns="importDetailsColumns" :data-source="importDetailsData" :loading="loading"
  275. :pagination="{
  276. current: paginationImportDetails.current,
  277. pageSize: paginationImportDetails.pageSize,
  278. total: paginationImportDetails.total,
  279. showSizeChanger: true,
  280. showQuickJumper: true,
  281. showTotal: (total) => `共 ${total} 条`,
  282. onChange: handleTableChangeImportDetails,
  283. }" />
  284. </a-tab-pane>
  285. <a-tab-pane key="exportDetails" tab="出口详单">
  286. <a-table :columns="exportDetailsColumns" :data-source="exportDetailsData" :loading="loading"
  287. :pagination="{
  288. current: paginationExportDetails.current,
  289. pageSize: paginationExportDetails.pageSize,
  290. total: paginationExportDetails.total,
  291. showSizeChanger: true,
  292. showQuickJumper: true,
  293. showTotal: (total) => `共 ${total} 条`,
  294. onChange: handleTableChangeExportDetails,
  295. }" />
  296. </a-tab-pane>
  297. </a-tabs>
  298. </a-modal>
  299. </div>
  300. </a-tab-pane>
  301. <a-tab-pane key="companies" tab="企业列表">
  302. <company-list :companies="companiesData" @compare="handleCompareClick" />
  303. </a-tab-pane>
  304. <a-tab-pane key="tradeAnalysis" tab="贸易类分析报告">
  305. <div class="analysis-content">
  306. <div class="analysis-item">
  307. <trade-analysis :monthly-trend-data="monthlyTrendData" />
  308. </div>
  309. <div class="analysis-item">
  310. <HsCodeAnalysis :hs-code-data="hsCodeData" @update:sortMode="handleSortMode" />
  311. </div>
  312. <a-row :gutter="16">
  313. <!-- 新增行 -->
  314. <a-col :span="12">
  315. <!-- 左侧列 -->
  316. <div class="analysis-item">
  317. <OriginCountryAnalysis :origin-country-data="originCountryData" @update:sortMode="handleSortModeOriginCountryData" />
  318. </div>
  319. </a-col>
  320. <a-col :span="12">
  321. <!-- 右侧列 -->
  322. <div class="analysis-item">
  323. <DestinationCountryAnalysis :destination-country-data="destinationCountryData" @update:sortMode="handleSortModeDestinationCountryData"/>
  324. </div>
  325. </a-col>
  326. </a-row>
  327. <div class="analysis-item">
  328. <SupplierList :queryParam = "queryParam"/>
  329. </div>
  330. <div class="analysis-item">
  331. <BuyerList :queryParam = "queryParam" />
  332. </div>
  333. </div>
  334. </a-tab-pane>
  335. <a-tab-pane key="shippingAnalysis" tab="航运类分析报告">
  336. <div class="analysis-content">
  337. <div class="analysis-item">
  338. <OrigPortAnalysis :origPortData="origPortData"></OrigPortAnalysis>
  339. </div>
  340. <div class="analysis-item">
  341. <DestPortAnalysis :destPortData="destPortData"></DestPortAnalysis>
  342. </div>
  343. <div class="analysis-item">
  344. <TransTypeAnalysis :transTypeData="transTypeData"></TransTypeAnalysis>
  345. </div>
  346. <div class="analysis-item">
  347. <IncotermsAnalysis :incotermsData="incotermsData"></IncotermsAnalysis>
  348. </div>
  349. </div>
  350. </a-tab-pane>
  351. </a-tabs>
  352. </div>
  353. </template>
  354. <script lang="ts" setup>
  355. import { reactive, ref, onMounted } from 'vue';
  356. import dayjs, { Dayjs } from 'dayjs';
  357. import {
  358. list,
  359. listCompanies,
  360. getTrendReport,
  361. getHsCodeReport,
  362. getOrigCountryReport,
  363. getDestCountryReport,
  364. getSupplierReport,
  365. getBuyerReport,
  366. getOrigPortReport,
  367. getDestPortReport,
  368. getTransTypeReport,
  369. getIncotermsReport,
  370. getCompanyInfo,
  371. getMonthly,
  372. getCompanyHsCode,
  373. getCompanyPartner,
  374. getRegionDistribution,
  375. getCompanyRecord
  376. } from './customsData.api';
  377. import { columns } from './customsData.data';
  378. import CompanyList from './components/CompanyList.vue';
  379. import TradeAnalysis from './components/TradeAnalysis.vue';
  380. import HsCodeAnalysis from './components/HsCodeAnalysis.vue';
  381. import OriginCountryAnalysis from './components/OriginCountryAnalysis.vue';
  382. import DestinationCountryAnalysis from './components/DestinationCountryAnalysis.vue';
  383. import SupplierList from './components/SupplierList.vue';
  384. import BuyerList from './components/BuyerList.vue';
  385. import OrigPortAnalysis from './components/OrigPortAnalysis.vue';
  386. import DestPortAnalysis from './components/DestPortAnalysis.vue';
  387. import TransTypeAnalysis from './components/TransTypeAnalysis.vue';
  388. import IncotermsAnalysis from './components/IncotermsAnalysis.vue';
  389. import FreightHistory from './components/FreightHistory.vue';
  390. import TradePartners from './components/TradePartners.vue';
  391. import RegionDistribution from './components/RegionDistribution.vue';
  392. import CompanyProduct from './components/CompanyProduct.vue';
  393. const supplier = ref<{ name: string; address: string; postal_code: string } | null>(null);
  394. // 当前激活的标签页
  395. const activeTabKey = ref('transaction');
  396. // 查询参数
  397. const queryParam = reactive<any>({});
  398. const form = ref({
  399. product: '',
  400. hsCode: '',
  401. startDate: dayjs('20230101'),//后面更换
  402. endDate: dayjs('20230630'),//后面更换
  403. supplier: '',
  404. supplierReg: '',
  405. supplierAddress: '',
  406. options: ['excludeNVL'],
  407. buyer: '',
  408. buyerReg: '',
  409. buyerAddress: '',
  410. buyerOptions: ['excludeNVL'],
  411. originCountry: '',
  412. destinationCountry: '',
  413. port: '',
  414. transportMode: '',
  415. weightMin: null,
  416. weightMax: null,
  417. quantityMin: null,
  418. quantityMax: null,
  419. amountMin: null,
  420. amountMax: null,
  421. teuMin: null,
  422. teuMax: null,
  423. origPort: '',
  424. destPort: '',
  425. customs: '',
  426. incoterms: '',
  427. mainOrderNo: '',
  428. subOrderNo: '',
  429. containerNo: '',
  430. carrier: '',
  431. shipName: '',
  432. brand: '',
  433. other: '',
  434. });
  435. // 添加控制数据展示的状态
  436. const showDataContent = ref(false);
  437. // 修改 handleSearch 方法
  438. const handleSearch = async () => {
  439. showDataContent.value = true;
  440. // 构建查询参数
  441. const params = {
  442. // 固定参数
  443. source_type: 1,
  444. data_source: ['IMP_AMERICA_BL_SEA'],
  445. // 表单参数
  446. prod_desc: form.value.product,
  447. hs_code: form.value.hsCode,
  448. supplier: form.value.supplier,
  449. supplier_reg: form.value.supplierReg,
  450. supplier_address: form.value.supplierAddress,
  451. buyer: form.value.buyer,
  452. buyer_reg: form.value.buyerReg,
  453. buyer_address: form.value.buyerAddress,
  454. origin_country: form.value.originCountry,
  455. destination_country: form.value.destinationCountry,
  456. orig_port: form.value.origPort,
  457. dest_port: form.value.destPort,
  458. customs: form.value.customs,
  459. transport_mode: form.value.transportMode,
  460. incoterms: form.value.incoterms,
  461. main_order_no: form.value.mainOrderNo,
  462. sub_order_no: form.value.subOrderNo,
  463. container_no: form.value.containerNo,
  464. carrier: form.value.carrier,
  465. ship_name: form.value.shipName,
  466. brand: form.value.brand,
  467. // 日期处理
  468. date: form.value.startDate && form.value.endDate ? [form.value.startDate.format('YYYYMMDD'), form.value.endDate.format('YYYYMMDD')] : undefined,
  469. // 范围值处理
  470. weight: form.value.weightMin || form.value.weightMax ? [form.value.weightMin || '', form.value.weightMax || ''] : undefined,
  471. quantity: form.value.quantityMin || form.value.quantityMax ? [form.value.quantityMin || '', form.value.quantityMax || ''] : undefined,
  472. amount: form.value.amountMin || form.value.amountMax ? [form.value.amountMin || '', form.value.amountMax || ''] : undefined,
  473. teu: form.value.teuMin || form.value.teuMax ? [form.value.teuMin || '', form.value.teuMax || ''] : undefined,
  474. // 选项处理
  475. supplier_ex_log: form.value.options?.includes('excludeLogistics'),
  476. supplier_ex_nvl: form.value.options?.includes('excludeNVL'),
  477. buyer_ex_log: form.value.buyerOptions?.includes('excludeLogistics'),
  478. buyer_ex_nvl: form.value.buyerOptions?.includes('excludeNVL'),
  479. };
  480. // 移除所有 undefined 和空字符串的属性
  481. Object.keys(params).forEach((key) => {
  482. if (params[key] === undefined || params[key] === '') {
  483. delete params[key];
  484. }
  485. });
  486. // 将参数保存到 queryParam 以供其他地方使用
  487. Object.assign(queryParam, params);
  488. if (activeTabKey.value === 'transaction') {
  489. pagination.value.current = 1; // Reset to first page
  490. await handleTableChange(pagination.value);
  491. } else if (activeTabKey.value === 'companies') {
  492. await handleTabChange('companies'); // Load companies data
  493. } else if (activeTabKey.value === 'tradeAnalysis') {
  494. await handleTabChange('tradeAnalysis'); // Load trade analysis data
  495. } else if (activeTabKey.value === 'shippingAnalysis') {
  496. await handleTabChange('shippingAnalysis'); // Load shipping analysis data
  497. }
  498. };
  499. const showAdvancedSearch = ref(false);
  500. const toggleAdvancedSearch = () => {
  501. showAdvancedSearch.value = !showAdvancedSearch.value;
  502. };
  503. // 修改 onMounted,不再自动加载数据
  504. onMounted(() => {
  505. Object.assign(queryParam, {
  506. source_type: 1,
  507. data_source: ['IMP_AMERICA_BL_SEA'],
  508. });
  509. });
  510. // Add these new variables
  511. const tableData = ref([]);
  512. const loading = ref(false);
  513. const pagination = ref({
  514. current: 1,
  515. pageSize: 10,
  516. total: 0,
  517. });
  518. // Replace registerTransactionTable with this new method
  519. const handleTableChange = async (pag, filters, sorter) => {
  520. loading.value = true;
  521. try {
  522. const params = {
  523. sort: sortMode.value,
  524. page: pag.current,
  525. page_size: pag.pageSize,
  526. ...queryParam,
  527. };
  528. const res = await list(params);
  529. if (res.result.data.result && res.result.data.result.docs) {
  530. // 处理日期格式
  531. tableData.value = res.result.data.result.docs.map((item) => ({
  532. ...item,
  533. date: item.date ? item.date.split(' ')[0] : '', // 只保留日期部分
  534. }));
  535. pagination.value = {
  536. current: pag.current,
  537. pageSize: pag.pageSize,
  538. total: res.result.data.result.numFound || 0,
  539. };
  540. }
  541. } catch (error) {
  542. console.error('Failed to fetch data:', error);
  543. } finally {
  544. loading.value = false;
  545. }
  546. };
  547. // 企业列表数据
  548. const companiesData = ref([]);
  549. // 处理企业列表数据
  550. const handleCompaniesData = (companies) => {
  551. companiesData.value = companies.map((company) => ({
  552. name: company.name,
  553. totalRecords: company.total_records,
  554. matchedRecords: company.matched_records,
  555. address: company.address,
  556. phone: company.phone,
  557. email: company.email,
  558. }));
  559. };
  560. // 处理对比点击
  561. const handleCompareClick = (company) => {
  562. console.log('Compare clicked for company:', company.name);
  563. };
  564. // Consolidate handleTabChange function
  565. const handleTabChange = async (key: string) => {
  566. activeTabKey.value = key;
  567. if (key === 'companies') {
  568. try {
  569. const res = await listCompanies(queryParam);
  570. if (res.result && res.result.companies) {
  571. handleCompaniesData(res.result.companies);
  572. }
  573. } catch (error) {
  574. console.error('Failed to fetch companies:', error);
  575. }
  576. } else if (key === 'tradeAnalysis') {
  577. // Load trade analysis data
  578. await Promise.all([
  579. loadMonthlyTrendData(),
  580. fetchHsCodeData(),
  581. fetchOriginCountryData(),
  582. fetchDestinationCountryData(),
  583. fetchBuyerData(),
  584. ]);
  585. } else if (key === 'shippingAnalysis') {
  586. // Load shipping analysis data
  587. await Promise.all([
  588. fetchOrigPortData(),
  589. fetchDestPortData(),
  590. fetchTransTypeData(),
  591. fetchIncotermsData()
  592. ]);
  593. }
  594. };
  595. // 月度趋势分析数据
  596. const monthlyTrendData = ref([]);
  597. // 加载月度趋势分析数据
  598. const loadMonthlyTrendData = async () => {
  599. try {
  600. const params = {
  601. ...queryParam,
  602. };
  603. const res = await getTrendReport(params);
  604. if (res.result.data.result && res.result.data.result) {
  605. // 处理日期格式
  606. monthlyTrendData.value = res.result.data.result.map((item) => ({
  607. ...item,
  608. }));
  609. }
  610. } catch (error) {
  611. console.error('Failed to fetch data:', error);
  612. } finally {
  613. loading.value = false;
  614. }
  615. };
  616. const hsCodeData = ref({});
  617. // 获取HS编码数据
  618. const fetchHsCodeData = async (sortMode = 'count') => {
  619. const params = {
  620. ...queryParam,
  621. sort: sortMode,
  622. };
  623. const res = await getHsCodeReport(params);
  624. hsCodeData.value = res.result.data.result;
  625. };
  626. const originCountryData = ref({});
  627. // 获取原产国数据
  628. const fetchOriginCountryData = async (sortMode = 'count') => {
  629. const params = {
  630. ...queryParam,
  631. sort: sortMode
  632. };
  633. const res = await getOrigCountryReport(params);
  634. originCountryData.value = res.result.data.result;
  635. };
  636. const destinationCountryData = ref({});
  637. // 获取目的国数据
  638. const fetchDestinationCountryData = async (sortMode = 'count') => {
  639. const params = {
  640. ...queryParam,
  641. sort: sortMode
  642. };
  643. const res = await getDestCountryReport(params);
  644. destinationCountryData.value = res.result.data.result;
  645. };
  646. const buyerData = ref({});
  647. // 获取采购商数据
  648. const fetchBuyerData = async () => {
  649. const params = {
  650. ...queryParam,
  651. };
  652. const res = await getBuyerReport(params);
  653. buyerData.value = res.result.data.result;
  654. };
  655. const origPortData = ref({});
  656. // 获取起运港数据
  657. const fetchOrigPortData = async () => {
  658. const params = {
  659. ...queryParam,
  660. };
  661. const res = await getOrigPortReport(params);
  662. origPortData.value = res.result.data.result;
  663. }
  664. const destPortData = ref({});
  665. // 获取起运港数据
  666. const fetchDestPortData = async () => {
  667. const params = {
  668. ...queryParam,
  669. };
  670. const res = await getDestPortReport(params);
  671. destPortData.value = res.result.data.result;
  672. }
  673. const transTypeData = ref({});
  674. // 获取运输方式数据
  675. const fetchTransTypeData = async () => {
  676. const params = {
  677. ...queryParam,
  678. };
  679. const res = await getTransTypeReport(params);
  680. transTypeData.value = res.result.data.result;
  681. }
  682. const incotermsData = ref({});
  683. // 获取成交方式数据
  684. const fetchIncotermsData = async () => {
  685. const params = {
  686. ...queryParam,
  687. };
  688. const res = await getIncotermsReport(params);
  689. incotermsData.value = res.result.data.result;
  690. }
  691. const isModalVisible = ref(false);
  692. // Add a new ref to store the selected supplierId
  693. const selectedComId = ref<string | null>(null);
  694. // Modify the handleSupplierClick function to set the selectedComId
  695. const handleSupplierClick = async (supplierId) => {
  696. loading.value = true;
  697. selectedComId.value = supplierId; // Set the selected comId
  698. const params = {
  699. com_id: supplierId,
  700. com_role: 2,
  701. ...queryParam,
  702. };
  703. const response = await getCompanyInfo(params);
  704. supplier.value = {
  705. ...response.result.data.result,
  706. postal_code: response.result.data.result.postal_code ? response.result.data.result.postal_code.join(',') : ''
  707. };
  708. loading.value = false;
  709. isModalVisible.value = true;
  710. activeTabKey1.value = 'tradeOverview';
  711. await loadFreightHistoryData();
  712. await fetchCompanyHsCodeData();
  713. await fetchTradePartnersData();
  714. await fetchRegionDistributionData();
  715. };
  716. const handleFreightIntroClick = async () => {
  717. }
  718. const handleBuyerClick = async (buyerId) => {
  719. loading.value = true;
  720. selectedComId.value = buyerId;
  721. const params = {
  722. com_id: buyerId,
  723. com_role: 1,
  724. ...queryParam,
  725. };
  726. const response = await getCompanyInfo(params);
  727. supplier.value = {
  728. ...response.result.data.result,
  729. postal_code: response.result.data.result.postal_code ? response.result.data.result.postal_code.join(',') : ''
  730. };
  731. loading.value = false;
  732. isModalVisible.value = true;
  733. activeTabKey1.value = 'tradeOverview';
  734. await loadFreightHistoryData();
  735. await fetchCompanyHsCodeData();
  736. await fetchTradePartnersData();
  737. await fetchRegionDistributionData();
  738. }
  739. const freightHistoryData = ref([]);
  740. // Fetch freight history data for both roles and combine
  741. const loadFreightHistoryData = async () => {
  742. try {
  743. const params = {
  744. com_id: selectedComId.value,
  745. ...queryParam,
  746. };
  747. // Fetch export data (com_role: 2)
  748. const exportParams = { ...params, com_role: 2 };
  749. const exportRes = await getMonthly(exportParams);
  750. const exportData = exportRes?.result?.data?.result?.as_supplier?.monthly?.buckets?.map((item) => ({
  751. val: item.val,
  752. num_supplier: item.count,
  753. num_buyer: 0, // Initialize import count
  754. })) || [];
  755. // Fetch import data (com_role: 1)
  756. const importParams = { ...params, com_role: 1 };
  757. const importRes = await getMonthly(importParams);
  758. const importData = importRes?.result?.data?.result?.as_buyer?.monthly?.buckets?.map((item) => ({
  759. val: item.val,
  760. num_supplier: 0,
  761. num_buyer: item.count, // Initialize export count
  762. })) || [];
  763. // Combine data by date
  764. const combinedData = [...exportData, ...importData].reduce((acc, item) => {
  765. const existing = acc.find((i) => i.val === item.val);
  766. if (existing) {
  767. existing.num_supplier += item.num_supplier;
  768. existing.num_buyer += item.num_buyer;
  769. } else {
  770. acc.push(item);
  771. }
  772. return acc;
  773. }, []);
  774. freightHistoryData.value = combinedData;
  775. } catch (error) {
  776. console.error('Failed to fetch freight history data:', error);
  777. }
  778. };
  779. // 企业HS编码数据
  780. const companyHsCodeData = ref({});
  781. // 获取企业HS编码数据
  782. const fetchCompanyHsCodeData = async () => {
  783. const params = {
  784. com_id: selectedComId.value,
  785. com_role: 2,
  786. ...queryParam,
  787. };
  788. const res = await getCompanyHsCode(params);
  789. companyHsCodeData.value.as_supplier = res.data.result.as_supplier.hs_code;
  790. const params1 = {
  791. com_id: selectedComId.value,
  792. com_role: 1,
  793. ...queryParam,
  794. };
  795. const res1 = await getCompanyHsCode(params1);
  796. companyHsCodeData.value.as_buyer = res1.data.result.as_buyer?.hs_code;
  797. };
  798. // 贸易伙伴数据
  799. const tradePartnersData = ref({});
  800. // 获取贸易伙伴数据
  801. const fetchTradePartnersData = async () => {
  802. const params = {
  803. com_id: selectedComId.value,
  804. com_role: 2,
  805. ...queryParam,
  806. };
  807. const res = await getCompanyPartner(params);
  808. tradePartnersData.value.as_supplier = res.data.result.as_supplier.buyer;
  809. const params1 = {
  810. com_id: selectedComId.value,
  811. com_role: 1,
  812. ...queryParam,
  813. };
  814. const res1 = await getCompanyPartner(params1);
  815. tradePartnersData.value.as_buyer = res1.data.result.as_buyer.buyer;
  816. };
  817. const regionDistributionData = ref({});
  818. // 获取企业区域分布数据
  819. const fetchRegionDistributionData = async () => {
  820. const params = {
  821. com_id: selectedComId.value,
  822. com_role: 2,
  823. ...queryParam,
  824. };
  825. const res = await getRegionDistribution(params);
  826. regionDistributionData.value.as_supplier = res.data.result.as_supplier.dest_country;
  827. const params1 = {
  828. com_id: selectedComId.value,
  829. com_role: 1,
  830. ...queryParam,
  831. };
  832. const res1 = await getRegionDistribution(params1);
  833. regionDistributionData.value.as_buyer = res1.data.result.as_buyer.dest_country;
  834. };
  835. // Call the function to load data
  836. onMounted(() => {
  837. });
  838. // Current active tab for the modal
  839. const activeTabKey1 = ref('tradeOverview'); // Set default tab key
  840. // Handle tab change for the modal
  841. const handleTabChange1 = async (key: string) => {
  842. activeTabKey1.value = key;
  843. if (key === 'tradeOverview') {
  844. await loadFreightHistoryData(); // Load freight history data
  845. } else if (key === 'importDetails') {
  846. await loadImportDetailsData();
  847. } else if (key === 'exportDetails') {
  848. await loadExportDetailsData(); // Load export details data
  849. } else if (key === 'exportTradingPartners') {
  850. // Load export trading partners data
  851. console.log('Load export trading partners data');
  852. } else if (key === 'enterpriseAtlas') {
  853. // Load enterprise atlas data
  854. console.log('Load enterprise atlas data');
  855. }
  856. };
  857. const exportDetailsData = ref([]); // 新增用于存储出口详单数据
  858. const exportDetailsColumns = [ // 新增表格列定义
  859. {
  860. title: '供应商',
  861. dataIndex: 'supplier_t',
  862. key: 'supplier_t',
  863. },
  864. {
  865. title: '采购商',
  866. dataIndex: 'buyer_t',
  867. key: 'buyer_t',
  868. },
  869. {
  870. title: '货运介绍',
  871. dataIndex: 'freightDescription',
  872. key: 'freightDescription',
  873. },
  874. {
  875. title: '日期',
  876. dataIndex: 'date',
  877. key: 'date',
  878. sorter: (a, b) => new Date(a.date) - new Date(b.date),
  879. },
  880. ];
  881. const paginationExportDetails = ref({
  882. current: 1,
  883. pageSize: 10,
  884. total: 0,
  885. });
  886. const handleTableChangeExportDetails = async (pag, filters, sorter) => {
  887. paginationExportDetails.value.current = pag.current; // 更新当前页
  888. paginationExportDetails.value.pageSize = pag.pageSize; // 更新每页条数
  889. // 其他处理逻辑...
  890. };
  891. const loadExportDetailsData = async () => { // 新增加载出口详单数据的方法
  892. loading.value = true;
  893. try {
  894. const params = {
  895. com_id: selectedComId.value,
  896. com_role: 2,
  897. ...queryParam,
  898. };
  899. const res = await getCompanyRecord(params);
  900. exportDetailsData.value = res.data.result.as_supplier.response.docs.map((item) => ({
  901. ...item,
  902. date: item.date ? item.date.split(' ')[0] : '', // 只保留日期部分
  903. })); // 存储回的数据
  904. } catch (error) {
  905. console.error('Failed to fetch export details:', error);
  906. } finally {
  907. loading.value = false;
  908. }
  909. };
  910. const importDetailsData = ref([]); // 新增用于存储进口详单数据
  911. const importDetailsColumns = [ // 新增表格列定义
  912. {
  913. title: '供应商',
  914. dataIndex: 'supplier_t',
  915. key: 'supplier_t',
  916. },
  917. {
  918. title: '采购商',
  919. dataIndex: 'buyer_t',
  920. key: 'buyer_t',
  921. },
  922. {
  923. title: '货运介绍',
  924. dataIndex: 'freightDescription',
  925. key: 'freightDescription',
  926. },
  927. {
  928. title: '日期',
  929. dataIndex: 'date',
  930. key: 'date',
  931. sorter: (a, b) => new Date(a.date) - new Date(b.date),
  932. },
  933. ];
  934. const paginationImportDetails = ref({
  935. current: 1,
  936. pageSize: 10,
  937. total: 0,
  938. });
  939. const handleTableChangeImportDetails = async (pag, filters, sorter) => {
  940. paginationImportDetails.value.current = pag.current; // 更新当前页
  941. paginationImportDetails.value.pageSize = pag.pageSize; // 更新每页条数
  942. // 其他处理逻辑...
  943. };
  944. const loadImportDetailsData = async () => { // 新增加载进口详单数据的方法
  945. loading.value = true;
  946. try {
  947. const params = {
  948. com_id: selectedComId.value,
  949. com_role: 1,
  950. ...queryParam,
  951. };
  952. const res = await getCompanyRecord(params);
  953. importDetailsData.value = res.data.result.as_buyer.response.docs.map((item) => ({
  954. ...item,
  955. date: item.date ? item.date.split(' ')[0] : '', // 只保留日期部分
  956. })); // 存储回的数据
  957. } catch (error) {
  958. console.error('Failed to fetch import details:', error);
  959. } finally {
  960. loading.value = false;
  961. }
  962. };
  963. // 修改关闭弹窗的逻辑
  964. const closeModal = () => {
  965. activeTabKey1.value = 'tradeOverview'
  966. isModalVisible.value = false; // 关闭弹窗
  967. resetModalData(); // 重置数据
  968. };
  969. // 添加重置数据的函数
  970. const resetModalData = () => {
  971. supplier.value = null; // 重置供应商信息
  972. freightHistoryData.value = []; // 重置货运历史数据
  973. companyHsCodeData.value = {}; // 重置企业HS编码数据
  974. tradePartnersData.value = {}; // 重置贸易伙伴数据
  975. regionDistributionData.value = {}; // 重置区域分布数据
  976. };
  977. const handleSortMode = (value) => {
  978. fetchHsCodeData(value);
  979. };
  980. const handleSortModeOriginCountryData = (value) => {
  981. fetchOriginCountryData(value);
  982. }
  983. const handleSortModeDestinationCountryData = (value) => {
  984. fetchDestinationCountryData(value);
  985. }
  986. const sortMode = ref('date'); // Set default sort mode to 'date'
  987. // Modify handleSortChange to include sort parameter in the API request
  988. const handleSortChange = async (event) => {
  989. sortMode.value = event.target.value; // Update sort mode
  990. await handleTableChange(pagination.value)
  991. };
  992. </script>
  993. <style scoped lang="less">
  994. .search-form {
  995. padding: 20px;
  996. background-color: #f5f5f5;
  997. border-radius: 8px;
  998. margin-bottom: 16px;
  999. .query-container {
  1000. background-color: #fff;
  1001. padding: 16px;
  1002. border-radius: 4px;
  1003. box-shadow: 0 1px 4px rgba(0, 0, 0, 0.05);
  1004. .ant-row {
  1005. padding: 12px 0;
  1006. &:not(:last-child) {
  1007. border-bottom: 1px solid #e8e8e8;
  1008. }
  1009. }
  1010. .button-row {
  1011. padding: 16px 0 0;
  1012. border-bottom: none !important;
  1013. background-color: transparent !important;
  1014. }
  1015. .checkbox-group {
  1016. display: flex;
  1017. align-items: center;
  1018. height: 100%;
  1019. }
  1020. }
  1021. }
  1022. // 添加新的样式
  1023. .table-container {
  1024. background-color: #fff;
  1025. border-radius: 8px;
  1026. padding: 16px 24px;
  1027. margin: 0 20px;
  1028. box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
  1029. transition: opacity 0.3s ease-in-out;
  1030. .custom-tabs {
  1031. :deep(.ant-tabs-nav) {
  1032. margin-bottom: 24px;
  1033. &::before {
  1034. border-bottom: 1px solid #f0f0f0;
  1035. }
  1036. .ant-tabs-tab {
  1037. padding: 12px 24px;
  1038. font-size: 14px;
  1039. transition: all 0.3s;
  1040. &:hover {
  1041. color: @primary-color;
  1042. }
  1043. &.ant-tabs-tab-active {
  1044. .ant-tabs-tab-btn {
  1045. font-weight: 500;
  1046. }
  1047. }
  1048. }
  1049. .ant-tabs-ink-bar {
  1050. height: 3px;
  1051. border-radius: 3px 3px 0 0;
  1052. }
  1053. }
  1054. :deep(.ant-tabs-content) {
  1055. padding: 8px 0;
  1056. }
  1057. }
  1058. }
  1059. // 分析报告内容样式
  1060. .analysis-content {
  1061. background: #fff;
  1062. .analysis-item {
  1063. margin-bottom: 24px; // 添加组件之间的间距
  1064. &:last-child {
  1065. margin-bottom: 0; // 最后一个组件不需要底部间距
  1066. }
  1067. :deep(.ant-card) {
  1068. transition: all 0.3s ease;
  1069. box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
  1070. &:hover {
  1071. box-shadow: 0 4px 16px rgba(0, 0, 0, 0.12);
  1072. }
  1073. }
  1074. }
  1075. }
  1076. // 表格工具栏样式
  1077. :deep(.ant-table-wrapper) {
  1078. .ant-table-title {
  1079. padding: 16px 0;
  1080. }
  1081. }
  1082. // 按钮样式优化
  1083. :deep(.ant-btn) {
  1084. height: 32px;
  1085. padding: 0 16px;
  1086. border-radius: 4px;
  1087. &.ant-btn-primary {
  1088. box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
  1089. }
  1090. }
  1091. // 表格内容样式优化
  1092. :deep(.ant-table) {
  1093. .ant-table-thead>tr>th {
  1094. background: #fafafa;
  1095. font-weight: 500;
  1096. }
  1097. .ant-table-tbody>tr>td {
  1098. transition: background 0.3s;
  1099. }
  1100. .ant-table-tbody>tr:hover>td {
  1101. background: #f5f5f5;
  1102. }
  1103. }
  1104. // Add these new styles
  1105. .table-toolbar {
  1106. margin-bottom: 16px;
  1107. display: flex;
  1108. justify-content: flex-end;
  1109. }
  1110. </style>