chartDetail.vue 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. <template>
  2. <div
  3. :class="[
  4. 'chart-position-analysis-page',
  5. isPcShow ? 'chart-position-analysis-page-pc' : '',
  6. ]"
  7. v-loading="pageLoading"
  8. element-loading-text="加载中"
  9. >
  10. <div class="chart-wrap-box">
  11. <div class="wrap-top">
  12. <div>
  13. <el-radio-group
  14. v-model="tabKey"
  15. @change="chartItemInfo = chartListState[tabKey]"
  16. v-show="chartListState.BuyList.List"
  17. >
  18. <el-radio-button
  19. :label="tab.key"
  20. v-for="tab in tabKeys"
  21. :key="tab.key"
  22. >{{ tab.label }}</el-radio-button
  23. >
  24. </el-radio-group>
  25. </div>
  26. <div>
  27. <el-date-picker
  28. v-model="selectDate"
  29. type="date"
  30. :placeholder="$t('ToolBox.PositionAnalysis.select_date_placeholder')"
  31. value-format="yyyy-MM-dd"
  32. :picker-options="pickerOption"
  33. @change="$emit('handleOpt','date')"
  34. style="margin-right: 10px"
  35. />
  36. <el-button type="primary" plain @click="$emit('handleOpt','beforeDate')">{{$t('ToolBox.PositionAnalysis.see_before_day')}}</el-button>
  37. <el-button type="primary" plain @click="$emit('handleOpt','nextDate')" :disabled="disabledNextBtn">{{$t('ToolBox.PositionAnalysis.see_next_day')}}</el-button>
  38. </div>
  39. </div>
  40. <div
  41. class="chart-wrap"
  42. v-if="chartItemInfo&&chartItemInfo.List && chartListState.BuyList.List"
  43. >
  44. <div class="top-info-box">
  45. <span>{{$route.query.classify_type}}</span>
  46. <span>{{ chartItemInfo.name }}</span>
  47. <span
  48. ><span style="color: #999; margin-right: 2px">{{$t('ToolBox.PositionAnalysis.count_name')}} </span
  49. >{{ chartItemInfo.TotalDealValue }}</span
  50. >
  51. <span
  52. ><span style="color: #999; margin-right: 2px">{{$t('ToolBox.PositionAnalysis.compare_yesterday')}} </span
  53. >{{ chartItemInfo.TotalDealChange }}</span
  54. >
  55. </div>
  56. <chart-box :keyVal="tabKey" :data="chartItemInfo" />
  57. </div>
  58. <div class="empty-wrap" v-else>
  59. <tableNoData :text="$t('ToolBox.PositionAnalysis.no_data')" />
  60. </div>
  61. </div>
  62. </div>
  63. </template>
  64. <script>
  65. import {
  66. apiPositionAnalysisInfo,
  67. apiPositionAnalysisList,
  68. } from "@/api/modules/positionAnalysis";
  69. import chartBox from "./chartBox.vue";
  70. export default {
  71. components: { chartBox },
  72. watch: {
  73. "$route.query": {
  74. handler(nval) {
  75. this.selectDate = "";
  76. this.getInfo();
  77. },
  78. },
  79. },
  80. props:{
  81. disabledNextBtn:Boolean,
  82. selectDate:String,
  83. pickerOption:Object
  84. },
  85. computed:{
  86. tabKeys(){
  87. return [
  88. { label: this.$t('ToolBox.PositionAnalysis.long_position')/* "多单" */, key: "BuyList" },
  89. { label: this.$t('ToolBox.PositionAnalysis.short_position')/* "空单" */, key: "SoldList" },
  90. { label: this.$t('ToolBox.PositionAnalysis.net_long_position')/* "净多单" */, key: "CleanBuyList" },
  91. { label: this.$t('ToolBox.PositionAnalysis.net_short_position')/* "净空单" */, key: "CleanSoldList" },
  92. ]
  93. },
  94. chartListState(){
  95. return {
  96. BuyList: {
  97. name: this.$t('ToolBox.PositionAnalysis.long_position')/* "多单" */,
  98. labelName: this.$t('ToolBox.PositionAnalysis.hold_long_position')/* "持多单量" */,
  99. },
  100. SoldList: {
  101. name: this.$t('ToolBox.PositionAnalysis.short_position')/* "空单" */,
  102. labelName: this.$t('ToolBox.PositionAnalysis.hold_short_position')/* "持空单量" */,
  103. },
  104. CleanBuyList: {
  105. name:this.$t('ToolBox.PositionAnalysis.net_long_position') /* "净多单" */,
  106. labelName: this.$t('ToolBox.PositionAnalysis.net_long_position_num')/* "净多单量" */,
  107. },
  108. CleanSoldList: {
  109. name: this.$t('ToolBox.PositionAnalysis.net_short_position')/* "净空单" */,
  110. labelName: this.$t('ToolBox.PositionAnalysis.net_short_position_num')/* "净空单量" */,
  111. },
  112. }
  113. }
  114. },
  115. data() {
  116. return {
  117. isPcShow: true,
  118. isRefresh: false,
  119. minDate: new Date("2000/01/01"),
  120. latestDate: "", //
  121. selectDate: "",
  122. allClassifyTypeList: [],
  123. pageLoading: false,
  124. tabKey: "BuyList",
  125. chartItemInfo: null,
  126. };
  127. },
  128. methods: {
  129. getAllClassifyType() {
  130. apiPositionAnalysisList().then((res) => {
  131. if (res.Ret !== 200) return;
  132. const arr = res.Data || [];
  133. // 将数据展开
  134. arr.forEach((item) => {
  135. item.Items &&
  136. item.Items.forEach((itemC1) => {
  137. itemC1.Items &&
  138. itemC1.Items.forEach((itemC2) => {
  139. this.allClassifyTypeList.push({
  140. exchange: item.Exchange,
  141. classify_name: itemC1.ClassifyName,
  142. classify_type: itemC2.ClassifyType,
  143. });
  144. });
  145. });
  146. });
  147. });
  148. },
  149. handleClassifyTypeChange(type) {
  150. if (!this.allClassifyTypeList.length) return;
  151. const currentExchange = this.$route.query.exchange;
  152. const currentClassifyName = this.$route.query.classify_name;
  153. const currentClassifyType = this.$route.query.classify_type;
  154. // 找index
  155. let indexNum = 0;
  156. this.allClassifyTypeList.forEach((item, index) => {
  157. if (
  158. item.exchange === currentExchange &&
  159. item.classify_name === currentClassifyName &&
  160. item.classify_type === currentClassifyType
  161. )
  162. indexNum = index;
  163. });
  164. let obj = {};
  165. if (type === "before") {
  166. obj =
  167. this.allClassifyTypeList[
  168. indexNum === 0 ? this.allClassifyTypeList.length - 1 : indexNum - 1
  169. ];
  170. } else {
  171. obj =
  172. this.allClassifyTypeList[
  173. indexNum === this.allClassifyTypeList.length - 1 ? 0 : indexNum + 1
  174. ];
  175. }
  176. this.$router.replace({
  177. query: {
  178. ...this.$route.query,
  179. exchange: obj.exchange,
  180. classify_name: obj.classify_name,
  181. classify_type: obj.classify_type,
  182. },
  183. });
  184. },
  185. //切换 前一天\后一天 如果遇到周六日则跳过
  186. handleDateChange(type) {
  187. let num = 1;
  188. if (type === "before") {
  189. if (this.$moment(this.selectDate).isoWeekday() === 1) {
  190. //向前一天时 当前为周一则 跳到 上周五
  191. num = 3;
  192. }
  193. this.selectDate = this.$moment(this.selectDate)
  194. .add(-num, "days")
  195. .format("YYYY-MM-DD");
  196. } else {
  197. if (this.$moment(this.selectDate).isoWeekday() === 5) {
  198. //向前一天时 当前为周五则 跳到 下周一
  199. num = 3;
  200. }
  201. this.selectDate = this.$moment(this.selectDate)
  202. .add(num, "days")
  203. .format("YYYY-MM-DD");
  204. }
  205. this.getInfo();
  206. },
  207. async getInfo() {
  208. if(!this.$route.query.classify_name||!this.$route.query.classify_type) return
  209. this.pageLoading = true;
  210. const res = await apiPositionAnalysisInfo({
  211. DataTime: this.selectDate || "",
  212. ClassifyName: this.$route.query.classify_name,
  213. ClassifyType: this.$route.query.classify_type,
  214. Exchange: this.$route.query.exchange,
  215. });
  216. this.pageLoading = false;
  217. if (res.Ret === 200) {
  218. const obj = res.Data || {};
  219. for (let key in this.chartListState) {
  220. this.chartListState[key] = {
  221. ...this.chartListState[key],
  222. ...obj[key],
  223. };
  224. }
  225. this.chartItemInfo = this.chartListState[this.tabKey];
  226. this.latestDate = res.Data.LastDataTime;
  227. if (res.Data.DataTime) {
  228. this.selectDate = this.$moment(res.Data.DataTime).format(
  229. "YYYY-MM-DD"
  230. );
  231. this.$emit('setInfo',{date: this.selectDate})
  232. }
  233. this.$store.commit(
  234. "SET_TITLE",
  235. `${this.$route.query.classify_type} ${this.$moment(
  236. this.selectDate
  237. ).format("YYYYMMDD")}持仓`
  238. );
  239. } else {
  240. // 清空数据
  241. for (let key in this.chartListState) {
  242. this.chartListState[key].List = null;
  243. }
  244. }
  245. },
  246. getHandles(data) {
  247. // 监听选择的日期改变
  248. if (data.opt == "date") {
  249. this.selectDate = data.val;
  250. this.getInfo();
  251. } else if (data.opt === "beforeDate") {
  252. this.handleDateChange("before");
  253. } else if (data.opt === "nextDate") {
  254. this.handleDateChange("next");
  255. } else if (data.opt === "beforeClassifyType") {
  256. this.handleClassifyTypeChange("before");
  257. } else if (data.opt === "nextClassifyType") {
  258. this.handleClassifyTypeChange("next");
  259. }
  260. },
  261. },
  262. mounted() {
  263. this.getAllClassifyType();
  264. this.getInfo();
  265. },
  266. };
  267. </script>
  268. <style lang="scss" scoped>
  269. .chart-position-analysis-page {
  270. padding-bottom: calc(160px + constant(safe-area-inset-bottom));
  271. padding-bottom: calc(160px + env(safe-area-inset-bottom));
  272. overflow-x: auto;
  273. }
  274. .chart-wrap-box{
  275. min-width: 743px;
  276. height: 100%;
  277. overflow-y: hidden;
  278. }
  279. .chart-wrap {
  280. width: 100%;
  281. margin-bottom: 60px;
  282. .top-info-box {
  283. padding: 0 34px 30px 34px;
  284. text-align: center;
  285. span {
  286. display: inline-block;
  287. margin-right: 50px;
  288. }
  289. }
  290. }
  291. .wrap-top {
  292. display: flex;
  293. justify-content: space-between;
  294. margin-bottom: 20px;
  295. }
  296. .empty-wrap {
  297. text-align: center;
  298. padding: 150px 0;
  299. box-sizing: border-box;
  300. }
  301. // @media (min-width: 600px){
  302. .chart-position-analysis-page-pc {
  303. padding: 0;
  304. height: 100%;
  305. .top-sticky-wrap {
  306. display: none;
  307. }
  308. .bot-fixed-wrap {
  309. display: none;
  310. }
  311. .empty-wrap {
  312. font-size: 16px;
  313. }
  314. .chart-wrap {
  315. margin-bottom: 60px;
  316. .top-info-box {
  317. padding: 0 20px 20px;
  318. text-align: center;
  319. font-size: 20px;
  320. span {
  321. display: inline-block;
  322. margin-right: 30px;
  323. }
  324. }
  325. }
  326. }
  327. // }
  328. </style>