index.vue 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  1. <template>
  2. <div class="statistic">
  3. <div class="statistic-tr">
  4. <div class="statistic-block">
  5. <div class="title">离岛旅客总量统计
  6. <a-date-picker v-model:value="state.date1"/>
  7. </div>
  8. <div class="content content-1">
  9. <div class="left">
  10. <div class="total">
  11. <div class="card">{{ total1[0] }}</div>
  12. <div class="card">{{ total1[1] }}</div>
  13. <div class="dou"></div>
  14. <div class="card">{{ total1[2] }}</div>
  15. <div class="card">{{ total1[3] }}</div>
  16. <div class="card">{{ total1[4] }}</div>
  17. <div class="dou"></div>
  18. <div class="card">{{ total1[5] }}</div>
  19. <div class="card">{{ total1[6] }}</div>
  20. <div class="card">{{ total1[7] }}</div>
  21. <div class="unit">/离岛旅客量</div>
  22. </div>
  23. <pieChart :data="state.arr1" :color="state.color1"/>
  24. </div>
  25. <div class="line"/>
  26. <div class="right">
  27. <template v-for="(item, index) in state.arr1">
  28. <div class="right-item">
  29. <div class="value" :style="{color: state.color1[index]}">{{ item.value }}</div>
  30. <div class="name">{{ item.name }}</div>
  31. </div>
  32. </template>
  33. </div>
  34. </div>
  35. </div>
  36. </div>
  37. <div class="statistic-tr">
  38. <div class="statistic-block">
  39. <div class="title">旅客日流量分析
  40. <a-date-picker v-model:value="state.date2"/>
  41. </div>
  42. <div class="content">
  43. <lineChart :data="state.arr2" :color="state.color2" unit="时"/>
  44. </div>
  45. </div>
  46. <div class="statistic-block">
  47. <div class="title">旅客月流量分析
  48. <a-date-picker v-model:value="state.date3" picker="month"/>
  49. </div>
  50. <div class="content">
  51. <lineChart :data="state.arr3" :color="state.color3" unit="日"/>
  52. </div>
  53. </div>
  54. </div>
  55. <div class="statistic-tr">
  56. <div class="statistic-block">
  57. <div class="title">查验通过率分析
  58. <a-date-picker v-model:value="state.date4"/>
  59. </div>
  60. <div class="content" style="display: flex">
  61. <pie2Chart :data="state.arr4" :color="state.color4" :index="0"/>
  62. <pie2Chart :data="state.arr4" :color="state.color6" :index="1"/>
  63. </div>
  64. </div>
  65. <div class="statistic-block">
  66. <div class="title">平均查验时间分析
  67. <a-date-picker v-model:value="state.date5"/>
  68. </div>
  69. <div class="content">
  70. <barChart :data="state.arr5" :color="state.color5"/>
  71. </div>
  72. </div>
  73. </div>
  74. <div class="statistic-tr">
  75. <div class="statistic-block">
  76. <div class="title">拦截成功率分析
  77. <a-date-picker v-model:value="state.date6"/>
  78. </div>
  79. <div class="content" style="display: flex">
  80. <pie2Chart :data="state.arr6" :color="state.color4" :index="0"/>
  81. <pie2Chart :data="state.arr6" :color="state.color6" :index="1"/>
  82. </div>
  83. </div>
  84. <div class="statistic-block">
  85. <div class="title">风险趋势分析
  86. <a-date-picker v-model:value="state.date7"/>
  87. </div>
  88. <div class="content">
  89. <pieChart :data="state.arr7" :color="state.color7" :type="2"/>
  90. </div>
  91. </div>
  92. </div>
  93. </div>
  94. </template>
  95. <script setup name="demo2">
  96. import basicApi from '@/api/gsc/basic'
  97. import pieChart from './chart/pie.vue'
  98. import lineChart from './chart/line.vue'
  99. import pie2Chart from './chart/pie2.vue'
  100. import barChart from './chart/bar.vue'
  101. import dayjs from "dayjs";
  102. const {proxy} = getCurrentInstance()
  103. const state = reactive({
  104. date1: dayjs(new Date()),
  105. date2: dayjs(new Date()),
  106. date3: dayjs(new Date()),
  107. date4: dayjs(new Date()),
  108. date5: dayjs(new Date()),
  109. date6: dayjs(new Date()),
  110. date7: dayjs(new Date()),
  111. arr1: [],
  112. arr2: [],
  113. arr3: [],
  114. arr4: [],
  115. arr5: [],
  116. arr6: [],
  117. arr7: [],
  118. color1: ['#16C843', '#00EAFF', '#FB9A04', '#00A2FF', '#5AD8A6', '#F4FF78'],
  119. color2: ['rgba(1, 255, 246, 1)', 'rgba(1, 255, 246, 0)'],
  120. color3: ['rgba(14, 156, 255, 1)', 'rgba(14, 156, 255, 0)'],
  121. color4: ['rgba(0, 194, 255, 1)', 'rgba(255, 0, 61, 1)'],
  122. color5: ['#00A8FF', '#0F3352'],
  123. color6: ['#00A3FF', '#BDFF00'],
  124. color7: ['#16C843', '#00EAFF', '#FB9A04', '#00A2FF', '#9966FF', '#FC6767', '#ABB2B4'],
  125. })
  126. const initData = () => {
  127. basicApi.passengerstatisticsList({type: 1}).then(res => {
  128. state.arr1 = res
  129. })
  130. basicApi.passengerstatisticsList({type: 2}).then(res => {
  131. state.arr2 = res
  132. })
  133. basicApi.passengerstatisticsList({type: 3}).then(res => {
  134. state.arr3 = res
  135. })
  136. basicApi.passengerstatisticsList({type: 4}).then(res => {
  137. state.arr4 = res
  138. })
  139. basicApi.passengerstatisticsList({type: 5}).then(res => {
  140. state.arr5 = res
  141. })
  142. basicApi.passengerstatisticsList({type: 6}).then(res => {
  143. state.arr6 = res
  144. })
  145. basicApi.passengerstatisticsList({type: 7}).then(res => {
  146. state.arr7 = res
  147. })
  148. }
  149. const total1 = computed(() => {
  150. let total = 0
  151. state.arr1.forEach(v => {
  152. total += Number(v.value)
  153. })
  154. const str = String(total).split('')
  155. for (let i = 0; i < (9 - str.length); i++) {
  156. str.unshift('0')
  157. }
  158. return str
  159. })
  160. onMounted(() => {
  161. initData()
  162. })
  163. </script>
  164. <style lang="less" scoped>
  165. .statistic {
  166. width: 100%;
  167. height: 100%;
  168. overflow-y: auto;
  169. display: flex;
  170. flex-direction: column;
  171. gap: 10px;
  172. .statistic-tr {
  173. background: #14204A;
  174. border-radius: 8px;
  175. padding: 24px;
  176. display: flex;
  177. gap: 24px;
  178. .statistic-block {
  179. flex: 1;
  180. box-shadow: inset 0px 0px 100px 0px rgba(0, 228, 255, 0.12);
  181. border-radius: 0px 0px 0px 0px;
  182. border: 2px solid rgba(0, 174, 255, 0.33);
  183. .title {
  184. height: 46px;
  185. line-height: 46px;
  186. padding: 0 16px;
  187. background: linear-gradient(90deg, rgba(31, 180, 255, 0.12) 0%, rgba(31, 180, 255, 0) 100%);
  188. color: #ffffff;
  189. font-size: 20px;
  190. font-weight: bold;
  191. display: flex;
  192. align-items: center;
  193. :deep(.ant-picker) {
  194. margin-left: auto;
  195. height: 30px;
  196. background: linear-gradient(180deg, rgba(31, 180, 255, 0.32) 0%, rgba(31, 180, 255, 0) 100%);
  197. border-radius: 0;
  198. border-top: 1px solid #1FC6FF;
  199. .ant-picker-input {
  200. > input {
  201. color: #1DC8FF;
  202. &::placeholder {
  203. color: #1DC8FF;
  204. }
  205. }
  206. .ant-picker-suffix {
  207. > span {
  208. color: #1DC8FF;
  209. }
  210. }
  211. }
  212. }
  213. }
  214. }
  215. }
  216. }
  217. .content {
  218. height: 300px;
  219. width: 100%;
  220. &.content-1 {
  221. height: 290px;
  222. padding-top: 24px;
  223. display: flex;
  224. gap: 40px;
  225. .left {
  226. width: 595px;
  227. height: 100%;
  228. display: flex;
  229. flex-direction: column;
  230. .total {
  231. height: 79px;
  232. width: 100%;
  233. background: rgba(0, 174, 255, 0.05);
  234. border-radius: 0px 0px 0px 0px;
  235. border: 2px solid rgba(0, 174, 255, 0.33);
  236. display: flex;
  237. align-items: center;
  238. padding-left: 10px;
  239. .card {
  240. background-image: url("./card.png");
  241. background-size: 42px 56px;
  242. width: 43px;
  243. height: 56px;
  244. display: flex;
  245. align-items: center;
  246. justify-content: center;
  247. font-family: Bebas;
  248. font-weight: 400;
  249. font-size: 38px;
  250. color: #FFFFFF;
  251. }
  252. .dou {
  253. background-image: url("./dou.png");
  254. width: 12px;
  255. height: 16px;
  256. margin-top: 40px;
  257. }
  258. .unit {
  259. font-weight: 500;
  260. font-size: 18px;
  261. color: #68C3FF;
  262. }
  263. > * {
  264. margin-left: 8px;
  265. }
  266. }
  267. }
  268. .line {
  269. width: 1px;
  270. height: calc(100% - 16px);
  271. background-color: rgba(18, 104, 155, 0.4);
  272. }
  273. .right {
  274. flex: 1;
  275. height: 100%;
  276. display: grid;
  277. grid-template-columns: repeat(3, 1fr);
  278. grid-template-rows: repeat(auto-fill, 113px);
  279. gap: 24px 16px;
  280. padding-right: 16px;
  281. .right-item {
  282. box-shadow: inset 0px 0px 5px 2px #0CBDAF;
  283. display: flex;
  284. flex-direction: column;
  285. padding-left: 13px;
  286. background-color: rgba(31, 180, 255, 0.12);
  287. font-family: "PingFang SC";
  288. position: relative;
  289. &:after {
  290. z-index: 1;
  291. content: '';
  292. position: absolute;
  293. width: 94px;
  294. height: 75px;
  295. background-image: url('./feiji.png');
  296. top: 18px;
  297. right: 22px;
  298. }
  299. .value {
  300. z-index: 2;
  301. font-weight: 500;
  302. font-size: 40px;
  303. line-height: 60px;
  304. margin-top: 13px;
  305. }
  306. .name {
  307. z-index: 2;
  308. font-weight: 400;
  309. font-size: 16px;
  310. color: rgba(255, 255, 255, 0.6);
  311. }
  312. &:nth-child(n+4) {
  313. &:after {
  314. width: 84px;
  315. height: 84px;
  316. background-image: url('./port.png');
  317. }
  318. }
  319. }
  320. }
  321. }
  322. }
  323. </style>