yey.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. <template>
  2. <div>
  3. <el-card>
  4. <el-form :model="skillParams" ref="skillParams" class="clearfix topForm" style="float: left">
  5. <div class="leftForm-skill">
  6. <el-form-item
  7. style="display:flex;"
  8. prop="channel"
  9. name="channel"
  10. :rules="[{ required: true, message: '请选择渠道', trigger: 'blur' }]"
  11. label="渠道选择"
  12. >
  13. <el-select
  14. v-model="skillParams.channel"
  15. @change="channelChange"
  16. style="width:160px"
  17. placeholder="请选择渠道"
  18. >
  19. <el-option
  20. v-for="item in skillChanneList"
  21. :key="item.code"
  22. :label="item.title"
  23. :value="item.code"
  24. ></el-option>
  25. </el-select>
  26. </el-form-item>
  27. <el-form-item class="dateForm-skill" name="data" label="查询日期" style="display:flex">
  28. <el-date-picker
  29. style="width:140px"
  30. v-model="skillParams.startDate"
  31. type="date"
  32. placeholder="起"
  33. ></el-date-picker>
  34. <el-date-picker
  35. style="width:140px"
  36. v-model="skillParams.endDate"
  37. type="date"
  38. placeholder="止"
  39. ></el-date-picker>
  40. </el-form-item>
  41. </div>
  42. <el-form-item>
  43. <el-button type="primary" @click="submitForm('skillParams')">搜索</el-button>
  44. </el-form-item>
  45. </el-form>
  46. <el-button type="primary" @click="onExportExcel" style="float: right">导出</el-button>
  47. <el-table id="table" ref="table1" :data="skillData" border style="width: 100%" :height="tableHeight" >
  48. <el-table-column
  49. label="日期"
  50. fixed
  51. align="left"
  52. header-align="center"
  53. min-width="140px"
  54. >
  55. <template slot-scope="scope">
  56. <span>{{changeDate(scope.row.day)}}</span>
  57. </template>
  58. </el-table-column>
  59. <el-table-column label="流量与观看" header-align="center">
  60. <el-table-column
  61. label="UV"
  62. prop="uv"
  63. align="right"
  64. header-align="center"
  65. min-width="100px"
  66. ></el-table-column>
  67. <el-table-column
  68. label="VIP用户 | -UV1"
  69. align="right"
  70. header-align="center"
  71. :render-header="renderheader"
  72. prop="vipUv"
  73. min-width="120px"
  74. ></el-table-column>
  75. <el-table-column
  76. label="非VIP用户 | UV2=UV-UV1"
  77. align="right"
  78. header-align="center"
  79. :render-header="renderheader"
  80. prop="waitVipUv"
  81. min-width="140px"
  82. ></el-table-column>
  83. <el-table-column
  84. label="VIP用户占比 | =UV1/UV"
  85. align="right"
  86. header-align="center"
  87. :render-header="renderheader"
  88. prop="vipProportion"
  89. min-width="140px"
  90. ></el-table-column>
  91. <el-table-column
  92. label="VIP权限用户 | =V0"
  93. align="right"
  94. header-align="center"
  95. :render-header="renderheader"
  96. prop="vipCount"
  97. min-width="140px"
  98. v-if="skillParams.channel === 'XYYF' || skillParams.channel === 'XYYFMOBILE'"
  99. ></el-table-column>
  100. <el-table-column
  101. label="VIP用户活跃率 | =UV1/V0"
  102. align="right"
  103. header-align="center"
  104. :render-header="renderheader"
  105. min-width="140px"
  106. prop="aliveVipProportion"
  107. v-if="skillParams.channel === 'XYYF' || skillParams.channel === 'XYYFMOBILE'"
  108. >
  109. </el-table-column>
  110. <!-- <el-table-column
  111. label="VV"
  112. prop="vv"
  113. align="right"
  114. header-align="center"
  115. min-width="100px"
  116. ></el-table-column>-->
  117. <el-table-column
  118. label="VV"
  119. align="right"
  120. header-align="center"
  121. :render-header="renderheader"
  122. prop="vv"
  123. min-width="140px"
  124. ></el-table-column>
  125. <el-table-column
  126. label="忠实用户"
  127. align="right"
  128. header-align="center"
  129. :render-header="renderheader"
  130. prop="fealtyCount"
  131. min-width="140px"
  132. v-if="skillParams.channel === 'XYYF' || skillParams.channel === 'XYYFMOBILE'"
  133. ></el-table-column>
  134. <el-table-column
  135. label="忠实用户占比"
  136. align="right"
  137. header-align="center"
  138. :render-header="renderheader"
  139. prop="fealtyProportion"
  140. min-width="140px"
  141. v-if="skillParams.channel === 'XYYF' || skillParams.channel === 'XYYFMOBILE'"
  142. ></el-table-column>
  143. <el-table-column
  144. label="人均播放 | =VV/UV"
  145. align="right"
  146. header-align="center"
  147. :render-header="renderheader"
  148. prop="perCapitaPlay"
  149. min-width="140px"
  150. ></el-table-column>
  151. </el-table-column>
  152. <el-table-column label="拉新,付费与转换率" header-align="center">
  153. <el-table-column
  154. label="连续包月 | -Qm1"
  155. align="right"
  156. header-align="center"
  157. :render-header="renderheader"
  158. prop="consecutiveMonthly"
  159. min-width="100px"
  160. ></el-table-column>
  161. <el-table-column
  162. label="新增连续包月 | -Qm1-1"
  163. align="right"
  164. header-align="center"
  165. :render-header="renderheader"
  166. min-width="140px"
  167. >
  168. <template slot-scope="scope">
  169. <span>{{scope.row.appendConsecutiveMonthly || 0}}</span>
  170. </template>
  171. </el-table-column>
  172. <el-table-column
  173. label="续包月续费 | -Qm1-2"
  174. align="right"
  175. header-align="center"
  176. :render-header="renderheader"
  177. min-width="140px"
  178. >
  179. <template slot-scope="scope">
  180. <span>{{scope.row.continueConsecutiveMonthly || 0}}</span>
  181. </template>
  182. </el-table-column>
  183. <el-table-column
  184. label="月包 | -Qm2"
  185. align="right"
  186. header-align="center"
  187. :render-header="renderheader"
  188. prop="monthlyPayment"
  189. min-width="90px"
  190. ></el-table-column>
  191. <el-table-column
  192. label="年包 | -Qy"
  193. align="right"
  194. header-align="center"
  195. :render-header="renderheader"
  196. prop="yearlyPayment"
  197. ></el-table-column>
  198. <el-table-column
  199. label="付费订单数量| Q=Qm1+Qm2+Qy"
  200. align="right"
  201. header-align="center"
  202. :render-header="renderheader"
  203. prop="totalPayment"
  204. min-width="160px"
  205. ></el-table-column>
  206. <el-table-column
  207. label="r1 | =Qm1/Q"
  208. align="right"
  209. header-align="center"
  210. :render-header="renderheader"
  211. prop="consecutiveConversion"
  212. min-width="130px"
  213. ></el-table-column>// 月包占比
  214. <el-table-column
  215. label="r2 | =Qm2/Q"
  216. align="right"
  217. header-align="center"
  218. :render-header="renderheader"
  219. prop="monthlyConversion"
  220. min-width="130px"
  221. ></el-table-column>// 年包占比
  222. <el-table-column
  223. label="r3 | =Qy/Q"
  224. align="right"
  225. header-align="center"
  226. :render-header="renderheader"
  227. prop="yearlyConversion"
  228. min-width="130px"
  229. ></el-table-column>
  230. <el-table-column
  231. label="R | =(Qm1+Qm2+Qy)/UV2"
  232. align="right"
  233. header-align="center"
  234. :render-header="renderheader"
  235. prop="totalConversion"
  236. min-width="190px"
  237. ></el-table-column>
  238. <el-table-column
  239. label="R1 | =(Qm1-1+Qm2+Qy)/UV2"
  240. align="right"
  241. header-align="center"
  242. :render-header="renderheader"
  243. min-width="190px"
  244. >
  245. <template slot-scope="scope">
  246. <span>{{scope.row.appendTotalConversion || 0}}</span>
  247. </template>
  248. </el-table-column>
  249. </el-table-column>
  250. </el-table>
  251. <!-- <el-pagination
  252. v-if="orderList.totalElements"
  253. background
  254. layout="prev, pager, next"
  255. :total="orderList.totalElements"
  256. @current-change="changePage">
  257. </el-pagination>-->
  258. </el-card>
  259. </div>
  260. </template>
  261. <script>
  262. import { mapGetters } from "vuex";
  263. import formatDate from "../../utils/formatTime";
  264. import { getBaiduCourseList } from '../../api/skill'
  265. import downTable from '@/utils/downTable'
  266. export default {
  267. data() {
  268. return {
  269. tableHeight: 500,
  270. dateValue: "",
  271. skillParams: {
  272. skillId: "",
  273. startDate: "",
  274. endDate: "",
  275. channel: "",
  276. classId: "",
  277. channelCode: "",
  278. packageId: ""
  279. },
  280. channelList: "",
  281. skillChanneList: [
  282. { title: "百度-小度在家", code: "BAIDU" },
  283. { title: "阿里-天猫精灵", code: "ALI" },
  284. // { title: "百度-义方小学堂", code: "XYYF" }, //注释掉百度义方小学堂
  285. { title: "蜜蜂视频", code: "MIFENG" },
  286. ],
  287. baiduCourseList: []
  288. };
  289. },
  290. computed: {
  291. ...mapGetters({
  292. skillList: "skillList",
  293. orderList: "orderList",
  294. productList: "productList",
  295. skillData: "skillData"
  296. })
  297. },
  298. created() {
  299. // this.$store.dispatch("getOrderList", this.skillParams);
  300. this.skillParams.startDate = this.getYesterDay();
  301. this.skillParams.endDate = this.getYesterDay();
  302. this.tableHeight = document.documentElement.clientHeight * 0.75;
  303. },
  304. methods: {
  305. // 搜索
  306. submitForm(formName) {
  307. this.skillParams.startDate = this.skillParams.startDate
  308. ? formatDate(this.skillParams.startDate, 2)
  309. : "";
  310. this.skillParams.endDate = this.skillParams.endDate
  311. ? formatDate(this.skillParams.endDate, 2)
  312. : "";
  313. this.$refs[formName].validate(valid => {
  314. if (valid) {
  315. console.log(this.skillParams.channel)
  316. if(this.skillParams.channel === 'BAIDU'){
  317. this.$store.dispatch("getSkillData", this.skillParams);
  318. } else if(this.skillParams.channel === 'ALI') {
  319. this.$store.dispatch("getAliData", this.skillParams);
  320. } else {
  321. console.log(this.skillParams)
  322. this.skillParams.channelCode = "100501";
  323. this.skillParams.packageId = "10050101";
  324. this.$store.dispatch("getMiFengData", this.skillParams);
  325. }
  326. } else {
  327. console.log("error submit!!");
  328. return false;
  329. }
  330. });
  331. this.$refs.table1.doLayout();
  332. },
  333. // 渠道下课程分页
  334. changePage(e) {
  335. this.skillParams.pageNo = e;
  336. this.$store.dispatch("getSkillData", this.skillParams);
  337. },
  338. // 表头折行
  339. renderheader(h, { column, $index }) {
  340. return h("span", {}, [
  341. h("span", {}, column.label.split("|")[0]),
  342. h("br"),
  343. h("span", {}, column.label.split("|")[1])
  344. ]);
  345. },
  346. getYesterDay() {
  347. let yesterday = new Date().getTime() - 86400000;
  348. return formatDate(yesterday, 2);
  349. },
  350. channelChange(val) {
  351. if (val === 'BAIDU' ) {
  352. this.skillParams.skillId = "6";
  353. } else if( val ==='ALI') {
  354. this.skillParams.skillId = "13";
  355. }
  356. // 清空数据
  357. this.$store.dispatch("clearList");
  358. },
  359. columnZero() {
  360. return "0";
  361. },
  362. columnZeroPercent(row, column, index) {
  363. if (row.totalPayment && column.property === "yearlyConversion") {
  364. return "100.00%";
  365. } else {
  366. return "0.00%";
  367. }
  368. // if(row.)
  369. },
  370. changeDate(date) {
  371. return formatDate(date, 4)
  372. },
  373. // 导出表格
  374. onExportExcel() {
  375. const nameList = this.skillChanneList.filter((item) => item.code === this.skillParams.channel)[0]
  376. const name = (nameList && nameList.title) || ''
  377. const list = this.skillList.filter((item) => item.skillId === this.skillParams.skillId)[0] || ''
  378. const productName = list && list.skillName ? '-' + list.skillName : ''
  379. const date = this.skillParams.startDate + '至' + this.skillParams.endDate
  380. downTable('table', [
  381. {wch: 15}, // "characters"
  382. {wch: 6},
  383. {wch: 15},
  384. {wch: 20},
  385. {wch: 20},
  386. {wch: 6},
  387. {wch: 15},
  388. {wch: 15},
  389. {wch: 18},
  390. {wch: 18},
  391. {wch: 10},
  392. {wch: 15},
  393. {wch: 25},
  394. {wch: 10},
  395. {wch: 10},
  396. {wch: 10},
  397. {wch: 20},
  398. {wch: 25}
  399. ], name + productName + date)
  400. }
  401. }
  402. };
  403. </script>
  404. <style lang="less" scoped>
  405. .pruductForm-skill {
  406. margin-left: 10px;
  407. display: flex;
  408. }
  409. .topForm {
  410. display: flex;
  411. align-items: center;
  412. flex-direction: row;
  413. flex-shrink: 0;
  414. justify-content: space-between;
  415. }
  416. .leftForm-skill {
  417. display: flex;
  418. align-items: center;
  419. flex-direction: row;
  420. justify-content: space-between;
  421. }
  422. .dateForm-skill {
  423. margin-left: 18px;
  424. }
  425. </style>