index.blade.php 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. @extends('layouts.base')
  2. @section('title', "活动列表")
  3. @section('content')
  4. <style>
  5. .rightlist #app .rightlist-head{line-height:50px;padding:15px 0;}
  6. .rightlist #app{margin-left:30px;}
  7. .el-form-item__label{padding-right:30px;}
  8. .tip{font-size:12px;color:#999;}
  9. .rightlist-head-con{padding-right:20px;font-size:16px;color:#888;}
  10. /* .rightlist-head-con{float:left;padding-right:20px;font-size:16px;color:#888;} */
  11. .el-tag{font-weight:700;font-size:15px;margin-bottom:30px;}
  12. .el-icon-edit{font-size:16px;padding:0 15px;color:#409EFF;cursor: pointer;}
  13. /* 滑块选择小白点 */
  14. .el-switch.is-checked .el-switch__core::after {left: 100%;margin-left: -17px;}
  15. .el-switch__core::after {content: "";position: absolute;top: 1px;left: 1px;border-radius: 100%;transition: all .3s;width: 16px;height: 16px;background-color: #fff;}
  16. [v-cloak]{
  17. display:none;
  18. }
  19. /* css */
  20. .list-title{display:flex;width:100%;background:#f9f9f9;padding:15px 10px;font-weight:900;border:1px solid #e9e9e9;}
  21. .list-title .list-title-1{display:flex;align-items:center;justify-content: center;}
  22. .list-info{display:flex;border-left:1px solid #e9e9e9;border-right:1px solid #e9e9e9;padding:15px 10px;justify-content: space-between;background:#f9f9f9;}
  23. .list-con{display:flex;width:100%;border:1px solid #e9e9e9;font-size:12px;font-weight:500}
  24. .con-img{padding:10px;border-right:1px solid #e9e9e9;float:left;height:100%}
  25. .con-img-title{float:left;margin-left:5px;font-size:12px;font-weight:800;width:220px}
  26. .list-num{flex:3;display:flex;align-items:center;border-right:1px solid #e9e9e9;justify-content: center;}
  27. .list-gen{display:flex;align-items:center;justify-content: center;line-height:28px;}
  28. .list-gen-txt{flex:1;border-right:1px solid #e9e9e9;border-bottom:1px solid #e9e9e9;align-item:center;justify-content: center;display:flex;}
  29. .list-opt{flex:1;display:flex;align-items:center;border-left:1px solid #e9e9e9;justify-content: center;}
  30. /* 弹出框 */
  31. .dialog-title{display:flex;font-weight:800;line-height:50px}
  32. .dialog-con{flex:1;display:flex;justify-content: center;border:1px #ccc solid;align-items:center;}
  33. </style>
  34. <div id="qrcode" ref="qrcode" style="display:none;"></div>
  35. <div class="rightlist">
  36. <div id="app" v-loading="all_loading" v-cloak>
  37. <el-form :inline="true" :model="search_form" ref="search_form" style="padding:20px 0 0 0;">
  38. <el-row>
  39. <el-col :xs="12" :md="12" :lg="7">
  40. <el-form-item label="活动编号" prop="">
  41. <el-input v-model="search_form.id" placeholder="活动编号"></el-input>
  42. </el-form-item>
  43. </el-col>
  44. <el-col :xs="12" :md="12" :lg="7">
  45. <el-form-item label="商品名称" prop="">
  46. <el-input v-model="search_form.goods_name" placeholder="商品名称"></el-input>
  47. </el-form-item>
  48. </el-col>
  49. <el-col :xs="12" :md="12" :lg="7">
  50. <el-form-item label="活动名称" prop="">
  51. <el-input v-model="search_form.title" placeholder="活动名称"></el-input>
  52. </el-form-item>
  53. </el-col>
  54. <a href="#">
  55. <el-button type="success" icon="el-icon-search" @click="search()">搜索</el-button>
  56. </a>
  57. </el-col>
  58. </el-row>
  59. </el-form>
  60. <div style="padding:15px 0;">
  61. <a href="{{ yzWebFullUrl('plugin.fight-groups.admin.controllers.fight-groups-store.edit') }}">
  62. <el-button type="primary" icon="el-icon-plus" size="small">新增活动</el-button>
  63. </a>
  64. <el-popover class="item" placement="bottom" effect="light">
  65. <div style="text-align:center;">
  66. <div style="padding:5px 0;text-align:left;">手机扫码访问:</div>
  67. <img :src="img" alt="" style="width:100px;height:100px;">
  68. <el-input v-model="url" ref="home" style="position:absolute;opacity:0;"></el-input>
  69. <div style="padding-top:5px;">
  70. <a href="#" @click="copy()"><span>复制链接</span></a>&nbsp;&nbsp;
  71. <a href="#"><span>电脑上查看</span></a>
  72. </div>
  73. </div>
  74. <el-button icon="el-icon-mobile-phone" size="small" slot="reference" @click="homeCode()">拼团首页</el-button>
  75. </el-popover>
  76. </div>
  77. <div>
  78. <el-radio-group v-model="status" style="margin-bottom: 30px;">
  79. <el-radio-button label="0">全部</el-radio-button>
  80. <el-radio-button label="1">未开始</el-radio-button>
  81. <el-radio-button label="2">活动中</el-radio-button>
  82. <el-radio-button label="3">已结束</el-radio-button>
  83. </el-radio-group>
  84. </div>
  85. <!-- 活动列表start -->
  86. <div class="list-title" style="">
  87. <div style="flex:0 0 320px;">活动商品</div>
  88. <div class="list-title-1" style="flex:3;">库存</div>
  89. <div class="list-title-1" style="flex:6;">活动概况</div>
  90. <div class="list-title-1" style="flex:1;">管理</div>
  91. </div>
  92. <div v-for="(item,index) in list">
  93. <div class="list-info">
  94. <div >
  95. <strong>编号:</strong>[[item.id]]&nbsp;&nbsp;&nbsp;
  96. <strong>活动名称:</strong>[[item.title]]
  97. </div>
  98. <div ><span style="align-self:flex-end"><strong>开始时间:</strong>[[item.start_time]]</span></div>
  99. </div>
  100. <div class="list-con">
  101. <div style="flex:0 0 320px;">
  102. <div class="con-img">
  103. <div style="width:60px;height:60px;float:left">
  104. <img :src="item.has_one_goods&&item.has_one_goods!=null?item.has_one_goods.thumb:''" style="width:60px;height:60px;" alt="">
  105. </div>
  106. <div class="con-img-title">[[item.has_one_goods&&item.has_one_goods!=null?item.has_one_goods.title:'']]</div>
  107. <div style="float:left;width:100%">原价:¥[[item.has_one_goods&&item.has_one_goods!=null?item.has_one_goods.price:'']]</div>
  108. </div>
  109. </div>
  110. <div class="list-num">
  111. 剩余库存 : [[item.last_stock]]&nbsp;&nbsp;<el-button size="mini" @click="stock(item.id,item.has_one_goods.has_option)">库存明细</el-button>
  112. </div>
  113. <div style="flex:6;">
  114. <div v-for="(item1,index1) in item.has_many_level" class="list-gen">
  115. <div class="list-gen-txt">
  116. [[item1.member_num]]人团
  117. </div>
  118. <div class="list-gen-txt">
  119. [[item1.expire_in]]小时
  120. </div>
  121. <div class="list-gen-txt">
  122. 已成团:[[item1.has_many_success_team_count]]
  123. </div>
  124. <div class="list-gen-txt">
  125. 已开团:[[item1.has_many_opened_team_count]]
  126. </div>
  127. <div class="list-gen-txt">
  128. <!-- 已结束(手动) -->
  129. <span v-if="item1.status==0">未开始&nbsp;&nbsp;<a @click="manualEnd(item1.id)">手动结束</a></span>
  130. <span v-if="item1.status==1">已开始&nbsp;&nbsp;<a @click="manualEnd(item1.id)">手动结束</a></span>
  131. <span v-if="item1.status==2">已结束</span>
  132. </div>
  133. </div>
  134. </div>
  135. <div class="list-opt">
  136. <div>
  137. <el-popover class="item" placement="left" effect="light">
  138. <div style="text-align:center;">
  139. <div style="padding:5px 0;text-align:left;">活动推广二维码:</div>
  140. <img :src="img" alt="" style="width:100px;height:100px;">
  141. <div style="padding-top:5px;">
  142. <el-button @click="copyList(index)" type="mini">复制活动链接</el-button>
  143. <input v-model="item.url" :ref="'list'+index" style="position:absolute;opacity:0;height:1px;" />
  144. </div>
  145. </div>
  146. <a slot="reference" @click="listCode(index)">活动推广</a>
  147. </el-popover>
  148. <a v-if="is_show_arr[index]" :href="'{{ yzWebFullUrl('plugin.fight-groups.admin.controllers.fight-groups-store.edit', array('id' => '')) }}'+[[item.id]]"><div>编辑活动</div></a>
  149. <a :href="'{{ yzWebFullUrl('plugin.fight-groups.admin.controllers.team-store.index', array('id' => '')) }}'+[[item.id]]"><div>开团列表</div></a>
  150. <a @click="count(item.id)"><div>数据统计</div></a>
  151. <!-- <a :href="'{{ yzWebFullUrl('plugin.fight-groups.admin.controllers.order.index', array('fight_groups_id' => '')) }}'+[[item.id]]"><div>订单列表</div></a> -->
  152. </div>
  153. </div>
  154. </div>
  155. </div>
  156. <!-- 活动列表end -->
  157. <!-- 分页 -->
  158. <el-row>
  159. <el-col :span="24" align="right" style="padding:15px 5% 15px 0">
  160. <el-pagination layout="prev, pager, next" @current-change="currentChange" :current-page="current_page" :total="total" :page-size="per_size" background v-loading="loading"></el-pagination>
  161. </el-col>
  162. </el-row>
  163. <!-- 库存弹出框 -->
  164. <el-dialog title="库存" :visible.sync="stock_url" width="60%" v-loading="table_loading">
  165. <div class="dialog-title">
  166. <div class="dialog-con">规格</div>
  167. <div class="dialog-con">剩余库存</div>
  168. <div class="dialog-con">成团销量</div>
  169. </div>
  170. <div style="display:flex;font-weight:800;line-height:50px" v-for="(item,index) in stock_list" :key="index">
  171. <div class="dialog-con">[[item.option_title]]</div>
  172. <div class="dialog-con">[[item.last_stock]]</div>
  173. <div class="dialog-con">[[item.num]]</div>
  174. <!-- <div class="dialog-con">[[stock_list.data[0].option_title]]</div>
  175. <div class="dialog-con">[[stock_list.last_stock]]</div>
  176. <div class="dialog-con">[[stock_list.data[0].num]]</div> -->
  177. </div>
  178. <span slot="footer" class="dialog-footer">
  179. <el-button @click="stock_url = false">关 闭</el-button>
  180. <!-- <el-button type="primary" @click="img_text_url = false">确 定</el-button> -->
  181. </span>
  182. </el-dialog>
  183. <!-- 统计弹出框 -->
  184. <el-dialog title="数据统计" :visible.sync="count_url" width="60%" v-loading="table_loading">
  185. <div class="dialog-title">
  186. <div class="dialog-con">规格名称</div>
  187. <div class="dialog-con">成团销量</div>
  188. <div class="dialog-con">销量明细</div>
  189. </div>
  190. <div class="dialog-title" v-for="(item,index) in count_list">
  191. <div class="dialog-con">[[item.option_title]]</div>
  192. <div class="dialog-con">[[item.stock - item.last_stock]]</div>
  193. <div class="dialog-con" style="flex-wrap: wrap;">
  194. <div v-for="(item1,index1) in item.level" >
  195. [[item1.member_num]]人团 系统成团:[[item1.purchase_quantity_system]] 虚拟成团:[[item1.purchase_quantity_virtual]]
  196. </div>
  197. </div>
  198. </div>
  199. <span slot="footer" class="dialog-footer">
  200. <el-button @click="count_url = false">关 闭</el-button>
  201. <!-- <el-button type="primary" @click="img_text_url = false">确 定</el-button> -->
  202. </span>
  203. </el-dialog>
  204. </div>
  205. <script src="{{resource_get('plugins/fight-groups/views/admin/qrcode.min.js')}}"></script>
  206. <script>
  207. var app = new Vue({
  208. el:"#app",
  209. delimiters: ['[[', ']]'],
  210. data() {
  211. let data = {!! $data?:'{}' !!};
  212. console.log(data);
  213. return{
  214. tab:"all",
  215. stock_url:false,//库存弹出框
  216. count_url:false,//统计弹出框
  217. status:0,
  218. url:"",//首页链接
  219. search_form:{
  220. goods_name:"",
  221. id:"",
  222. title:""
  223. },
  224. search_goods:{},
  225. img:"",
  226. list:[],
  227. is_show_arr:[],//是否显示'编辑'按钮
  228. stock_list:{
  229. data:[{option_title:""}],
  230. },//库存信息
  231. count_list:{},
  232. loading:false,
  233. dialog_loading:false,
  234. all_loading:false,
  235. table_loading:false,
  236. //商品分页
  237. total:0,
  238. per_size:0,
  239. current_page:0,
  240. //门店名称
  241. store_name:'',
  242. store_id:0,
  243. rules:{},
  244. }
  245. },
  246. watch: {
  247. status() {
  248. var that = this;
  249. let json = {id:that.search_form.id,goods_name:that.search_form.goods_name,status:that.status,title:that.search_form.title};
  250. console.log(json)
  251. that.getData(json);
  252. }
  253. },
  254. created() {
  255. this.getData('{}');
  256. },
  257. methods: {
  258. getData(json) {
  259. var that = this;
  260. that.all_loading = true;
  261. that.$http.post("{!! yzWebFullUrl('plugin.fight-groups.admin.controllers.fight-groups-store.search') !!}",json).then(response => {
  262. console.log(response);
  263. if(response.data.result==1){
  264. that.list = response.data.data.data;
  265. let arr = [];
  266. // 列表时间处理及'编辑'按钮控制开关
  267. that.list.forEach((item,index) => {
  268. if(!item.has_one_goods){
  269. this.$set(item,'has_one_goods',{})
  270. let title = this.escapeHTML(item.has_one_goods ? item.has_one_goods.title : '')
  271. this.$set(item.has_one_goods,'title',title)
  272. console.log(item.has_one_goods.title,1)
  273. }else{
  274. console.log(item.has_one_goods,2)
  275. item.has_one_goods.title = this.escapeHTML(item.has_one_goods.title)
  276. }
  277. item.start_time = that.timeStyle(item.start_time)
  278. for(let i=0;i<item.has_many_level.length;i++) {
  279. if(item.has_many_level[i].status==0 || item.has_many_level[i].status==1) {
  280. arr.push(1);
  281. break;
  282. }
  283. }
  284. if(arr.length-1<index){
  285. arr.push(0);
  286. }
  287. });
  288. this.is_show_arr = arr;
  289. that.total = response.data.data.total;
  290. that.per_size = response.data.data.per_page;
  291. that.current_page = response.data.data.current_page;
  292. that.url = response.data.data.url;
  293. }
  294. else{
  295. that.$message.error(response.data);
  296. }
  297. console.log(this.list)
  298. that.all_loading = false;
  299. }),function(res){
  300. console.log(res);
  301. that.all_loading = false;
  302. };
  303. },
  304. // 查看库存
  305. stock(id,option) {
  306. console.log(id)
  307. console.log(option)
  308. let data = {
  309. id:id,
  310. option:option
  311. }
  312. var that = this;
  313. that.table_loading = true;
  314. that.stock_url = true;
  315. that.$http.post("{!! yzWebFullUrl('plugin.fight-groups.admin.controllers.fight-groups-store.stock-detail') !!}",{data:data}).then(response => {
  316. if(response.data.result==1){
  317. that.stock_list = response.data.data;
  318. console.log(that.stock_list)
  319. }
  320. else{
  321. that.$message.error(response.data.msg);
  322. }
  323. that.table_loading = false;
  324. }),function(res){
  325. console.log(res);
  326. that.table_loading = false;
  327. };
  328. },
  329. // 统计
  330. count(id) {
  331. console.log(id)
  332. var that = this;
  333. that.table_loading = true;
  334. that.count_url = true;
  335. that.$http.post("{!! yzWebFullUrl('plugin.fight-groups.admin.controllers.fight-groups-store.data-statistics') !!}",{id:id}).then(response => {
  336. if(response.data.result==1){
  337. that.count_list = response.data.data;
  338. console.log(that.count_list)
  339. }
  340. else{
  341. that.$message.error(response.data.msg);
  342. }
  343. that.table_loading = false;
  344. }),function(res){
  345. console.log(res);
  346. that.table_loading = false;
  347. };
  348. },
  349. // 手动结束
  350. manualEnd(id) {
  351. console.log("12312313")
  352. var that = this;
  353. that.table_loading = true;
  354. that.$http.post("{!! yzWebFullUrl('plugin.fight-groups.admin.controllers.fight-groups-store.close-level') !!}",{id:id}).then(response => {
  355. if(response.data.result==1){
  356. that.$message.success(response.data.msg);
  357. this.getData('{}');
  358. }
  359. else{
  360. that.$message.error(response.data.msg);
  361. }
  362. that.table_loading = false;
  363. }),function(res){
  364. console.log(res);
  365. that.table_loading = false;
  366. };
  367. },
  368. // 搜索
  369. search() {
  370. var that = this;
  371. let json = {id:that.search_form.id,goods_name:that.search_form.goods_name,status:that.status,title:that.search_form.title,}
  372. that.getData(json);
  373. },
  374. // 分页
  375. currentChange(page) {
  376. var that = this;
  377. let json = {id:that.search_form.id,goods_name:that.search_form.goods_name,status:that.status,title:that.search_form.title,page:page}
  378. that.getData(json);
  379. },
  380. qrcodeScan (url) {//生成二维码
  381. let qrcode = new QRCode('qrcode', {
  382. width: 100, // 二维码宽度
  383. height: 100, // 二维码高度
  384. render: 'image',
  385. text: url
  386. });
  387. console.log($("canvas"))
  388. var data = $("canvas")[$("canvas").length-1].toDataURL().replace("image/png", "image/octet-stream;");
  389. console.log(url)
  390. // this.$set(this.img,data);
  391. this.img = data;
  392. console.log(this.img)
  393. },
  394. // 首页二维码
  395. homeCode() {
  396. this.qrcodeScan(this.url);
  397. console.log("asdasd")
  398. },
  399. // 复制首页链接
  400. copy() {
  401. that = this;
  402. let Url = that.$refs['home'];
  403. Url.select(); // 选择对象
  404. document.execCommand("Copy",false);
  405. that.$message({message:"复制成功!",type:"success"});
  406. },
  407. // 活动二维码
  408. listCode(index) {
  409. console.log(index)
  410. this.qrcodeScan(this.list[index].url);
  411. },
  412. // 复制活动链接
  413. copyList(index) {
  414. that = this;
  415. let Url = that.$refs['list'+index];
  416. Url[0].select(); // 选择对象
  417. document.execCommand("Copy",false);
  418. that.$message({message:"复制成功!",type:"success"});
  419. },
  420. submit(){
  421. var that = this;
  422. console.log(this.data);
  423. that.$http.post("{!! yzWebFullUrl('plugin.wechat.admin.reply.controller.default-reply.add') !!}",{keywords_id:that.data.id}).then(response => {
  424. console.log(response);
  425. if(response.data.result==1){
  426. that.keyword_list = response.data.data;
  427. that.$message.success("保存成功!");
  428. window.location.href='{!! yzWebFullUrl('plugin.wechat.admin.reply.controller.default-reply.index') !!}';
  429. }
  430. else{
  431. that.$message.error(response.data);
  432. }
  433. }),function(res){
  434. console.log(res);
  435. };
  436. },
  437. escapeHTML(a) {
  438. a = "" + a;
  439. return a.replace(/&amp;/g, "&").replace(/&lt;/g, "<").replace(/&gt;/g, ">").replace(/&quot;/g, "\"").replace(/&apos;/g, "'");;
  440. },
  441. add0(m) {
  442. return m<10?'0'+m:m
  443. },
  444. timeStyle(time) {
  445. let time1 = new Date(time*1000);
  446. let y = time1.getFullYear();
  447. let m = time1.getMonth()+1;
  448. let d = time1.getDate();
  449. let h = time1.getHours();
  450. let mm = time1.getMinutes();
  451. let s = time1.getSeconds();
  452. return y+'-'+this.add0(m)+'-'+this.add0(d)+' '+this.add0(h)+':'+this.add0(mm)+':'+this.add0(s);
  453. },
  454. },
  455. })
  456. </script>
  457. @endsection