初始上传

This commit is contained in:
2026-04-04 17:27:12 +08:00
parent 4d80d28eb4
commit b7e11774ee
11191 changed files with 1588469 additions and 0 deletions

View File

@@ -0,0 +1,69 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\api\controller;
use addon\live\model\Room;
use app\api\controller\BaseApi;
/**
* 直播
*/
class Live extends BaseApi
{
/**
* 获取直播间信息
*/
public function info()
{
$room = new Room();
$condition = [
'site_id' => $this->site_id,
'live_status' => '101'
];
// 优先查询进行中的
$room_info = $room->getRoomInfo($condition);
if (empty($room_info[ 'data' ])) {
$condition[ 'live_status' ] = '102';
// 没有进行中的查询未开始的
$room_info = $room->getRoomInfo($condition);
}
return $this->response($room_info);
}
/**
* 获取直播间列表
* @return false|string
*/
public function page()
{
$page = $this->params['page'] ?? 1;
$page_size = $this->params['page_size'] ?? PAGE_LIST_ROWS;
$room = new Room();
$data = $room->getRoomPageList([ 'site_id' => $this->site_id ], '*', 'live_status asc', $page, $page_size);
return $this->response($data);
}
/**
* 修改直播间状态
*/
public function modifyLiveStatus()
{
$room_id = $this->params[ 'room_id' ] ?? 0;
$status = $this->params[ 'status' ] ?? '';
if (empty($status)) return $this->response($this->error());
$room = new Room();
$res = $room->updateRoomInfo([ 'live_status' => $status ], [ [ 'roomid', '=', $room_id ], [ 'site_id', '=', $this->site_id ] ]);
return $this->response($res);
}
}

View File

@@ -0,0 +1,21 @@
<?php
namespace addon\live\component\controller;
use app\component\controller\BaseDiyView;
/**
* 小程序直播·组件
*
*/
class LiveInfo extends BaseDiyView
{
/**
* 设计界面
*/
public function design()
{
return $this->fetch("live_info/design.html");
}
}

View File

@@ -0,0 +1,25 @@
.live-wrap { /*background: #fff;*/ /* border-radius: 8px; */overflow: hidden;}
.live-wrap .banner-wrap {width: 100%;position: relative;line-height: 1;display: flex;}
.live-wrap .banner-wrap img {width: 100%;height:104px;}
.live-wrap .banner-wrap .shade {width: 100%;height: 100%;position: absolute;background: rgba(180, 180, 180, 0.3);left: 0;top: 0;z-index: 5;}
.live-wrap .banner-wrap .wrap {width: 100%;height: 100%;position: absolute;left: 0;top: 0;z-index: 10;padding: 10px;box-sizing: border-box;}
.live-wrap .banner-wrap .wrap .room-name {font-size: 14px;color: #fff;line-height: 1;width: 100%;white-space: nowrap;overflow: hidden;text-overflow: ellipsis;display: flex;align-items: center;}
.live-wrap .banner-wrap .room-name .status-name {display: inline-block;font-size: 12px;color: #fff;padding: 4px 6px;background-color: #FF4544;border-radius: 18px;margin-right: 10px;}
.live-wrap .banner-wrap .room-name .status-name img{height: 9px;width: 9px;vertical-align: middle;margin-right: 5px;}
.live-wrap .room-name .status-name .iconzhibozhong {font-size: 12px;color: #fff;margin-right: 2px;}
.live-wrap .room-info {padding: 10px 10px;background: #fff;display: flex;align-items: center;height: 30px;color: #303133;font-size: 14px;}
.live-wrap .room-info .anchor-img {width: 30px;height: 30px;border-radius: 50%;overflow: hidden;margin-right: 10px;}
.live-wrap .room-info .anchor-name, .live-wrap .room-info .goods-text {font-size: 12px;line-height: 1;color:#303133;}
.live-wrap .room-info .separate {color: #808080;margin: 0 5px;line-height: 1;}
.component-live-info h3 {font-size: 14px; /* font-weight: 600; */padding: 5px 10px 10px 10px;}
.component-live-info .layui-form-item .layui-input-inline {padding-left: 20px;}
/* 复选 */
.component-live-info .checkbox-wrap .layui-form-checkbox, .checkbox-wrap .layui-input-inline-checkbox .layui-form-checkbox {float: right;}
.component-live-info .checkbox-wrap .layui-form-item .layui-form-checkbox[lay-skin=primary] {margin-top: -4px;padding-left:0;}
.component-live-info .checkbox-wrap .layui-form-item .layui-input-inline-checkbox .layui-form-checkbox[lay-skin=primary] {margin-top: 8px;}
.component-live-info .layui-form-checkbox[lay-skin=primary] {margin-top: -4px;padding-left:0;}
.component-live-info .tab-wrap{display: none !important;}

View File

@@ -0,0 +1,49 @@
<nc-component :data="data[index]" class="component-live-info">
<!-- 预览 -->
<template slot="preview">
<div class="live-wrap">
<div class="banner-wrap">
<img src="{$resource_path}/img/live_default_banner.png">
<div class="shade"></div>
<div class="wrap">
<div class="room-name">
<span class="status-name"><img src="{$resource_path}/img/live_default_icon.png"/>直播中</span>
双十一活动,限时秒杀!
</div>
</div>
</div>
<div class="room-info" v-if="nc.isShowAnchorInfo || nc.isShowLiveGood">
<template v-if="nc.isShowAnchorInfo">
<img src="{:img('public/static/img/default_img/head.png')}" class="anchor-img">
<span class="anchor-name">主播:主播昵称</span>
</template>
<template v-if="nc.isShowAnchorInfo && nc.isShowLiveGood">
<span class="separate">|</span>
</template>
<template v-if="nc.isShowLiveGood">
<span class="goods-text">直播商品:名称</span>
</template>
</div>
</div>
</template>
<!-- 内容编辑 -->
<template slot="edit-content">
<template v-if="nc.lazyLoad">
<live-set></live-set>
</template>
</template>
<!-- 样式编辑 -->
<template slot="edit-style"></template>
<!-- 资源 -->
<template slot="resource">
<css src="{$resource_path}/css/design.css"></css>
<js src="{$resource_path}/js/design.js"></js>
</template>
</nc-component>

Binary file not shown.

After

Width:  |  Height:  |  Size: 169 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 325 B

View File

@@ -0,0 +1,50 @@
// 显示内容
var showContentHtml = '<div class="layui-form-item goods-show-box checkbox-wrap">';
showContentHtml += '<div class="layui-input-block">';
showContentHtml += '<div class="layui-input-inline-checkbox">';
showContentHtml += '<span>主播信息</span>';
showContentHtml += '<div @click="changeStatus(\'isShowAnchorInfo\')" class="layui-unselect layui-form-checkbox" :class="{\'layui-form-checked\': (data.isShowAnchorInfo == 1)}" lay-skin="primary"><i class="layui-icon layui-icon-ok"></i></div>';
showContentHtml += '</div>';
showContentHtml += '<div class="layui-input-inline-checkbox">';
showContentHtml += '<span>直播商品</span>';
showContentHtml += '<div @click="changeStatus(\'isShowLiveGood\')" class="layui-unselect layui-form-checkbox" :class="{\'layui-form-checked\': (data.isShowLiveGood == 1)}" lay-skin="primary"><i class="layui-icon layui-icon-ok"></i></div>';
showContentHtml += '</div>';
showContentHtml += '</div>';
showContentHtml += '</div>';
Vue.component("live-show-content", {
template: showContentHtml,
data: function () {
return {
data: this.$parent.data,
isShowAnchorInfo: this.$parent.data.isShowAnchorInfo,
isShowLiveGood: this.$parent.data.isShowLiveGood,
};
},
created: function () {
if(!this.$parent.data.verify) this.$parent.data.verify = [];
this.$parent.data.verify.push(this.verify);//加载验证方法
},
methods: {
verify :function () {
var res = { code: true, message: "" };
return res;
},
changeStatus: function(field) {
this.$parent.data[field] = this.$parent.data[field] ? 0 : 1;
}
}
});
// 显示内容
var liveSetHtml = '<div></div>';
Vue.component("live-set", {
template: liveSetHtml,
data: function () {
return {};
},
created: function () {
this.$parent.data.ignore = ['marginBoth', 'textColor', 'elementAngle', 'componentAngle', 'elementBgColor', 'componentBgColor', 'pageBgColor']; //加载忽略内容 -- 其他设置中的属性设置
this.$parent.data.ignoreLoad = true; // 等待忽略数组赋值后加载
}
});

67
addon/live/config/diy_view.php Executable file
View File

@@ -0,0 +1,67 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
return [
// 自定义模板页面类型,格式:[ 'title' => '页面类型名称', 'name' => '页面标识', 'path' => '页面路径', 'value' => '页面数据json格式' ]
'template' => [],
// 后台自定义组件——装修
'util' => [
[
'name' => 'LiveInfo',
'title' => '小程序直播',
'type' => 'PROMOTION',
'value' => '{}',
'sort' => '30009',
'support_diy_view' => '',
'max_count' => 1,
'icon' => 'iconfont iconzhibojian'
]
],
// 自定义页面路径
'link' => [
[
'name' => 'LIVE',
'title' => '直播',
'parent' => 'MARKETING_LINK',
'wap_url' => '',
'web_url' => '',
'sort' => 0,
'child_list' => [
[
'name' => 'LIVE_LIST',
'title' => '直播',
'wap_url' => '/pages_tool/live/list',
'web_url' => '',
'sort' => 0
]
]
]
],
// 自定义图标库
'icon_library' => [],
// uni-app 组件,格式:[ 'name' => '组件名称/文件夹名称', 'path' => '文件路径/目录路径' ]多个逗号隔开自定义组件名称前缀必须是diy-,也可以引用第三方组件
'component' => [],
// uni-app 页面,多个逗号隔开
'pages' => [],
// 模板信息,格式:'title' => '模板名称', 'name' => '模板标识', 'cover' => '模板封面图', 'preview' => '模板预览图', 'desc' => '模板描述'
'info' => [],
// 主题风格配色格式可以自由定义扩展【在uni-app中通过this.themeStyle... 获取定义的颜色字段例如this.themeStyle.main_color】
'theme' => [],
// 自定义页面数据,格式:[ 'title' => '页面名称', 'name' => "页面标识", 'value' => [页面数据json格式] ]
'data' => []
];

28
addon/live/config/event.php Executable file
View File

@@ -0,0 +1,28 @@
<?php
// 事件定义文件
return [
'bind' => [
],
'listen' => [
//展示活动
'ShowPromotion' => [
'addon\live\event\ShowPromotion',
],
'WeappMenu' => [
'addon\live\event\WeappMenu',
],
// 轮询更新直播商品状态
'LiveGoodsStatus' => [
'addon\live\event\LiveGoodsStatus',
],
// 轮询更新直播间状态
'LiveRoomStatus' => [
'addon\live\event\LiveRoomStatus',
]
],
'subscribe' => [
],
];

21
addon/live/config/info.php Executable file
View File

@@ -0,0 +1,21 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
return [
'name' => 'live',
'title' => '小程序直播',
'description' => '在小程序中实现直播互动与商品销售闭环',
'type' => 'tool', //插件类型 system :系统插件(自动安装), promotion:扩展营销插件 tool:工具插件
'status' => 1,
'author' => '',
'version' => '5.5.3',
'version_no' => '553250709001',
'content' => '',
];

106
addon/live/config/menu_shop.php Executable file
View File

@@ -0,0 +1,106 @@
<?php
// +----------------------------------------------------------------------
// | 平台端菜单设置
// +----------------------------------------------------------------------
return [
[
'name' => 'LIVE_ROOT',
'title' => '小程序直播',
'url' => 'live://shop/room/index',
'picture' => 'addon/live/shop/view/public/img/live_new.png',
'picture_selected' => 'addon/live/shop/view/public/img/live_select.png',
'parent' => 'PROMOTION_TOOL',
'is_show' => 1,
'sort' => 1,
'child_list' => [
[
'name' => 'LIVE_ROOM',
'title' => '直播间',
'url' => 'live://shop/room/index',
'is_show' => 1,
'sort' => 1,
'child_list' => [
[
'name' => 'ADD_LIVE_ROOM',
'title' => '添加直播间',
'url' => 'live://shop/room/add',
'is_show' => 0,
'sort' => 1,
'type' => 'button',
'type' => 'button',
],
[
'name' => 'DELETE_LIVE_ROOM',
'title' => '删除直播间',
'url' => 'live://shop/room/delete',
'is_show' => 0,
'sort' => 2,
'type' => 'button',
'type' => 'button',
],
[
'name' => 'LIVE_ROOM_OPERATE',
'title' => '运营',
'url' => 'live://shop/room/operate',
'is_show' => 0,
'sort' => 3,
'type' => 'button',
'type' => 'button',
],
[
'name' => 'SYNC_LIVE_ROOM',
'title' => '同步直播间',
'url' => 'live://shop/room/sync',
'is_show' => 0,
'sort' => 4,
'type' => 'button',
'type' => 'button',
],
[
'name' => 'ADD_GOODS_TO_LIVE_ROOM',
'title' => '添加商品到直播间',
'url' => 'live://shop/room/addGoods',
'is_show' => 0,
'sort' => 5,
'type' => 'button',
'type' => 'button',
]
]
],
[
'name' => 'LIVE_GOODS',
'title' => '直播商品',
'url' => 'live://shop/goods/index',
'is_show' => 1,
'sort' => 2,
'child_list' => [
[
'name' => 'ADD_LIVE_GOODS',
'title' => '添加商品',
'url' => 'live://shop/goods/add',
'is_show' => 0,
'sort' => 1,
'type' => 'button',
],
[
'name' => 'DELETE_LIVE_GOODS',
'title' => '删除商品',
'url' => 'live://shop/goods/delete',
'is_show' => 0,
'sort' => 2,
'type' => 'button',
],
[
'name' => 'SYNC_LIVE_GOODS',
'title' => '同步直播商品库',
'url' => 'live://shop/goods/sync',
'is_show' => 0,
'sort' => 3,
'type' => 'button',
]
]
],
]
]
];

1
addon/live/data/install.sql Executable file
View File

@@ -0,0 +1 @@
SET NAMES 'utf8';

1
addon/live/data/uninstall.sql Executable file
View File

@@ -0,0 +1 @@
SET NAMES 'utf8';

25
addon/live/event/Install.php Executable file
View File

@@ -0,0 +1,25 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\event;
/**
* 应用安装
*/
class Install
{
/**
* 执行安装
*/
public function handle()
{
return success();
}
}

View File

@@ -0,0 +1,30 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\event;
use addon\live\model\Goods;
/**
* 获取商品审核状态
*/
class LiveGoodsStatus
{
/**
* 获取商品审核状态
* @param $param
*/
public function handle($param)
{
$goods = new Goods();
$goods->getGoodsAuditStatus($param[ 'relate_id' ]);
}
}

View File

@@ -0,0 +1,30 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\event;
use addon\live\model\Room;
/**
* 轮询更新直播间状态
*/
class LiveRoomStatus
{
/**
* 轮询更新直播间状态
* @param $param
*/
public function handle($param)
{
$room = new Room();
$room->updateRoomStatus($param[ 'relate_id' ]);
}
}

View File

@@ -0,0 +1,46 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\event;
/**
* 活动展示
*/
class ShowPromotion
{
/**
* 活动展示
* @return array
*/
public function handle()
{
$data = [
'shop' => [
[
//插件名称
'name' => 'live',
//店铺端展示分类 shop:营销活动 member:互动营销
'show_type' => 'tool',
//展示主题
'title' => '小程序直播',
//展示介绍
'description' => '商家直播带货渠道',
//展示图标
'icon' => 'addon/live/icon.png',
//跳转链接
'url' => 'live://shop/room/index',
]
]
];
return $data;
}
}

30
addon/live/event/UnInstall.php Executable file
View File

@@ -0,0 +1,30 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\event;
/**
* 应用卸载
*/
class UnInstall
{
/**
* 执行卸载
*/
public function handle()
{
try {
return error('', "系统插件不允许删除");
} catch (\Exception $e) {
return error('', $e->getMessage());
}
}
}

36
addon/live/event/WeappMenu.php Executable file
View File

@@ -0,0 +1,36 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\event;
/**
* 小程序菜单
*/
class WeappMenu
{
/**
* 小程序菜单
* @param $param
* @return array
*/
public function handle($param)
{
if (addon_is_exit('live', $param[ 'site_id' ])) {
$data = [
'title' => '小程序直播',
'description' => '在小程序中实现直播互动与商品销售闭环',
'url' => 'live://shop/room/index',
'icon' => 'addon/live/icon.png'
];
return $data;
}
}
}

BIN
addon/live/icon.png Executable file

Binary file not shown.

After

Width:  |  Height:  |  Size: 929 B

198
addon/live/model/Goods.php Executable file
View File

@@ -0,0 +1,198 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\model;
use app\model\BaseModel;
use app\model\system\Cron;
use app\model\upload\Upload;
class Goods extends BaseModel
{
private $liveStatus = [
0 => '未审核',
1 => '审核中',
2 => '审核通过',
3 => '审核驳回'
];
/**
* 获取直播间列表
* @param array $condition
* @param bool $field
* @param string $order
* @param int $page
* @param int $list_rows
* @param string $alias
* @param array $join
* @return array
*/
public function getGoodsPageList($condition = [], $field = true, $order = '', $page = 1, $list_rows = PAGE_LIST_ROWS, $alias = 'a', $join = [])
{
$data = model('weapp_goods')->pageList($condition, $field, $order, $page, $list_rows, $alias, $join);
if (!empty($data[ 'list' ])) {
foreach ($data[ 'list' ] as $k => $item) {
$data[ 'list' ][ $k ][ 'status_name' ] = $this->liveStatus[ $item[ 'status' ] ] ?? '';
}
}
return $this->success($data);
}
/**
* 同步商品库商品
*/
public function syncGoods($start, $limit, $site_id, $status = 2)
{
$live = new Live($site_id);
$result = $live->getGoodsList($start, $limit, $status);
if ($result[ 'code' ] < 0) return $result;
if (!empty($result[ 'data' ][ 'goods' ])) {
$upload = new Upload($site_id, 'shop');
$upload->setPath('upload/live/goods/');
$goodsID_arr = [];
foreach ($result[ 'data' ][ 'goods' ] as $item) {
$goodsID_arr[] = $item[ 'goodsId' ];
preg_match("/(pages\/goods\/detail\?sku_id=)(\d*)$/", $item[ 'url' ], $matches);
$data = [
'site_id' => $site_id,
'goods_id' => $item[ 'goodsId' ],
'name' => $item[ 'name' ],
'price' => $item[ 'price' ],
'status' => $status,
'url' => $item[ 'url' ],
'sku_id' => $matches[ 2 ] ?? 0,
'third_party_tag' => $item[ 'thirdPartyTag' ]
];
$room_info = model('weapp_goods')->getInfo([ [ 'goods_id', '=', $item[ 'goodsId' ] ], [ 'site_id', '=', $site_id ] ], 'id');
if (empty($room_info)) {
if (is_url($item[ 'coverImgUrl' ])) {
$pull_result = $upload->remotePull($item[ 'coverImgUrl' ]);
$pull_result = $pull_result[ 'data' ];
if (isset($pull_result[ 'pic_path' ]) && !empty($pull_result[ 'pic_path' ])) {
$data[ 'cover_img' ] = $pull_result[ 'pic_path' ];
} else {
$data[ 'cover_img' ] = $item[ 'coverImgUrl' ];
}
}
model('weapp_goods')->add($data);
} else {
model('weapp_goods')->update($data, [ [ 'id', '=', $room_info[ 'id' ] ] ]);
}
}
$list_id_arr = model('weapp_goods')->getList([ [ 'site_id', '=', $site_id ] ]);
foreach ($list_id_arr as $key => $val) {
if (!in_array($val[ 'goods_id' ], $goodsID_arr)) {
model('weapp_goods')->delete([ 'goods_id' => $val[ 'goods_id' ] ]);
}
}
$total_page = ceil($result[ 'data' ][ 'total' ] / $limit);
return $this->success([ 'page' => $start, 'total_page' => $total_page ]);
} else {
return $this->success([ 'page' => $start, 'total_page' => 1 ]);
}
}
/**
* 添加商品
* @param $param
*/
public function addGoods($param)
{
if (!preg_match("/(pages\/goods\/detail\?sku_id=)(\d*)$/", $param[ 'url' ], $matches)) {
return $this->error('', '商品链接格式不正确');
}
$live = new Live($param[ 'site_id' ]);
if (is_url($param[ 'goods_pic' ])) {
$upload = new Upload($param[ 'site_id' ]);
$goods_pic = $upload->setPath("common/temp/" . date("Ymd") . '/')->remotePull($param[ 'goods_pic' ])[ 'data' ][ 'pic_path' ] ?? '';
$result = $live->addImageMedia($goods_pic);
} else {
$result = $live->addImageMedia($param[ 'goods_pic' ]);
}
if ($result[ 'code' ] < 0) return $result;
$audit = [
'goodsInfo' => [
'coverImgUrl' => $result[ 'data' ][ 'media_id' ],
'name' => $param[ 'name' ],
'priceType' => $param[ 'price_type' ],
'price' => $param[ 'price' ],
'price2' => $param[ 'price2' ],
'url' => $param[ 'url' ]
]
];
if ($param[ 'price_type' ] == 1) unset($audit[ 'goodsInfo' ][ 'price2' ]);
$result = $live->addGoodsAudit($audit);
if ($result[ 'code' ] < 0) return $result;
$data = [
'site_id' => $param[ 'site_id' ],
'goods_id' => $result[ 'data' ][ 'goodsId' ],
'name' => $param[ 'name' ],
'cover_img' => $param[ 'goods_pic' ],
'price' => $param[ 'price' ],
'status' => 1,
'url' => $param[ 'url' ],
'audit_id' => $result[ 'data' ][ 'auditId' ],
'sku_id' => $matches[ 2 ],
'third_party_tag' => 2
];
$result = model('weapp_goods')->add($data);
$cron_info = model("cron")->getInfo([ [ 'event', '=', 'LiveGoodsStatus' ], [ 'relate_id', '=', $param[ 'site_id' ] ] ]);
if (empty($cron_info)) {
$cron = new Cron();
$cron->addCron(2, 10, '小程序商品获取审核状态', 'LiveGoodsStatus', time(), $param[ 'site_id' ]);
}
return $this->success($result);
}
/**
* 删除商品
* @param $id
* @param $site_id
*/
public function deleteGoods($id, $site_id)
{
$info = model('weapp_goods')->getInfo([ [ 'site_id', '=', $site_id ], [ 'id', '=', $id ] ], 'goods_id');
if (empty($info)) return $this->error('', '未获取到商品信息');
$live = new Live($site_id);
$result = $live->deleteGoods($info[ 'goods_id' ]);
if ($result[ 'code' ] < 0) return $result;
$res = model('weapp_goods')->delete([ [ 'site_id', '=', $site_id ], [ 'id', '=', $id ] ]);
return $this->success($res);
}
/**
* 获取直播商品审核状态
* @param $id
*/
public function getGoodsAuditStatus($site_id)
{
$prefix = config("database")[ "connections" ][ "mysql" ][ "prefix" ];
$data = model('weapp_goods')->query("SELECT GROUP_CONCAT(goods_id) as goods_id FROM {$prefix}weapp_goods WHERE site_id = {$site_id} AND status = 1");
if (isset($data[ 0 ]) && isset($data[ 0 ][ 'goods_id' ]) && !empty($data[ 0 ][ 'goods_id' ])) {
$live = new Live($site_id);
$result = $live->getGoodsStatus(explode(',', $data[ 0 ][ 'goods_id' ]));
if ($result[ 'code' ] < 0) return $result;
foreach ($result[ 'data' ] as $item) {
if ($item[ 'audit_status' ] != 1) {
model('weapp_goods')->update([ 'status' => $item[ 'audit_status' ] ], [ [ 'site_id', '=', $site_id ], [ 'goods_id', '=', $item[ 'goods_id' ] ] ]);
}
}
}
}
}

298
addon/live/model/Live.php Executable file
View File

@@ -0,0 +1,298 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\model;
use addon\weapp\model\Config as WeappConfigModel;
use addon\wxoplatform\model\Config as WxOplatformConfigModel;
use app\model\BaseModel;
use EasyWeChat\Factory;
class Live extends BaseModel
{
private $app;
/**
* 微信直播间接口错误码
* @var array
*/
private $room_error = [
1003 => '商品id不存在',
47001 => '入参格式不符合规范',
200002 => '入参错误',
300001 => '禁止创建/更新商品 或 禁止编辑&更新房间',
300002 => '名称长度不符合规则',
300006 => '图片上传失败',
300022 => '此房间号不存在',
300023 => '房间状态 拦截(当前房间状态不允许此操作)',
300024 => '商品不存在',
300025 => '商品审核未通过',
300026 => '房间商品数量已经满额',
300027 => '导入商品失败',
300028 => '房间名称违规',
300029 => '主播昵称违规',
300030 => '主播微信号不合法',
300031 => '直播间封面图不合规',
300032 => '直播间分享图违规',
300033 => '添加商品超过直播间上限',
300034 => '主播微信昵称长度不符合要求',
300035 => '主播微信号不存在',
300036 => '主播微信号未实名认证',
];
/**
* 微信直播商品接口错误码
* @var array
*/
private $goods_error = [
300001 => '商品创建功能被封禁',
300002 => '名称长度不符合规则',
300003 => '价格输入不合规',
300004 => '商品名称存在违规违法内容',
300005 => '商品图片存在违规违法内容',
300006 => '图片上传失败',
300007 => '线上小程序版本不存在该链接',
300008 => '添加商品失败',
300009 => '商品审核撤回失败',
300010 => '商品审核状态不对',
300011 => 'API不允许操作非API创建的商品',
300012 => '没有提审额度每天500次提审额度',
300013 => '提审失败',
300014 => '审核中,无法删除',
300017 => '商品未提审',
300021 => '商品添加成功,审核失败',
];
public function __construct($site_id = 0)
{
//微信小程序配置
$weapp_config_model = new WeappConfigModel();
$weapp_config = $weapp_config_model->getWeappConfig($site_id)[ "data" ][ "value" ];
if (isset($weapp_config[ 'is_authopen' ]) && addon_is_exit('wxoplatform')) {
$plateform_config_model = new WxOplatformConfigModel();
$plateform_config = $plateform_config_model->getOplatformConfig()[ "data" ][ "value" ];
$config = [
'app_id' => $plateform_config[ "appid" ] ?? '',
'secret' => $plateform_config[ "secret" ] ?? '',
'token' => $plateform_config[ "token" ] ?? '',
'aes_key' => $plateform_config[ "aes_key" ] ?? '',
'log' => [
'level' => 'debug',
'permission' => 0777,
'file' => 'runtime/log/wechat/oplatform.logs',
],
];
$open_platform = Factory::openPlatform($config);
$this->app = $open_platform->miniProgram($weapp_config[ 'authorizer_appid' ], $weapp_config[ 'authorizer_refresh_token' ]);
} else {
$config = [
'app_id' => $weapp_config[ "appid" ] ?? '',
'secret' => $weapp_config[ "appsecret" ] ?? '',
'response_type' => 'array',
'log' => [
'level' => 'debug',
'permission' => 0777,
'file' => 'runtime/log/wechat/easywechat.logs',
],
];
$this->app = Factory::miniProgram($config);
}
}
/**
* 添加永久图片素材
* @param $path
* @return array
* @throws \GuzzleHttp\Exception\GuzzleException
*/
public function addImageMedia($path)
{
try {
$result = $this->app->media->uploadImage($path);
if (isset($result[ 'errcode' ])) {
return $this->error('', '图片上传失败');
} else {
return $this->success($result);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 获取直播间列表
* @param int $start
* @param int $limit
* @return array
* @throws \GuzzleHttp\Exception\GuzzleException
*/
public function getRoomList($start = 0, $limit = PAGE_LIST_ROWS)
{
try {
$result = $this->app->live->room->getLiveInfo([ 'start' => $start, 'limit' => $limit ]);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success([ 'list' => $result[ 'room_info' ], 'total' => $result[ 'total' ] ]);
} else {
return $this->error('', $this->room_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 获取回放视频
* @param $room_id
* @param int $start
* @param int $limit
* @return array
* @throws \GuzzleHttp\Exception\GuzzleException
*/
public function getPlaybackInfo($room_id, $start = 0, $limit = PAGE_LIST_ROWS)
{
try {
$data = [
'action' => 'get_replay',
'room_id' => $room_id,
'start' => $start,
'limit' => $limit
];
$result = $this->app->live->room->getLiveInfo($data);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success([ 'list' => $result[ 'live_replay' ], 'total' => $result[ 'total' ] ]);
} else {
return $this->error('', $this->room_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 创建直播间
* @param $param
*/
public function createRoom($param)
{
try {
$result = $this->app->live->room->create($param);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success($result);
} else {
return $this->error('', $this->room_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 给直播间添加商品
* @param int $room_id
* @param array $goods_ids
* @return array
* @throws \GuzzleHttp\Exception\GuzzleException
*/
public function roomAddGoods(int $room_id, array $goods_ids)
{
try {
$result = $this->app->live->room->addGoods([ 'roomId' => $room_id, 'ids' => $goods_ids ]);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success($result);
} else {
return $this->error('', $this->room_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 获取商品列表
* @param int $start
* @param int $limit
* @param int $status
* @return array
* @throws \GuzzleHttp\Exception\GuzzleException
*/
public function getGoodsList($start = 0, $limit = PAGE_LIST_ROWS, $status = 2)
{
try {
$result = $this->app->live->goods->getGoodsList([ 'offset' => $start, 'limit' => $limit, 'status' => $status ]);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success($result);
} else {
return $this->error('', $this->goods_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 添加商品审核
* @param $param
* @return array
*/
public function addGoodsAudit($param)
{
try {
$result = $this->app->live->goods->add($param);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success($result);
} else {
return $this->error('', $this->goods_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 删除商品
*/
public function deleteGoods($goods_id)
{
try {
$result = $this->app->live->goods->delete([ 'goodsId' => $goods_id ]);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success($result);
} else {
return $this->error('', $this->goods_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
/**
* 获取商品状态
* @param array $goods_ids
* @return array
* @throws \GuzzleHttp\Exception\GuzzleException
*/
public function getGoodsStatus(array $goods_ids)
{
try {
$result = $this->app->live->goods->getStatus([ 'goods_ids' => $goods_ids ]);
if (isset($result[ 'errcode' ]) && $result[ 'errcode' ] == 0) {
return $this->success($result[ 'goods' ]);
} else {
return $this->error('', $this->goods_error[ abs($result[ 'errcode' ]) ] ?? $result[ 'errmsg' ]);
}
} catch (\Exception $e) {
return $this->error('', $e->getMessage());
}
}
}

214
addon/live/model/Room.php Executable file
View File

@@ -0,0 +1,214 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\model;
use app\model\BaseModel;
use app\model\system\Cron;
use app\model\upload\Upload;
class Room extends BaseModel
{
private $liveStatus = [
'101' => '直播中',
'102' => '未开始',
'103' => '已结束',
'104' => '禁播',
'105' => '暂停中',
'106' => '异常',
'107' => '已过期',
];
/**
* 创建直播间
* @param $data
* @param $site_id
*/
public function createRoom($data, $site_id)
{
$live = new Live($site_id);
$res = $live->createRoom($data);
if ($res[ 'code' ] == 0) {
$this->syncLiveRoom(0, 1, $site_id);
$cron_info = model("cron")->getInfo([ [ 'event', '=', 'LiveRoomStatus' ], [ 'relate_id', '=', $site_id ] ]);
if (empty($cron_info)) {
$cron = new Cron();
$cron->addCron(2, 10, '轮询小程序直播状态', 'LiveRoomStatus', time(), $site_id);
}
}
return $res;
}
/**
* 编辑直播间信息
* @param array $data
* @param array $where
*/
public function updateRoomInfo($data = [], $where = [])
{
$res = model('weapp_live_room')->update($data, $where);
return $this->success($res);
}
/**
* 获取直播间列表
* @param array $condition
* @param bool $field
* @param string $order
* @param int $page
* @param int $list_rows
* @param string $alias
* @param array $join
* @return array
*/
public function getRoomPageList($condition = [], $field = true, $order = '', $page = 1, $list_rows = PAGE_LIST_ROWS, $alias = 'a', $join = [])
{
$data = model('weapp_live_room')->pageList($condition, $field, $order, $page, $list_rows, $alias, $join);
if (!empty($data[ 'list' ])) {
foreach ($data[ 'list' ] as $k => $item) {
$data[ 'list' ][ $k ][ 'status_name' ] = $this->liveStatus[ $item[ 'live_status' ] ] ?? '';
if (isset($item[ 'goods' ])) $data[ 'list' ][ $k ][ 'goods' ] = json_decode($item[ 'goods' ], true);
}
}
return $this->success($data);
}
/**
* 同步直播间列表
*/
public function syncLiveRoom($start, $limit, $site_id)
{
$live = new Live($site_id);
$result = $live->getRoomList($start, $limit);
if ($result[ 'code' ] < 0) return $result;
if (!empty($result[ 'data' ][ 'list' ])) {
$upload = new Upload($site_id, 'shop');
$upload->setPath('upload/live/room/');
foreach ($result[ 'data' ][ 'list' ] as $item) {
$data = [
'site_id' => $site_id,
'roomid' => $item[ 'roomid' ],
'name' => $item[ 'name' ],
'cover_img' => $item[ 'cover_img' ],
'start_time' => $item[ 'start_time' ],
'end_time' => $item[ 'end_time' ],
'anchor_name' => $item[ 'anchor_name' ],
'goods' => json_encode($item[ 'goods' ], JSON_UNESCAPED_UNICODE),
'live_status' => $item[ 'live_status' ],
];
$room_info = model('weapp_live_room')->getInfo([ [ 'roomid', '=', $item[ 'roomid' ] ], [ 'site_id', '=', $site_id ] ], 'id');
if (empty($room_info)) {
if (is_url($item[ 'share_img' ])) {
$pull_result = $upload->remotePull($item[ 'share_img' ]);
$pull_result = $pull_result[ 'data' ];
if (isset($pull_result[ 'pic_path' ]) && !empty($pull_result[ 'pic_path' ])) {
$data[ 'share_img' ] = $pull_result[ 'pic_path' ];
}
}
model('weapp_live_room')->add($data);
} else {
$data = [
// 'goods' => json_encode($item[ 'goods' ], JSON_UNESCAPED_UNICODE),
'live_status' => $item[ 'live_status' ],
];
model('weapp_live_room')->update($data, [ [ 'id', '=', $room_info[ 'id' ] ] ]);
}
}
$total_page = ceil($result[ 'data' ][ 'total' ] / $limit);
return $this->success([ 'page' => $start, 'total_page' => $total_page ]);
} else {
return $this->success([ 'page' => $start, 'total_page' => 1 ]);
}
}
/**
* 获取直播间信息
*/
public function getRoomInfo($condition = [], $field = '*')
{
$data = model('weapp_live_room')->getInfo($condition, $field);
if (!empty($data)) {
$data[ 'status_name' ] = $this->liveStatus[ $data[ 'live_status' ] ] ?? '';
if (isset($data[ 'goods' ]) && !empty($data[ 'goods' ])) $data[ 'goods' ] = json_decode($data[ 'goods' ], true);
}
return $this->success($data);
}
/**
* 添加商品到直播间
* @param $site_id
* @param $room_id
* @param $data
*/
public function addGoods($site_id, $room_id, $data)
{
if (empty($data)) return $this->error('', '请先选择要添加的商品');
$room_info = model('weapp_live_room')->getInfo([ [ 'site_id', '=', $site_id ], [ 'roomid', '=', $room_id ] ], 'goods');
if (empty($room_info)) return $this->error('', '未查找到直播间信息');
$data = json_decode($data, true);
$goods_ids = [];
$goods_data = [];
foreach ($data as $item) {
$goods_ids[] = $item['goods_id'];
$goods_data[] = [
'name' => $item['name'],
'cover_img' => $item['cover_img'],
'url' => $item['url'],
'price' => $item['price']
];
}
$live = new Live($site_id);
$result = $live->roomAddGoods($room_id, $goods_ids);
if ($result[ 'code' ] < 0) return $result;
if (!empty($room_info[ 'goods' ])) {
$room_goods = json_decode($room_info[ 'goods' ], true);
$goods_data = array_merge($room_goods, $goods_data);
}
$res = model('weapp_live_room')->update([ 'goods' => json_encode($goods_data, JSON_UNESCAPED_UNICODE) ], [ [ 'site_id', '=', $site_id ], [ 'roomid', '=', $room_id ] ]);
return $this->success($res);
}
/**
* 轮询更新直播间状态
* @param $site_id
*/
public function updateRoomStatus($site_id)
{
$count = model('weapp_live_room')->getCount([ [ 'site_id', '=', $site_id ], [ 'live_status', 'in', [ '101', '102', '105' ] ] ]);
if ($count) {
$start = 0;
$result = $this->syncLiveRoom($start, 20, $site_id);
if (isset($result[ 'code' ]) && $result[ 'code' ] == 0 && $result[ 'total_page' ] > 1) {
for ($i = 1; $i < $result[ 'data' ]; $i++) {
$this->syncLiveRoom($i, 20, $site_id);
}
}
}
}
/**
* 删除直播间
* @param $site_id
* @param $room_ids
*/
public function deleteRoom($site_id, $room_ids)
{
$res = model('weapp_live_room')->delete([ [ 'site_id', '=', $site_id ], [ 'id', 'in', $room_ids ] ]);
return $this->success($res);
}
}

View File

@@ -0,0 +1,87 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\shop\controller;
use app\shop\controller\BaseShop;
use addon\live\model\Goods as GoodsModel;
/**
* 直播间
*/
class Goods extends BaseShop
{
public function index()
{
if (request()->isJson()) {
$goods = new GoodsModel();
$page = input('page', 1);
$page_size = input('page_size', PAGE_LIST_ROWS);
$condition = [
'site_id' => $this->site_id
];
$data = $goods->getGoodsPageList($condition, '*', 'id desc', $page, $page_size);
return $data;
} else {
return $this->fetch('goods/index');
}
}
/**
* 同步直播商品库
*/
public function sync()
{
if (request()->isJson()) {
$goods = new GoodsModel();
$start = input('start', 0);
$res = $goods->syncGoods($start, 20, $this->site_id);
return $res;
}
}
/**
* 添加商品
* @return mixed
*/
public function add()
{
if (request()->isJson()) {
$goods = new GoodsModel();
$data = [
'site_id' => $this->site_id,
'name' => input('name', ''),
'goods_pic' => input('goods_pic', ''),
'price_type' => input('price_type', ''),
'price' => input('price', ''),
'price2' => input('price2', ''),
'url' => input('url', ''),
];
$res = $goods->addGoods($data);
return $res;
}
return $this->fetch('goods/add');
}
/**
* 删除商品
*/
public function delete()
{
if (request()->isJson()) {
$id = input('id', '');
$goods = new GoodsModel();
$res = $goods->deleteGoods($id, $this->site_id);
return $res;
}
}
}

View File

@@ -0,0 +1,200 @@
<?php
/**
* Niushop商城系统 - 团队十年电商经验汇集巨献!
* =========================================================
* Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
* ----------------------------------------------
* 官方网址: https://www.niushop.com
* =========================================================
*/
namespace addon\live\shop\controller;
use addon\live\model\Live;
use app\model\upload\Upload;
use app\shop\controller\BaseShop;
use addon\live\model\Room as RoomModel;
use addon\live\model\Goods as GoodsModel;
use think\App;
/**
* 直播间
*/
class Room extends BaseShop
{
public function __construct(App $app = null)
{
$this->replace = [
'LIVE_IMG' => __ROOT__ . '/addon/live/shop/view/public/img',
];
parent::__construct($app);
}
/**
* 直播间列表
* @return array|mixed
*/
public function index()
{
if (request()->isJson()) {
$room = new RoomModel();
$page = input('page', 1);
$page_size = input('page_size', PAGE_LIST_ROWS);
$condition = [
'site_id' => $this->site_id
];
$data = $room->getRoomPageList($condition, '*', 'roomid desc', $page, $page_size);
return $data;
} else {
return $this->fetch("room/index");
}
}
/**
* 同步直播间
* @return array
*/
public function sync()
{
if (request()->isJson()) {
$room = new RoomModel();
$start = input('start', 0);
$res = $room->syncLiveRoom($start, 20, $this->site_id);
return $res;
}
}
/**
* 添加直播间
*/
public function add()
{
if (request()->isJson()) {
$room = new RoomModel();
$data = [
'name' => input('name', ''),
'coverImg' => input('coverImg', ''),
'startTime' => strtotime(input('startTime', '')),
'endTime' => strtotime(input('endTime', '')),
'anchorName' => input('anchorName', ''),
'anchorWechat' => input('anchorWechat', ''),
'shareImg' => input('shareImg', ''),
'feedsImg' => input('feedsImg', ''),
'type' => input('type', 0),
'screenType' => 0,
'closeLike' => input('closeLike', 1),
'closeGoods' => input('closeGoods', 1),
'closeComment' => input('closeComment', 1),
'closeReplay' => input('closeReplay', 1),
'closeKf' => input('closeKf', 1)
];
$res = $room->createRoom($data, $this->site_id);
return $res;
}
return $this->fetch("room/add");
}
/**
* 添加图片素材
*/
public function addImageMedia()
{
if (request()->isJson()) {
$upload_model = new Upload($this->site_id, $this->app_module);
$thumb_type = input("thumb", "");
$name = input("name", "");
$param = array (
"thumb_type" => "",
"name" => "file",
"watermark" => 0,
"cloud" => 0
);
$path = "common/images/" . date("Ymd") . '/';
$result = $upload_model->setPath($path)->image($param);
if ($result[ 'code' ] < 0) return $result;
$live = new Live($this->site_id);
$media_result = $live->addImageMedia($result[ 'data' ][ 'pic_path' ]);
if ($media_result[ 'code' ] < 0) return $media_result;
return success(0, '上传成功', [ 'pic_info' => $result[ 'data' ], 'media_info' => $media_result[ 'data' ] ]);
}
}
/**
* 运营
*/
public function operate()
{
$room = new RoomModel();
if (request()->isJson()) {
$id = input('id', '');
$anchor_img = input('anchor_img', '-1');
$banner = input('banner', '-1');
$data = [];
if ($anchor_img != '-1') $data = [ 'anchor_img' => $anchor_img ];
if ($banner != '-1') $data = [ 'banner' => $banner ];
$res = $room->updateRoomInfo($data, [ [ 'site_id', '=', $this->site_id ], [ 'id', '=', $id ] ]);
return $res;
}
$id = input('id', '');
$room_info = $room->getRoomInfo([ [ 'site_id', '=', $this->site_id ], [ 'id', '=', $id ] ]);
if (empty($room_info[ 'data' ])) $this->error('未获取到直播间信息');
$this->assign('room_info', $room_info[ 'data' ]);
return $this->fetch("room/operate");
}
/**
* 查询商品
*/
public function getGoodsPageList()
{
if (request()->isJson()) {
$goods = new GoodsModel();
$page = input('page', 1);
$page_size = input('page_size', PAGE_LIST_ROWS);
$sku_id = input('sku_id', '');
$condition = [
[ 'site_id', '=', $this->site_id ],
[ 'status', '=', 2 ]
];
if (!empty($sku_id)) $condition[] = [ 'sku_id', 'not in', explode(',', $sku_id) ];
$data = $goods->getGoodsPageList($condition, '*', 'id desc', $page, $page_size);
return $data;
}
$ids = input('ids', '');
$this->assign('ids', $ids);
return $this->fetch("room/goods_select");
}
/**
* 添加商品到直播间
* @return array
*/
public function addGoods()
{
if (request()->isJson()) {
$room = new RoomModel();
$room_id = input('room_id', '');
$data = input('data', '');
$res = $room->addGoods($this->site_id, $room_id, $data);
return $res;
}
}
/**
* 删除直播间
* @return mixed
*/
public function delete()
{
if (request()->isJson()) {
$room = new RoomModel();
$room_ids = input('room_ids', '');
$res = $room->deleteRoom($this->site_id, $room_ids);
return $res;
}
}
}

View File

@@ -0,0 +1,246 @@
<style type="text/css">
.price-wrap{display: flex;top: 4px;position: relative;}
.price-wrap > span {display: inline-block;margin: 0 5px;}
.price-wrap > span.sepa{margin: 0 20px;}
</style>
<div class="layui-form form-wrap">
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" onclick="addGoods()">选择商品</button>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>商品名称:</label>
<div class="layui-input-block">
<input type="text" name="name" lay-verify="required|name" class="layui-input ns-len-mid" autocomplete="off" maxlength="14">
</div>
<div class="word-aux">
<!-- <p>商品名称必须为3-14个字符</p> -->
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>商品封面:</label>
<div class="layui-input-block">
<div class="upload-img-block">
<div class="upload-img-box ">
<div class="upload-default" id="ImgUpload">
<div class="upload">
<i class="iconfont iconshangchuan"></i>
<p>选择商品</p>
</div>
</div>
<div class="operation">
<div>
<i title="图片预览" class="iconfont iconreview js-preview" style="margin-right: 20px;"></i>
<i title="删除图片" class="layui-icon layui-icon-delete js-delete"></i>
</div>
<div class="replace_img js-replace">点击替换</div>
</div>
<input type="hidden" name="goods_pic" value="" lay-verify="goodsPic"/>
</div>
<!-- <p id="ImgUpload" class="no-replace">替换</p>
<input type="hidden" name="goods_pic" value="" lay-verify="goodsPic"/>
<i class="del">x</i> -->
</div>
</div>
<div class="word-aux">
<p>建议尺寸200像素 * 200像素图片大小不得超过80K</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>价格形式:</label>
<div class="layui-input-block">
<div class="layui-input-inline">
<input type="radio" name="price_type" value="1" title="一口价" checked>
</div>
<div class="layui-input-inline price-wrap">
<span>价格</span>
<input class="layui-input len-short input-num" type="number" min="0" name="price" value="0.00" lay-verify="required|flnum">
<span></span>
</div>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<div class="layui-input-inline">
<input type="radio" name="price_type" value="2" title="价格区间">
</div>
<div class="layui-input-inline price-wrap">
<span>价格</span>
<input class="layui-input len-short input-num" type="number" min="0" name="price" value="0.00" lay-verify="required|flnum">
<span></span>
<span class="sepa">-</span>
<input class="layui-input len-short input-num" type="number" min="0" name="price2" value="0.00" lay-verify="required|flnum">
<span></span>
</div>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<div class="layui-input-inline">
<input type="radio" name="price_type" value="3" title="显示折扣价">
</div>
<div class="layui-input-inline price-wrap">
<span>原价</span>
<input class="layui-input len-short input-num" type="number" min="0" name="price" value="0.00" lay-verify="required|flnum">
<span></span>
<span class="sepa"></span>
<span>现价</span>
<input class="layui-input len-short input-num" type="number" min="0" name="price2" value="0.00" lay-verify="required|flnum">
<span></span>
</div>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>商品链接:</label>
<div class="layui-input-block">
<input type="text" name="url" lay-verify="required" class="layui-input len-mid" autocomplete="off">
</div>
<div class="word-aux">
<p>请确保小程序页面路径可被访问例如pages/goods/detail?sku_id=sku_id</p>
</div>
</div>
<div class="form-row">
<button class="layui-btn" lay-submit lay-filter="save">添加</button>
<button class="layui-btn layui-btn-primary" onclick="backLiveGoodsList()">返回</button>
</div>
</div>
<script type="text/javascript">
var form,repeat_flag=false;
layui.use(['form'], function() {
form = layui.form;
form.render();
var upload = new Upload({
elem: '#ImgUpload',
multiple: true,
});
/**
* 表单验证
*/
form.verify({
name: function(value){
if (value.length < 3 || value.length > 14) {
return '商品名称必须为3-14个字符';
}
},
goodsPic: function(value){
var patt = /[\S]+/;
if (!patt.test(value)) {
return '请上传商品封面!';
}
},
});
form.on('submit(save)', function(data){
if (repeat_flag) return;
repeat_flag = true;
if (data.field.price_type == 1) {
data.field.price = parseFloat($('[name="price_type"][value="1"]').parents('.layui-input-block').find('[name="price"]').val());
data.field.price2 = 0;
if (data.field.price <= 0) {
layer.msg('请输入商品价格', {icon: 5, shift: 6});
return;
}
} else if (data.field.price_type == 2) {
data.field.price = parseFloat($('[name="price_type"][value="2"]').parents('.layui-input-block').find('[name="price"]').val());
data.field.price2 = parseFloat($('[name="price_type"][value="2"]').parents('.layui-input-block').find('[name="price2"]').val());
if (data.field.price <= 0 || data.field.price2 <= 0) {
layer.msg('请输入商品价格', {icon: 5, shift: 6});
return;
}
} else if (data.field.price_type == 3) {
data.field.price = parseFloat($('[name="price_type"][value="3"]').parents('.layui-input-block').find('[name="price"]').val());
data.field.price2 = parseFloat($('[name="price_type"][value="3"]').parents('.layui-input-block').find('[name="price2"]').val());
if (data.field.price <= 0) {
layer.msg('请输入原价', {icon: 5, shift: 6});
return;
}
if (data.field.price2 <= 0) {
layer.msg('请输入现价', {icon: 5, shift: 6});
return;
}
if (data.field.price2 >= data.field.price) {
layer.msg('现价不能小于或等于原价', {icon: 5, shift: 6});
return;
}
}
$.ajax({
type: 'POST',
url: ns.url("live://shop/goods/add"),
data: data.field,
dataType: 'JSON',
success: function(res) {
layer.msg(res.message);
repeat_flag = false;
if (res.code == 0) {
location.hash = ns.hash("live://shop/goods/index");
}
}
});
});
});
function backLiveGoodsList() {
location.hash = ns.hash("live://shop/goods/index");
}
/**
* 添加商品
*/
function addGoods() {
goodsSelect(function (data) {
if (Object.keys(data).length == 0) {
$('[name="name"]').val('');
$("input[name='goods_pic']").val('');
$("#ImgUpload").html('<div class="upload"><i class="iconfont iconshangchuan"></i><p>选择商品</p></div>');
$('#ImgUpload').parents('.upload-img-box').removeClass('hover');
$('[name="url"]').val('');
$('[name="price_type"][value="1"]').prop('checked', true);
$('[name="price_type"][value="1"]').parents('.layui-input-block').find('[name="price"]').val('0.00');
return;
}
data = data[Object.keys(data)[0]];
var sku_id = data.sku_list[0].sku_id;
var images = data.goods_image.split(',');
$('[name="name"]').val(data.goods_name);
if (images[0] != undefined) {
$("input[name='goods_pic']").val(imgHandle(images[0]));
$("#ImgUpload").html("<div id='preview_imgUpload' class='preview_img'><img layer-src class='img_prev' src=" + ns.img(images[0]) + " ></div>");
// $("#ImgUpload").html("<img src=" + ns.img(images[0]) + " >");
// $("#ImgUpload").removeClass("no-replace").addClass("replace");
loadImgMagnify();
$('#ImgUpload').parents('.upload-img-box').addClass('hover')
}
$('[name="url"]').val('pages/goods/detail?sku_id=' + sku_id);
$('[name="price_type"][value="1"]').prop('checked', true);
$('[name="price_type"][value="1"]').parents('.layui-input-block').find('[name="price"]').val(data.price);
form.render();
}, [], {mode: "spu", max_num: 1});
}
function imgHandle(src){
var arr = src.split("."),
suffix = arr[arr.length - 1];
arr.pop();
arr[arr.length - 1] = arr[arr.length - 1] + "_SMALL";
arr.push(suffix);
src = arr.join(".");
return src;
}
</script>

View File

@@ -0,0 +1,159 @@
<style>
.progress-layer {width:400px;background:#fff;position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);box-shadow:1px 1px 50px rgba(0,0,0,.3);padding:20px 20px;z-index:100;display:none;}
.progress-layer h3 {line-height:1;margin-bottom:15px;text-align:center;font-size:14px;}
.progress-layer .layui-progress-big,.progress-layer .layui-progress-big .layui-progress-bar {height:14px;line-height:14px;}
.progress-layer .layui-progress-text {line-height:14px;}
.goods-info {padding: 5px 0;align-items: center;flex-wrap:unset!important;}
.goods-info .room-name {padding-left: 5px;line-height: 1}
.goods-info img {width:50px;height: 50px;}
.single-filter-box{justify-content: end}
</style>
<div class="single-filter-box">
<button class="layui-btn" onclick="sync()">同步商品库</button>
<a href="{:href_url('live://shop/goods/add')}" class="layui-btn layui-btn-primary">添加商品</a>
</div>
<table id="goods_list" lay-filter="goods_list"></table>
<!-- 直播间信息 -->
<script type="text/html" id="goodsinfo">
<div class="layui-table-cell goods-info">
<img src="{{ ns.img(d.cover_img) }}">
<span class="room-name" title="{{ d.name }}">{{ d.name }}</span>
</div>
</script>
<!-- 操作 -->
<script type="text/html" id="operation">
<div class="table-btn">
{{# if( (d.status == 2 || d.status == 3) && d.third_party_tag != 0 ){ }}
<a class="layui-btn" lay-event="del">删除</a>
{{# } }}
</div>
</script>
<div class="progress-layer">
<h3>正在同步中...</h3>
<div class="layui-progress layui-progress-big" lay-showPercent="true" lay-filter="progress">
<div class="layui-progress-bar layui-bg-blue" lay-percent="0%"></div>
</div>
</div>
<script>
var form,table,element,syncClick = false,delete_flag = false;
layui.use(['form', 'element'], function() {
form = layui.form;
element = layui.element;
table = new Table({
elem: '#goods_list',
url: ns.url("live://shop/goods/index"),
cols: [
[{
title: '商品信息',
unresize: 'false',
width: '30%',
templet: "#goodsinfo"
}, {
title: '价格',
unresize: 'false',
width: '15%',
field: 'price'
}, {
field: 'url',
title: '商品链接',
unresize: 'false',
width: '30%'
}, {
field: 'status_name',
title: '状态',
unresize: 'false',
width: '15%',
}, {
title: '操作',
toolbar: '#operation',
unresize: 'false',
align:'right'
}]
]
});
table.tool(function(obj) {
var data = obj.data;
switch (obj.event) {
case 'del': //删除
layer.confirm('确定要删除该商品吗?', function(index) {
if (delete_flag) return false;
delete_flag = true;
layer.close(index);
$.ajax({
url: ns.url("live://shop/goods/delete"),
data: {
id: data.id
},
dataType: 'JSON',
type: 'POST',
success: function(res) {
layer.msg(res.message);
delete_flag = false;
if (res.code == 0) {
table.reload({
page: {
curr: 1
},
});
}
}
});
}, function() {
layer.close();
delete_flag = false;
});
break;
}
})
});
// 同步商品
function sync(start){
if (syncClick) return;
syncClick = true;
var start = start == undefined ? 0 : start;
$.ajax({
url: ns.url("live://shop/goods/sync"),
data: {
start: 0,
},
dataType: 'JSON',
type: 'POST',
success: function(res) {
syncClick = false;
if (res.code == 0) {
var data = res.data,
next = parseInt(start) + 1;
if (next < data.total_page) {
if (start == 0) {
$(".progress-layer").fadeOut();
}
var progress = (next / data.total_page * 100).toFixed(2);
element.progress('progress', progress + '%');
// 拉取下一页
sync(next);
} else {
if (!$(".progress-layer").is(':hidden')) $(".progress-layer").fadeOut();
layer.closeAll();
layer.msg('同步成功');
table.reload();
}
} else {
layer.msg(res.message);
}
}
});
}
</script>

Binary file not shown.

After

Width:  |  Height:  |  Size: 402 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 92 KiB

View File

@@ -0,0 +1,329 @@
<div class="layui-form form-wrap">
<div class="layui-card card-common card-brief">
<div class="layui-card-header">
<span class="card-title">基础设置</span>
</div>
<div class="layui-card-body">
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>直播类型:</label>
<div class="layui-input-block">
<input type="radio" name="type" value="0" title="手机直播" checked>
</div>
<div class="word-aux">
<p>通过“小程序直播主播端小程序”发起直播</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>直播标题:</label>
<div class="layui-input-block">
<input type="text" name="name" lay-verify="required|name" class="layui-input len-mid" autocomplete="off">
</div>
<div class="word-aux">
<p>直播标题必须为3-17个字符</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>开播时间:</label>
<div class="layui-input-block len-mid">
<input type="text" class="layui-input" name="startTime" lay-verify="required" id="startTime" autocomplete="off" readonly>
<i class=" iconrili iconfont calendar"></i>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>结束时间:</label>
<div class="layui-input-block len-mid end_time">
<input type="text" class="layui-input" name="endTime" lay-verify="required|time" id="endTime" autocomplete="off" readonly>
<i class=" iconrili iconfont calendar"></i>
</div>
<div class="word-aux">
<p>开播时间至少需在10分钟后</p>
<p>开播时间段仅供参考,不是实际直播间可以开播的时间。</p>
<p>直播间在开始时间前也可以开播,并且到结束时间后不会强制结束。</p>
<p>若到结束时间仍未开播,则直播间无法再开播。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>主播昵称:</label>
<div class="layui-input-block len-mid">
<input type="text" class="layui-input" name="anchorName" lay-verify="required|anchorName" autocomplete="off">
</div>
<div class="word-aux">
<p>主播昵称必须为2-15个字符</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>主播微信账号:</label>
<div class="layui-input-block len-mid">
<input type="text" class="layui-input" name="anchorWechat" lay-verify="required" autocomplete="off">
</div>
<div class="word-aux">
<p>每个直播间需要绑定一个用作核实主播身份</p>
<p>主播微信号,需通过实名认证</p>
</div>
</div>
</div>
</div>
<div class="layui-card card-common card-brief">
<div class="layui-card-header">
<span class="card-title">分享卡片样式配置</span>
</div>
<div class="layui-card-body">
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>分享卡片封面:</label>
<div class="layui-input-block">
<div class="upload-img-block">
<div class="upload-img-box">
<div class="upload-default" id="shareImgUpload">
<div class="upload">
<i class="iconfont iconshangchuan"></i>
<p>点击上传</p>
</div>
</div>
<div class="operation">
<div>
<i title="图片预览" class="iconfont iconreview js-preview" style="margin-right: 20px;"></i>
<i title="删除图片" class="layui-icon layui-icon-delete js-delete"></i>
</div>
<div class="replace_img js-replace">点击替换</div>
</div>
<input type="hidden" name="shareImg" value="" lay-verify="shareImg"/>
</div>
</div>
</div>
<div class="word-aux">
<p>观众在微信对话框内分享的直播间将以分享卡片的形式呈现。建议尺寸800像素 * 640像素图片大小不得超过1M。</p>
</div>
</div>
</div>
</div>
<div class="layui-card card-common card-brief">
<div class="layui-card-header">
<span class="card-title">直播间样式配置</span>
</div>
<div class="layui-card-body">
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>直播间背景墙:</label>
<div class="layui-input-block">
<div class="upload-img-block">
<div class="upload-img-box">
<div class="upload-default" id="coverImgUpload">
<div class="upload">
<i class="iconfont iconshangchuan"></i>
<p>点击上传</p>
</div>
</div>
<div class="operation">
<div>
<i title="图片预览" class="iconfont iconreview js-preview" style="margin-right: 20px;"></i>
<i title="删除图片" class="layui-icon layui-icon-delete js-delete"></i>
</div>
<div class="replace_img js-replace">点击替换</div>
</div>
<input type="hidden" name="coverImg" value="" lay-verify="coverImg"/>
</div>
</div>
</div>
<div class="word-aux">
<p>直播间背景墙是每个直播间的默认背景。建议尺寸1080像素 * 1920像素图片大小不得超过2M。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>购物直播频道封面:</label>
<div class="layui-input-block">
<div class="upload-img-block">
<div class="upload-img-box">
<div class="upload-default" id="feedsImgUpload">
<div class="upload">
<i class="iconfont iconshangchuan"></i>
<p>点击上传</p>
</div>
</div>
<div class="operation">
<div>
<i title="图片预览" class="iconfont iconreview js-preview" style="margin-right: 20px;"></i>
<i title="删除图片" class="layui-icon layui-icon-delete js-delete"></i>
</div>
<div class="replace_img js-replace">点击替换</div>
</div>
<input type="hidden" name="feedsImg" value="" lay-verify="feedsImg"/>
</div>
</div>
</div>
<div class="word-aux">
<p>购物直播频道封面图。建议尺寸建议像素800*800大小不超过100KB。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">直播间功能:</label>
<div class="layui-input-block">
<input type="checkbox" name="closeLike" value="0" title="开启点赞" lay-skin="primary" checked><br>
<input type="checkbox" name="closeGoods" value="0" title="开启货架" lay-skin="primary" checked><br>
<input type="checkbox" name="closeComment" value="0" title="开启评论" lay-skin="primary" checked><br>
<input type="checkbox" name="closeReplay" value="0" title="开启回放" lay-skin="primary" checked><br>
<input type="checkbox" name="closeKf" value="0" title="开启客服" lay-skin="primary" checked>
</div>
</div>
</div>
</div>
<div class="form-row">
<button class="layui-btn" lay-submit lay-filter="save">添加</button>
<button class="layui-btn layui-btn-primary" onclick="backLiveRoom()">返回</button>
</div>
</div>
<script type="text/javascript">
var form, laydate, repeat_flag = false, //防重复标识
minDate = "";
layui.use(['form', 'laydate'], function() {
form = layui.form,
laydate = layui.laydate;
form.render();
// 开始时间
laydate.render({
elem: '#startTime' ,//指定元素
type: 'datetime',
value: new Date(Date.parse(new Date()) + 3600000),
done: function(value){
minDate = value;
reRender();
}
});
//结束时间
laydate.render({
elem: '#endTime' ,//指定元素
type: 'datetime',
value: ''
});
/**
* 重新渲染结束时间
* */
function reRender(){
$("#endTime").remove();
$(".end_time").html('<input type="text" id="endTime" name="endTime" placeholder="请输入结束时间" lay-verify="required|time" class="layui-input len-mid" autocomplete="off">');
laydate.render({
elem: '#endTime',
type: 'datetime',
min: minDate
});
}
/**
* 表单验证
*/
form.verify({
name: function(value){
if (value.length < 3 || value.length > 17) {
return '直播标题必须为3-17个字符';
}
},
anchorName: function(value){
if (value.length < 2 || value.length > 15) {
return '主播昵称必须为2-15个字符';
}
},
time: function(value) {
var now_time = (new Date()).getTime();
var start_time = (new Date($("#start_time").val())).getTime();
var end_time = (new Date(value)).getTime();
if (now_time > end_time) {
return '结束时间不能小于当前时间!'
}
if (start_time > end_time) {
return '结束时间不能小于开始时间!';
}
},
shareImg: function(value){
var patt = /[\S]+/;
if (!patt.test(value)) {
return '请上传分享卡片封面!';
}
},
coverImg: function(value){
var patt = /[\S]+/;
if (!patt.test(value)) {
return '请上传直播间背景墙!';
}
},
feedsImg: function(value){
var patt = /[\S]+/;
if (!patt.test(value)) {
return '请上传直播间购物直播频道封面!';
}
}
});
// 添加
form.on('submit(save)', function(data){
if (repeat_flag) return;
repeat_flag = true;
$.ajax({
type: 'POST',
url: ns.url("live://shop/room/add"),
data: data.field,
dataType: 'JSON',
success: function(res) {
layer.msg(res.message);
repeat_flag = false;
if (res.code == 0) {
location.hash = ns.hash("live://shop/room/index");
}
}
});
});
var share_upload = new Upload({
elem: '#shareImgUpload',
url: ns.url("live://shop/room/addimagemedia"),
callback: function(res) {
if (res.code >= 0) {
$("input[name='shareImg']").val(res.data.media_info.media_id);
// $("#shareImgUpload").parent().find(".upload-img-box").html("<img src=" + ns.img(res.data.pic_info.pic_path) + " >");
$("#shareImgUpload").html("<div id='preview_imgUpload' class='preview_img'><img layer-src class='img_prev' src=" + ns.img(res.data.pic_info.pic_path) + " ></div>");
}
return layer.msg(res.message);
}
});
var cover_upload = new Upload({
elem: '#coverImgUpload',
url: ns.url("live://shop/room/addimagemedia"),
callback: function(res) {
if (res.code >= 0) {
$("input[name='coverImg']").val(res.data.media_info.media_id);
$("#coverImgUpload").html("<div id='preview_imgUpload' class='preview_img'><img layer-src class='img_prev' src=" + ns.img(res.data.pic_info.pic_path) + " ></div>");
}
return layer.msg(res.message);
}
});
var feeds_upload = new Upload({
elem: '#feedsImgUpload',
url: ns.url("live://shop/room/addimagemedia"),
callback: function(res) {
if (res.code >= 0) {
$("input[name='feedsImg']").val(res.data.media_info.media_id);
$("#feedsImgUpload").html("<div id='preview_imgUpload' class='preview_img'><img layer-src class='img_prev' src=" + ns.img(res.data.pic_info.pic_path) + " ></div>");
}
return layer.msg(res.message);
}
});
});
function backLiveRoom() {
location.hash = ns.hash("live://shop/room/index");
}
</script>

View File

@@ -0,0 +1,93 @@
<style type="text/css">
.table-wrap {padding: 0 20px;width: 560px;height: 385px;}
.table-wrap .layui-table-body {overflow-x: hidden;height: 260px;overflow-y: scroll;}
.goods-info {padding: 5px 0;align-items: center;flex-wrap:unset!important;}
.goods-info .room-name {padding-left: 5px;line-height: 1}
.goods-info img {width:50px;height: 50px;}
.info-wrap{height: 30px;line-height: 30px;background: #f2f2f2;color: #666;padding: 5px 10px;margin-top: 15px;}
</style>
<div class="table-wrap">
<div class="info-wrap">已选择 <span id="selectNum">0</span> 件 商品</div>
<table id="goods_list" lay-filter="goods_list"></table>
</div>
<script type="text/html" id="goodsinfo">
<div class="goods-info">
<img src="{{ ns.img(d.cover_img) }}">
<span class="room-name" title="{{ d.name }}">{{ d.name }}</span>
</div>
</script>
<script type="text/javascript">
var sku_id = '{$ids}',
selectedData = [];
selectedIdData = [];
$(function(){
var table = new Table({
id: 'goodsTable',
elem: '#goods_list',
url: ns.url("live://shop/room/getGoodsPageList"),
where: {
sku_id: sku_id
},
cols: [
[
{
title: '',
type: 'checkbox',
unresize: 'false',
width: '10%',
},
{
title: '商品信息',
unresize: 'false',
width: '90%',
templet: "#goodsinfo"
}
]
],
callback: function(res, curr, count){
if (res.data.length) {
var checkedNum = 0;
res.data.forEach(function(e,i){
if ($.inArray(e.id, selectedIdData) != -1) {
checkedNum += 1;
res.data[i]["LAY_CHECKED"] = 'true';
var index= res.data[i]['LAY_TABLE_INDEX'];
     $('.table-wrap tr[data-index=' + index + '] input[type="checkbox"]').prop('checked', true);
     $('.table-wrap tr[data-index=' + index + '] input[type="checkbox"]').next().addClass('layui-form-checked');
}
});
if (res.data.length == checkedNum) {
$('.table-wrap .layui-table-header tr input[type="checkbox"]').prop('checked', true);
     $('.table-wrap .layui-table-header tr input[type="checkbox"]').next().addClass('layui-form-checked');
}
}
}
});
// 监听checkbox选中状态改变事件
table.on('checkboxChange', function(obj){
if (obj.checked) {
if ($.inArray(obj.data.id, selectedIdData) == -1) {
selectedIdData.push(obj.data.id);
selectedData.push(obj.data)
}
} else {
selectedData.forEach(function(e,i){
if (JSON.stringify(e) == JSON.stringify(obj.data)) {
selectedData.splice(i, 1);
}
});
selectedIdData.splice($.inArray(obj.data.id, selectedIdData), 1);
}
$('#selectNum').text(selectedData.length);
})
});
function liveSelectGoodsCallbackListener(fun){
if (typeof fun == 'function') fun(selectedData);
}
</script>

View File

@@ -0,0 +1,230 @@
<style>
.progress-layer {width:400px;background:#fff;position:fixed;top:50%;left:50%;transform:translate(-50%,-50%);box-shadow:1px 1px 50px rgba(0,0,0,.3);padding:20px 20px;z-index:100;display:none;}
.progress-layer h3 {line-height:1;margin-bottom:15px;text-align:center;font-size:14px;}
.progress-layer .layui-progress-big,.progress-layer .layui-progress-big .layui-progress-bar {height:14px;line-height:14px;}
.progress-layer .layui-progress-text {line-height:14px;}
.room-info {padding: 5px 0;align-items: center;flex-wrap:unset!important;}
.room-info .room-img{width:50px;height: 50px;background: #f5f5f5;}
.room-info img {width:50px;height: 50px;}
.room-info .room-name {padding-left: 5px;line-height: 1;flex: 1}
.single-filter-box{justify-content: end}
.push-layer{text-align: center;padding: 30px 0;}
.push-layer .qrcode{width:100px;height:100px;margin:10px auto;display:block}
</style>
<div class="single-filter-box">
<button class="layui-btn" onclick="sync()">同步直播间</button>
<a href="{:href_url('live://shop/room/add')}" class="layui-btn layui-btn-primary">添加直播间</a>
</div>
<table id="room_list" lay-filter="room_list"></table>
<!-- 直播间信息 -->
<script type="text/html" id="roominfo">
<div class="table-btn room-info">
<div class="room-img">
<img src="{{ ns.img(d.share_img) }}">
</div>
<div class="room-name" title="{{ d.name }}">{{ d.name }}</div>
</div>
</script>
<!-- 操作 -->
<script type="text/html" id="operation">
<div class="table-btn">
<a class="layui-btn" lay-event="operate">运营</a>
{{# if( d.live_status == '102' ){ }}
<a class="layui-btn" lay-event="push">推流入口</a>
{{# } }}
<a class="layui-btn" lay-event="delete">删除</a>
</div>
</script>
<div class="progress-layer">
<h3>正在同步中...</h3>
<div class="layui-progress layui-progress-big" lay-showPercent="true" lay-filter="progress">
<div class="layui-progress-bar layui-bg-blue" lay-percent="0%"></div>
</div>
</div>
<script type="text/html" id="push">
<div class="push-layer">
<img src="LIVE_IMG/weapp_push.png" class="qrcode">
<p>主播可通过微信扫该小程序码或搜一搜小程序直播进入主播小程序进行推流</p>
</div>
</script>
<!-- 批量操作 -->
<script type="text/html" id="batchOperation">
<button class="layui-btn layui-btn-primary" lay-event="delete">批量删除</button>
</script>
<script>
var form,table,element,syncClick = false,repeat_flag = false;
layui.use(['form', 'element'], function() {
form = layui.form;
element = layui.element;
table = new Table({
elem: '#room_list',
url: ns.url("live://shop/room/index"),
bottomToolbar: "#batchOperation",
cols: [
[{
type: 'checkbox',
unresize: 'false',
width: '3%'
},{
title: '直播间信息',
unresize: 'false',
width: '17%',
templet: "#roominfo"
}, {
field: 'roomid',
title: '直播间ID',
unresize: 'false',
width: '10%'
}, {
field: 'anchor_name',
title: '主播昵称',
unresize: 'false',
width: '10%'
}, {
field: 'start_time',
title: '开播时间',
unresize: 'false',
width: '15%',
templet: function(data) {
return ns.time_to_date(data.start_time);
}
}, {
field: 'end_time',
title: '结束时间',
unresize: 'false',
width: '15%',
templet: function(data) {
return ns.time_to_date(data.end_time);
}
}, {
title: '商品数量',
unresize: 'false',
width: '10%',
templet: function(data) {
return data.goods.length;
}
}, {
field: 'status_name',
title: '状态',
unresize: 'false',
width: '10%',
}, {
title: '操作',
toolbar: '#operation',
unresize: 'false',
align:'right'
}]
]
});
table.tool(function(obj) {
var data = obj.data;
switch (obj.event) {
case 'operate': // 运营
location.hash = ns.hash("live://shop/room/operate", {"id": data.id});
break;
case 'push':
layer.open({
type: 1,
title: false,
skin: 'weapp-preview',
closeBtn: 1, //不显示关闭按钮
anim: 0,
shadeClose: false,
content: $('#push').html()
});
break;
case 'delete':
deleteRoom(data.id);
break;
}
})
// 批量操作
table.bottomToolbar(function (obj) {
if (obj.data.length < 1) {
layer.msg('请选择要操作的数据');
return;
}
var id_array = new Array();
for (i in obj.data) id_array.push(obj.data[i].id);
switch (obj.event) {
case "delete":
deleteRoom(id_array.toString());
break;
}
});
});
// 同步直播间
function sync(start){
if (syncClick) return;
syncClick = true;
start = start == undefined ? 0 : start;
$.ajax({
url: ns.url("live://shop/room/sync"),
data: {
start: 0,
},
dataType: 'JSON',
type: 'POST',
success: function(res) {
syncClick = false;
if (res.code == 0) {
var data = res.data,
next = parseInt(start) + 1;
if (next < data.total_page) {
if (start == 0) {
$(".progress-layer").fadeOut();
}
var progress = (next / data.total_page * 100).toFixed(2);
element.progress('progress', progress + '%');
// 拉取下一页
sync(next);
} else {
if (!$(".progress-layer").is(':hidden')) $(".progress-layer").fadeOut();
layer.closeAll();
layer.msg('同步成功');
table.reload();
}
} else {
layer.msg(res.message);
}
}
});
}
function
deleteRoom(room_ids){
layer.confirm('是否确定要删除所选直播间?', {title: '提示'}, function (index) {
if (repeat_flag) return;
repeat_flag = true;
layer.close(index);
$.ajax({
url: ns.url("live://shop/room/delete"),
data: {room_ids: room_ids},
dataType: 'JSON',
type: 'POST',
success: function (res) {
layer.msg(res.message);
repeat_flag = false;
if (res.code == 0) {
table.reload();
}
}
});
});
}
</script>

View File

@@ -0,0 +1,278 @@
<style type="text/css">
.goods-empty {padding: 100px 0;text-align: center;}
.goods-info {padding: 5px 0;align-items: center;flex-wrap:unset!important;}
.goods-info .room-name {padding-left: 5px;line-height: 1}
.goods-info img {width:50px;height: 50px;}
</style>
<div class="layui-form form-wrap">
<div class="layui-card card-common card-brief">
<div class="layui-card-header">
<span class="card-title">直播间信息</span>
</div>
<div class="layui-card-body">
<div class="layui-form-item">
<label class="layui-form-label">直播间:</label>
<div class="layui-input-block">
<p>{$room_info.name}</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">直播时间:</label>
<div class="layui-input-block">
<p>{:time_to_date($room_info.start_time)} - {:time_to_date($room_info.end_time)}</p>
</div>
</div>
</div>
</div>
<div class="layui-card card-common card-brief">
<div class="layui-card-header">
<span class="card-title">列表展示</span>
</div>
<div class="layui-card-body">
<div class="layui-form-item">
<label class="layui-form-label">主播头像:</label>
<div class="layui-input-block">
<div class="upload-img-block">
<div class="upload-img-box {notempty name="$room_info['anchor_img']"}hover{/notempty}">
<div class="upload-default" id="anchorImgUpload">
{if condition="$room_info.anchor_img"}
<div id="preview_imgUpload" class="preview_img">
<img layer-src src="{:img($room_info.anchor_img)}" class="img_prev"/>
</div>
{else/}
<div class="upload">
<i class="iconfont iconshangchuan"></i>
<p>点击上传</p>
</div>
{/if}
</div>
<div class="operation">
<div>
<i title="图片预览" class="iconfont iconreview js-preview" style="margin-right: 20px;"></i>
<i title="删除图片" class="layui-icon layui-icon-delete js-delete" ></i>
</div>
<div class="replace_img js-replace">点击替换</div>
</div>
<input type="hidden" name="anchor_img" value="{if condition="$room_info.anchor_img"}{$room_info.anchor_img}{/if}"/>
</div>
<!-- <p id="anchorImgUpload" class=" {if condition='$room_info.anchor_img'} replace {else/} no-replace{/if}">替换</p>
<i class="del {if condition="$room_info.anchor_img"}show{/if}">x</i> -->
</div>
</div>
<div class="word-aux">
<p>在直播列表中展示。建议尺寸100像素 * 100像素图片大小不得超过1M。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"><span class="required">*</span>直播间横幅:</label>
<div class="layui-input-block">
<div class="upload-img-block">
<div class="upload-img-box {notempty name="$room_info['banner']"}hover{/notempty}" >
<div class="upload-default" id="bannerUpload">
{if condition="$room_info.banner"}
<div id="preview_imgUpload" class="preview_img">
<img layer-src src="{:img($room_info.banner)}" class="img_prev"/>
</div>
{else/}
<i class="iconfont iconshangchuan"></i>
<p>点击上传</p>
{/if}
</div>
<div class="operation">
<div>
<i title="图片预览" class="iconfont iconreview js-preview" style="margin-right: 20px;"></i>
<i title="删除图片" class="layui-icon layui-icon-delete js-delete"></i>
</div>
<div class="replace_img js-replace">点击替换</div>
</div>
<input type="hidden" name="banner" value="{if condition="$room_info.banner"}{$room_info.banner}{/if}"/>
</div>
<!-- <p class="{if condition='$room_info.banner'} replace {else/} no-replace{/if}">替换</p>
<i class="del {if condition="$room_info.banner"}show{/if}">x</i> -->
</div>
</div>
<div class="word-aux">
<p>在直播列表中展示。建议尺寸702像素 * 208像素图片大小不得超过2M。</p>
</div>
</div>
</div>
</div>
<div class="layui-card card-common card-brief">
<div class="layui-card-header">
<span class="card-title">商品货架</span>
</div>
<div class="layui-card-body">
{notempty name="$room_info['goods']"}
{if $room_info['live_status'] == '102'}
<div class="single-filter-box">
<button class="layui-btn" onclick="importGoods()">从商品库导入商品</button>
</div>
{/if}
<table class="layui-table" lay-skin="nob">
<colgroup>
<col width="10%">
<col width="45">
<col width="15">
<col width="30">
<col>
</colgroup>
<thead>
<tr>
<th>序号</th>
<th>商品信息</th>
<th>价格</th>
<th>商品链接</th>
</tr>
</thead>
<tbody>
{foreach name="$room_info['goods']" item="vo" index="k"}
{php}
preg_match("/(pages\/goods\/detail\?sku_id=)(\d*)$/", $vo['url'], $matches);
{/php}
{if isset($matches[2])}
<tr data-sku="{$matches[2]}">
{else/}
<tr>
{/if}
<td>{$k}</td>
<td>
<div class="table-btn goods-info">
<img src="{:img($vo.cover_img)}">
<span class="room-name" title="{$vo.name}">{$vo.name}</span>
</div>
</td>
<td>{:sprintf("%.2f", $vo.price)}</td>
<td>{$vo.url}</td>
</tr>
{/foreach}
</tbody>
</table>
{else/}
{if $room_info['live_status'] == '102'}
<div class="goods-empty">暂无商品<a href="javascript:;" class="text-color" onclick="importGoods()">从商品库中导入</a></div>
{else/}
<div class="goods-empty">暂无商品</div>
{/if}
{/notempty}
</div>
</div>
</div>
<script type="text/javascript">
layui.use(['form'], function() {
var anchor_upload = new Upload({
elem: '#anchorImgUpload',
callback:function (res) {
if (res.code >= 0) {
$.ajax({
type: 'POST',
dataType: 'JSON',
url: ns.url("live://shop/room/operate"),
data: {
id: {$room_info.id},
anchor_img: res.data.pic_path
},
success: function(res) {}
});
}
},
deleteCallback:function () {
anchor_upload.delete();
$.ajax({
type: 'POST',
dataType: 'JSON',
url: ns.url("live://shop/room/operate"),
data: {
id: {$room_info.id},
anchor_img: ''
},
success: function(res) {
layer.msg(res.message);
}
});
}
});
var banner_upload = new Upload({
elem: '#bannerUpload',
callback:function (res) {
if (res.code >= 0) {
$.ajax({
type: 'POST',
dataType: 'JSON',
url: ns.url("live://shop/room/operate"),
data: {
id: {$room_info.id},
banner: res.data.pic_path
},
success: function(res) {}
});
}
},
deleteCallback:function () {
banner_upload.delete();
$.ajax({
type: 'POST',
dataType: 'JSON',
url: ns.url("live://shop/room/operate"),
data: {
id: {$room_info.id},
banner: ''
},
success: function(res) {
layer.msg(res.message);
}
});
}
});
});
function importGoods() {
var sku_id = [];
if ($('[data-sku]').length) {
$('[data-sku]').each(function (el) {
sku_id.push($(this).attr('data-sku'));
})
}
layer.open({
type: 2,
title: '从商品库导入商品',
content: ns.url("live://shop/room/getGoodsPageList", {request_mode: 'iframe', ids: sku_id.toString()}),
area: ['600px', '550px'],
btn: ['确定', '取消'],
yes: function (index, layero) {
form.render();
var iframeWin = document.getElementById(layero.find('iframe')[0]['name']).contentWindow;//得到iframe页的窗口对象执行iframe页的方法
iframeWin.liveSelectGoodsCallbackListener(function (data) {
if (data.length == 0) {
layer.msg('请选择要添加的商品', {icon: 5, shift: 6});
return;
}
$.ajax({
type: 'POST',
url: ns.url("live://shop/room/addGoods"),
data: {
room_id: {$room_info.roomid},
data: JSON.stringify(data)
},
dataType: 'JSON',
success: function (res) {
layer.msg(res.msg);
if (res.code == 0) {
layer.closeAll();
listenerHash(); // 刷新页面
}
}
});
})
}
})
}
</script>