diff --git a/admin_v3.sql b/admin_v3.sql index 7016a79c4..813b0704b 100644 --- a/admin_v3.sql +++ b/admin_v3.sql @@ -10,11 +10,131 @@ Target Server Type : MYSQL Target Server Version : 50559 File Encoding : 65001 -Date: 2018-03-13 15:44:55 +Date: 2018-03-23 17:07:39 */ SET FOREIGN_KEY_CHECKS=0; +-- ---------------------------- +-- Table structure for goods +-- ---------------------------- +DROP TABLE IF EXISTS `goods`; +CREATE TABLE `goods` ( + `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT, + `brand_id` bigint(20) unsigned DEFAULT '0' COMMENT '品牌ID', + `cate_id` bigint(20) unsigned DEFAULT '0' COMMENT '商品分类id', + `unit_id` bigint(20) DEFAULT NULL COMMENT '商品单位ID', + `spec_id` bigint(20) unsigned DEFAULT '0' COMMENT '规格ID', + `tags_id` varchar(255) DEFAULT '' COMMENT '商品标签ID', + `is_code` bigint(1) DEFAULT '1' COMMENT '是否有码商品', + `goods_title` varchar(255) DEFAULT '' COMMENT '商品标签', + `goods_content` text COMMENT '商品内容', + `goods_logo` varchar(255) DEFAULT '' COMMENT '商品LOGO', + `goods_image` text COMMENT '商品图片地址', + `goods_video` varchar(500) DEFAULT '' COMMENT '商品视频URL', + `goods_desc` varchar(500) DEFAULT '' COMMENT '商品描述', + `package_stock` bigint(20) unsigned DEFAULT '0' COMMENT '总库存数量', + `package_sale` bigint(20) unsigned DEFAULT '0' COMMENT '已销售数量', + `favorite_num` bigint(20) unsigned DEFAULT '0' COMMENT '收藏次数', + `sort` bigint(20) unsigned DEFAULT '0' COMMENT '数据排序', + `status` bigint(1) unsigned DEFAULT '1' COMMENT '商品状态(1有效,0无效)', + `is_deleted` bigint(1) unsigned DEFAULT '0' COMMENT '删除状态(1删除,0未删除)', + `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='商城商品主表'; + +-- ---------------------------- +-- Records of goods +-- ---------------------------- + +-- ---------------------------- +-- Table structure for goods_brand +-- ---------------------------- +DROP TABLE IF EXISTS `goods_brand`; +CREATE TABLE `goods_brand` ( + `id` int(11) unsigned NOT NULL AUTO_INCREMENT, + `brand_logo` varchar(1024) DEFAULT '' COMMENT '品牌logo', + `brand_cover` varchar(1024) DEFAULT '' COMMENT '品牌封面', + `brand_title` varchar(255) DEFAULT '' COMMENT '商品品牌名称', + `brand_desc` text COMMENT '商品品牌描述', + `brand_detail` text COMMENT '品牌图文信息', + `sort` int(11) unsigned DEFAULT '0' COMMENT '商品分类排序', + `status` tinyint(1) unsigned DEFAULT '1' COMMENT '商品状态(1有效,0无效)', + `is_deleted` tinyint(1) unsigned DEFAULT '0' COMMENT '删除状态(1删除,0未删除)', + `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='商品产品品牌'; + +-- ---------------------------- +-- Records of goods_brand +-- ---------------------------- + +-- ---------------------------- +-- Table structure for goods_cate +-- ---------------------------- +DROP TABLE IF EXISTS `goods_cate`; +CREATE TABLE `goods_cate` ( + `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT, + `pid` bigint(20) unsigned DEFAULT '0' COMMENT '上级分类编号', + `brand_id` bigint(20) DEFAULT '0' COMMENT '品牌ID', + `cate_title` varchar(255) DEFAULT '' COMMENT '商品分类名称', + `cate_desc` text COMMENT '商品分类', + `sort` bigint(20) unsigned DEFAULT '0' COMMENT '商品分类排序', + `status` bigint(1) unsigned DEFAULT '1' COMMENT '商品状态(1有效,0无效)', + `is_deleted` bigint(1) unsigned DEFAULT '0' COMMENT '删除状态(1删除,0未删除)', + `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='商城商品分类'; + +-- ---------------------------- +-- Records of goods_cate +-- ---------------------------- + +-- ---------------------------- +-- Table structure for goods_list +-- ---------------------------- +DROP TABLE IF EXISTS `goods_list`; +CREATE TABLE `goods_list` ( + `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT, + `goods_id` bigint(20) unsigned DEFAULT '0' COMMENT '商品ID', + `goods_spec` varchar(255) DEFAULT '' COMMENT '商品规格名称', + `goods_number` bigint(20) unsigned DEFAULT '0' COMMENT '商品礼品-商品数量', + `market_price` decimal(20,2) unsigned DEFAULT '0.00' COMMENT '销售价格', + `selling_price` decimal(20,2) unsigned DEFAULT '0.00' COMMENT '商品价格', + `goods_stock` bigint(20) unsigned DEFAULT '0' COMMENT '商品库存统计', + `goods_sale` bigint(20) unsigned DEFAULT '0' COMMENT '已销售数量', + `status` bigint(1) unsigned DEFAULT '1' COMMENT '商品状态(1有效,0无效)', + `is_deleted` bigint(1) unsigned DEFAULT '0' COMMENT '删除状态(1删除,0未删除)', + `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='商城商品列表'; + +-- ---------------------------- +-- Records of goods_list +-- ---------------------------- + +-- ---------------------------- +-- Table structure for goods_spec +-- ---------------------------- +DROP TABLE IF EXISTS `goods_spec`; +CREATE TABLE `goods_spec` ( + `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT, + `mch_id` bigint(20) unsigned DEFAULT '0' COMMENT '商户ID', + `spec_title` varchar(255) DEFAULT '' COMMENT '商品规格名称', + `spec_param` varchar(255) DEFAULT '' COMMENT '商品规格参数', + `spec_desc` varchar(255) DEFAULT '' COMMENT '商品规格描述', + `sort` bigint(20) unsigned DEFAULT '0' COMMENT '商品规格排序', + `status` bigint(1) unsigned DEFAULT '1' COMMENT '商品状态(1有效,0无效)', + `is_deleted` bigint(1) unsigned DEFAULT '0' COMMENT '删除状态(1删除,0未删除)', + `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', + PRIMARY KEY (`id`), + KEY `index_store_goods_spec_mch_id` (`mch_id`) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='商城商品规格'; + +-- ---------------------------- +-- Records of goods_spec +-- ---------------------------- + -- ---------------------------- -- Table structure for system_auth -- ---------------------------- @@ -63,7 +183,7 @@ CREATE TABLE `system_config` ( `value` varchar(500) DEFAULT NULL COMMENT '配置值', PRIMARY KEY (`id`), KEY `index_system_config_name` (`name`) -) ENGINE=InnoDB AUTO_INCREMENT=34 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='系统参数配置'; +) ENGINE=InnoDB AUTO_INCREMENT=43 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='系统参数配置'; -- ---------------------------- -- Records of system_config @@ -86,6 +206,15 @@ INSERT INTO `system_config` VALUES ('15', 'storage_oss_endpoint', 'oss-cn-beijin INSERT INTO `system_config` VALUES ('16', 'storage_oss_domain', 'cuci.oss-cn-beijing.aliyuncs.com'); INSERT INTO `system_config` VALUES ('17', 'storage_oss_keyid', '用你自己的吧'); INSERT INTO `system_config` VALUES ('18', 'storage_oss_secret', '用你自己的吧'); +INSERT INTO `system_config` VALUES ('34', 'wechat_appid', 'wx60a43dd8161666d4'); +INSERT INTO `system_config` VALUES ('35', 'wechat_appkey', '9890a0d7c91801a609d151099e95b61a'); +INSERT INTO `system_config` VALUES ('36', 'storage_oss_is_https', 'http'); +INSERT INTO `system_config` VALUES ('37', 'wechat_type', 'api'); +INSERT INTO `system_config` VALUES ('38', 'wechat_token', 'test'); +INSERT INTO `system_config` VALUES ('39', 'wechat_appsecret', 'a041bec98ed015d52b99acea5c6a16ef'); +INSERT INTO `system_config` VALUES ('40', 'wechat_encodingaeskey', 'BJIUzE0gqlWy0GxfPp4J1oPTBmOrNDIGPNav1YFH5Z5'); +INSERT INTO `system_config` VALUES ('41', 'wechat_thr_appid', 'wx60a43dd8161666d4'); +INSERT INTO `system_config` VALUES ('42', 'wechat_thr_appkey', '05db2aa335382c66ab56d69b1a9ad0ee'); -- ---------------------------- -- Table structure for system_log @@ -100,7 +229,11 @@ CREATE TABLE `system_log` ( `content` text NOT NULL COMMENT '操作内容描述', `create_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', PRIMARY KEY (`id`) -) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8 COMMENT='系统操作日志表'; +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='系统操作日志表'; + +-- ---------------------------- +-- Records of system_log +-- ---------------------------- -- ---------------------------- -- Table structure for system_menu @@ -121,7 +254,7 @@ CREATE TABLE `system_menu` ( `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', PRIMARY KEY (`id`), KEY `index_system_menu_node` (`node`) USING BTREE -) ENGINE=InnoDB AUTO_INCREMENT=56 DEFAULT CHARSET=utf8 COMMENT='系统菜单表'; +) ENGINE=InnoDB AUTO_INCREMENT=32 DEFAULT CHARSET=utf8 COMMENT='系统菜单表'; -- ---------------------------- -- Records of system_menu @@ -166,7 +299,7 @@ CREATE TABLE `system_node` ( `create_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', PRIMARY KEY (`id`), KEY `index_system_node_node` (`node`) -) ENGINE=InnoDB AUTO_INCREMENT=322 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='系统节点表'; +) ENGINE=InnoDB AUTO_INCREMENT=230 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='系统节点表'; -- ---------------------------- -- Records of system_node @@ -182,7 +315,6 @@ INSERT INTO `system_node` VALUES ('8', 'admin/auth/resume', '启用权限', '0', INSERT INTO `system_node` VALUES ('9', 'admin/auth/del', '删除权限', '0', '1', '1', '2018-01-23 12:39:25'); INSERT INTO `system_node` VALUES ('10', 'admin/config/index', '系统参数', '1', '1', '1', '2018-01-23 12:39:25'); INSERT INTO `system_node` VALUES ('11', 'admin/config/file', '文件存储', '0', '1', '1', '2018-01-23 12:39:26'); -INSERT INTO `system_node` VALUES ('12', 'admin/config/sms', '短信接口', '0', '1', '1', '2018-01-23 12:39:28'); INSERT INTO `system_node` VALUES ('13', 'admin/log/index', '日志记录', '1', '1', '1', '2018-01-23 12:39:28'); INSERT INTO `system_node` VALUES ('14', 'admin/log/sms', '短信记录', '0', '1', '1', '2018-01-23 12:39:29'); INSERT INTO `system_node` VALUES ('15', 'admin/log/del', '日志删除', '0', '1', '1', '2018-01-23 12:39:29'); @@ -294,7 +426,7 @@ CREATE TABLE `system_user` ( -- ---------------------------- -- Records of system_user -- ---------------------------- -INSERT INTO `system_user` VALUES ('10000', 'admin', '21232f297a57a5a743894a0e4a801fc3', '', '', '', '', '22409', '2018-03-13 15:43:48', '1', '1', '0', null, '2015-11-13 15:14:22'); +INSERT INTO `system_user` VALUES ('10000', 'admin', '21232f297a57a5a743894a0e4a801fc3', '22222222', '', '18993368867', '', '23296', '2018-03-23 17:05:40', '1', '2,4', '0', null, '2015-11-13 15:14:22'); -- ---------------------------- -- Table structure for wechat_fans diff --git a/application/goods/controller/Brand.php b/application/goods/controller/Brand.php new file mode 100644 index 000000000..5327a3419 --- /dev/null +++ b/application/goods/controller/Brand.php @@ -0,0 +1,151 @@ + + * @date 2017/03/27 14:43 + */ +class Brand extends BasicAdmin +{ + + /** + * 定义当前操作表名 + * @var string + */ + public $table = 'GoodsBrand'; + + /** + * 品牌列表 + * @return array|string + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function index() + { + $this->title = '品牌管理'; + $get = $this->request->get(); + $db = Db::name($this->table)->where(['is_deleted' => '0']); + if (isset($get['brand_title']) && $get['brand_title'] !== '') { + $db->whereLike('brand_title', "%{$get['brand_title']}%"); + } + if (isset($get['create_at']) && $get['create_at'] !== '') { + list($start, $end) = explode(' - ', $get['create_at']); + $db->whereBetween('create_at', ["{$start} 00:00:00", "{$end} 23:59:59"]); + } + return parent::_list($db->order('sort asc,id desc')); + } + + /** + * 添加品牌 + * @return array|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + * @throws \think\Exception + */ + public function add() + { + $this->title = '添加品牌'; + return $this->_form($this->table, 'form'); + } + + /** + * 编辑品牌 + * @return array|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + * @throws \think\Exception + */ + public function edit() + { + $this->title = '编辑品牌'; + return $this->_form($this->table, 'form'); + } + + /** + * 表单提交数据处理 + * @param array $data + */ + protected function _form_filter($data) + { + if ($this->request->isPost()) { + empty($data['brand_logo']) && $this->error('请上传品牌Logo图片'); + empty($data['brand_cover']) && $this->error('请上传品牌封面图片'); + } + } + + /** + * 添加成功回跳处理 + * @param bool $result + */ + protected function _form_result($result) + { + if ($result !== false) { + list($base, $spm, $url) = [url('@admin'), $this->request->get('spm'), url('goods/brand/index')]; + $this->success('数据保存成功!', "{$base}#{$url}?spm={$spm}"); + } + } + + /** + * 删除品牌 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function del() + { + if (DataService::update($this->table)) { + $this->success("品牌删除成功!", ''); + } + $this->error("品牌删除失败,请稍候再试!"); + } + + /** + * 品牌禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function forbid() + { + if (DataService::update($this->table)) { + $this->success("品牌禁用成功!", ''); + } + $this->error("品牌禁用失败,请稍候再试!"); + } + + /** + * 品牌签禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function resume() + { + if (DataService::update($this->table)) { + $this->success("品牌启用成功!", ''); + } + $this->error("品牌启用失败,请稍候再试!"); + } + +} diff --git a/application/goods/controller/Cate.php b/application/goods/controller/Cate.php new file mode 100644 index 000000000..ad9121d5a --- /dev/null +++ b/application/goods/controller/Cate.php @@ -0,0 +1,162 @@ + + * @date 2017/03/27 14:43 + */ +class Cate extends BasicAdmin +{ + + /** + * 定义当前操作表名 + * @var string + */ + public $table = 'GoodsCate'; + + /** + * 产品分类列表 + * @return array|string + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function index() + { + $brands = Db::name('GoodsBrand')->where(['status' => '1'])->order('sort asc,id asc')->select(); + $this->assign('brands', $brands); + $this->title = '产品分类'; + $db = Db::name($this->table)->where(['is_deleted' => '0']); + return parent::_list($db->order('sort asc,id asc'), false); + } + + /** + * 列表数据处理 + * @param array $data + */ + protected function _index_data_filter(&$data) + { + foreach ($data as &$vo) { + $vo['ids'] = join(',', ToolsService::getArrSubIds($data, $vo['id'])); + } + $data = ToolsService::arr2table($data); + } + + /** + * 添加菜单 + * @return array|string + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function add() + { + return $this->_form($this->table, 'form'); + } + + /** + * 编辑菜单 + * @return array|string + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function edit() + { + return $this->_form($this->table, 'form'); + } + + /** + * 表单数据前缀方法 + * @param array $vo + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + protected function _form_filter(&$vo) + { + if ($this->request->isGet()) { + // 读取所有品牌列表 + $brands = Db::name('GoodsBrand')->where(['status' => '1'])->order('sort asc,id asc')->select(); + $this->assign('brands', $brands); + // 读取上级分类 + $where = ['status' => '1', 'is_deleted' => '0']; + $_cates = (array)Db::name($this->table)->where($where)->order('sort desc,id desc')->select(); + array_unshift($_cates, ['id' => 0, 'pid' => -1, 'cate_title' => '--- 顶级分类 ---']); + $cates = ToolsService::arr2table($_cates); + foreach ($cates as $key => &$cate) { + if (isset($vo['pid'])) { + $path = "-{$vo['pid']}-{$vo['id']}"; + if ($vo['pid'] !== '' && (stripos("{$cate['path']}-", "{$path}-") !== false || $cate['path'] === $path)) { + unset($cates[$key]); + } + } + } + $this->assign('cates', $cates); + } + } + + /** + * 删除产品分类 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function del() + { + if (DataService::update($this->table)) { + $this->success("产品分类删除成功!", ''); + } + $this->error("产品分类删除失败,请稍候再试!"); + } + + /** + * 产品分类禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function forbid() + { + if (DataService::update($this->table)) { + $this->success("产品分类禁用成功!", ''); + } + $this->error("产品分类禁用失败,请稍候再试!"); + } + + /** + * 产品分类禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function resume() + { + if (DataService::update($this->table)) { + $this->success("产品分类启用成功!", ''); + } + $this->error("产品分类启用失败,请稍候再试!"); + } + +} diff --git a/application/goods/controller/Product.php b/application/goods/controller/Product.php new file mode 100644 index 000000000..482f666fc --- /dev/null +++ b/application/goods/controller/Product.php @@ -0,0 +1,264 @@ + + * @date 2017/03/27 14:43 + */ +class Product extends BasicAdmin +{ + + /** + * 定义当前操作表名 + * @var string + */ + public $table = 'Goods'; + + /** + * 普通产品 + * @return array|string + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function index() + { + $this->title = '产品管理'; + $get = $this->request->get(); + $db = Db::name($this->table)->where(['is_deleted' => '0']); + foreach (['tags_id', 'goods_title'] as $field) { + (isset($get[$field]) && $get[$field] !== '') && $db->whereLike($field, "%,{$get[$field]},%"); + } + foreach (['cate_id', 'brand_id'] as $field) { + (isset($get[$field]) && $get[$field] !== '') && $db->where($field, $get[$field]); + } + if (isset($get['create_at']) && $get['create_at'] !== '') { + list($start, $end) = explode(' - ', $get['create_at']); + $db->whereBetween('create_at', ["{$start} 00:00:00", "{$end} 23:59:59"]); + } + return parent::_list($db->order('status desc,sort asc,id desc')); + } + + /** + * 商城数据处理 + * @param array $data + */ + protected function _data_filter(&$data) + { + $result = ProductService::buildGoodsList($data); + $this->assign([ + 'brands' => $result['brand'], + 'cates' => ToolsService::arr2table($result['cate']), + ]); + } + + /** + * 添加产品 + * @return array|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + * @throws \think\Exception + */ + public function add() + { + if (!$this->request->isPost()) { + $this->title = '添加产品'; + $this->_form_assign(); + return $this->_form($this->table, 'form'); + } + try { + $data = $this->_form_build_data(); + Db::transaction(function () use ($data) { + $goodsID = Db::name($this->table)->insertGetId($data['main']); + foreach ($data['list'] as &$vo) { + $vo['goods_id'] = $goodsID; + } + Db::name('GoodsList')->insertAll($data['list']); + }); + } catch (HttpResponseException $exception) { + return $exception->getResponse(); + } catch (\Exception $e) { + $this->error('产品添加失败,请稍候再试!'); + } + list($base, $spm, $url) = [url('@admin'), $this->request->get('spm'), url('store/goods/index')]; + $this->success('添加产品成功!', "{$base}#{$url}?spm={$spm}"); + } + + /** + * 编辑产品 + * @return array|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function edit() + { + if (!$this->request->isPost()) { + $goods_id = $this->request->get('id'); + $goods = Db::name($this->table)->where(['id' => $goods_id, 'is_deleted' => '0'])->find(); + empty($goods) && $this->error('需要编辑的产品不存在!'); + $specWhere = ['goods_id' => $goods_id, 'is_deleted' => '0']; + $goods['list'] = Db::name('GoodsList')->where($specWhere)->select(); + $this->_form_assign(); + return $this->fetch('form', ['vo' => $goods, 'title' => '编辑产品']); + } + try { + $data = $this->_form_build_data(); + $goods_id = $this->request->post('id'); + $goods = Db::name($this->table)->where(['id' => $goods_id, 'is_deleted' => '0'])->find(); + empty($goods) && $this->error('产品编辑失败,请稍候再试!'); + foreach ($data['list'] as &$vo) { + $vo['goods_id'] = $goods_id; + } + Db::transaction(function () use ($data, $goods_id, $goods) { + // 更新产品主表 + $where = ['id' => $goods_id, 'is_deleted' => '0']; + Db::name('Goods')->where($where)->update(array_merge($goods, $data['main'])); + // 更新产品详细 + Db::name('GoodsList')->where(['goods_id' => $goods_id])->delete(); + Db::name('GoodsList')->insertAll($data['list']); + }); + } catch (HttpResponseException $exception) { + return $exception->getResponse(); + } catch (\Exception $e) { + $this->error('产品编辑失败,请稍候再试!' . $e->getMessage()); + } + list($base, $spm, $url) = [url('@admin'), $this->request->get('spm'), url('goods/product/index')]; + $this->success('产品编辑成功!', "{$base}#{$url}?spm={$spm}"); + } + + /** + * 表单数据处理 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + protected function _form_assign() + { + // 信息 + list($where, $order) = [['status' => '1', 'is_deleted' => '0'], 'sort asc,id desc']; + $specs = (array)Db::name('GoodsSpec')->where($where)->order($order)->select(); + $brands = (array)Db::name('GoodsBrand')->where($where)->order($order)->select(); + $cates = (array)Db::name('GoodsCate')->where($where)->order($order)->select(); + // 所有的产品信息 + $where = ['is_deleted' => '0', 'status' => '1']; + $goodsListField = 'goods_id,goods_spec,goods_stock,goods_sale'; + $goods = Db::name('Goods')->field('id,goods_title')->where($where)->select(); + $list = Db::name('GoodsList')->field($goodsListField)->where($where)->select(); + foreach ($goods as $k => $g) { + $goods[$k]['list'] = []; + foreach ($list as $v) { + ($g['id'] === $v['goods_id']) && $goods[$k]['list'][] = $v; + } + } + array_unshift($specs, ['spec_title' => ' - 不使用规格模板 -', 'spec_param' => '[]', 'id' => '0']); + $this->assign([ + 'specs' => $specs, + 'cates' => ToolsService::arr2table($cates), + 'brands' => $brands, + 'all' => $goods, + ]); + } + + /** + * 读取POST表单数据 + * @return array + */ + protected function _form_build_data() + { + list($main, $list, $post, $verify) = [[], [], $this->request->post(), false]; + empty($post['goods_logo']) && $this->error('产品LOGO不能为空,请上传后再提交数据!'); + // 产品主数据组装 + $main['cate_id'] = $this->request->post('cate_id', '0'); + $main['spec_id'] = $this->request->post('spec_id', '0'); + $main['brand_id'] = $this->request->post('brand_id', '0'); + $main['goods_logo'] = $this->request->post('goods_logo', ''); + $main['goods_title'] = $this->request->post('goods_title', ''); + $main['goods_video'] = $this->request->post('goods_video', ''); + $main['goods_image'] = $this->request->post('goods_image', ''); + $main['goods_desc'] = $this->request->post('goods_desc', '', null); + $main['goods_content'] = $this->request->post('goods_content', ''); + $main['tags_id'] = ',' . join(',', isset($post['tags_id']) ? $post['tags_id'] : []) . ','; + // 产品从数据组装 + if (!empty($post['goods_spec'])) { + foreach ($post['goods_spec'] as $key => $value) { + $goods = []; + $goods['goods_spec'] = $value; + $goods['market_price'] = $post['market_price'][$key]; + $goods['selling_price'] = $post['selling_price'][$key]; + $goods['status'] = intval(!empty($post['spec_status'][$key])); + !empty($goods['status']) && $verify = true; + $list[] = $goods; + } + } else { + $this->error('没有产品规格或套餐信息哦!'); + } + !$verify && $this->error('没有设置有效的产品规格!'); + return ['main' => $main, 'list' => $list]; + } + + /** + * 删除产品 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function del() + { + if (DataService::update($this->table)) { + $this->success("产品删除成功!", ''); + } + $this->error("产品删除失败,请稍候再试!"); + } + + /** + * 产品禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function forbid() + { + if (DataService::update($this->table)) { + $this->success("产品禁用成功!", ''); + } + $this->error("产品禁用失败,请稍候再试!"); + } + + /** + * 产品禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function resume() + { + if (DataService::update($this->table)) { + $this->success("产品启用成功!", ''); + } + $this->error("产品启用失败,请稍候再试!"); + } + +} diff --git a/application/goods/controller/Spec.php b/application/goods/controller/Spec.php new file mode 100644 index 000000000..74766d422 --- /dev/null +++ b/application/goods/controller/Spec.php @@ -0,0 +1,156 @@ + + * @date 2017/03/27 14:43 + */ +class Spec extends BasicAdmin +{ + + /** + * 定义当前操作表名 + * @var string + */ + public $table = 'GoodsSpec'; + + /** + * 产品列表 + * @return array|string + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + */ + public function index() + { + $this->title = '规格管理(请勿随意修改或删除)'; + $get = $this->request->get(); + $db = Db::name($this->table)->where(['is_deleted' => '0']); + if (isset($get['spec_title']) && $get['spec_title'] !== '') { + $db->whereLike('spec_title', "%{$get['spec_title']}%"); + } + if (isset($get['date']) && $get['date'] !== '') { + list($start, $end) = explode(' - ', $get['date']); + $db->whereBetween('create_at', ["{$start} 00:00:00", "{$end} 23:59:59"]); + } + return parent::_list($db->order('sort asc,id desc')); + } + + /** + * 列表数据处理 + * @param array $data + */ + protected function _index_data_filter(&$data) + { + foreach ($data as &$vo) { + $vo['spec_param'] = json_decode($vo['spec_param'], true); + } + } + + /** + * 添加产品 + * @return array|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + * @throws \think\Exception + */ + public function add() + { + $this->title = '添加规格'; + return $this->_form($this->table, 'form'); + } + + /** + * 编辑产品 + * @return array|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + * @throws \think\Exception + */ + public function edit() + { + $this->title = '编辑规格'; + return $this->_form($this->table, 'form'); + } + + /** + * 表单数据处理 + * @param array $vo + */ + protected function _form_filter(&$vo) + { + if ($this->request->isPost()) { + $param = json_decode($this->request->post('spec_param', '[]', null), true); + foreach ($param as &$v) { + $count = 1; + while ($count) { + $v['value'] = trim(str_replace([' ', '_', ',', ',', ';', ';'], ' ', $v['value'], $count)); + } + } + $vo['spec_param'] = json_encode($param, JSON_UNESCAPED_UNICODE); + } + } + + /** + * 删除产品规格 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function del() + { + if (DataService::update($this->table)) { + $this->success("产品规格删除成功!", ''); + } + $this->error("产品规格删除失败,请稍候再试!"); + } + + /** + * 产品规格禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function forbid() + { + if (DataService::update($this->table)) { + $this->success("产品规格禁用成功!", ''); + } + $this->error("产品规格禁用失败,请稍候再试!"); + } + + /** + * 产品规格禁用 + * @throws \think\Exception + * @throws \think\exception\PDOException + */ + public function resume() + { + if (DataService::update($this->table)) { + $this->success("产品规格启用成功!", ''); + } + $this->error("产品规格启用失败,请稍候再试!"); + } + +} diff --git a/application/goods/service/ProductService.php b/application/goods/service/ProductService.php new file mode 100644 index 000000000..f9b1928f7 --- /dev/null +++ b/application/goods/service/ProductService.php @@ -0,0 +1,134 @@ + '1', 'is_deleted' => '0']; + $cateList = Db::name('GoodsCate')->where($cateWhere)->order('sort asc,id desc')->column($cateField); + // 产品品牌处理 + $brandField = 'id,brand_logo,brand_cover,brand_title,brand_desc,brand_detail'; + $brandWhere = ['status' => '1', 'is_deleted' => '0']; + $brandList = Db::name('GoodsBrand')->where($brandWhere)->order('sort asc,id desc')->column($brandField); + // 无产品列表时 + if (empty($goodsList)) { + return ['list' => $goodsList, 'cate' => $cateList, 'brand' => $brandList]; + } + // 读取产品详情列表 + $specWhere = ['status' => '1', 'is_deleted' => '0', ['goods_id', 'in', array_column($goodsList, 'id')]]; + $specField = 'id,goods_id,goods_spec,goods_number,market_price,selling_price,goods_stock,goods_sale'; + $specList = Db::name('GoodsList')->where($specWhere)->column($specField); + foreach ($specList as $key => $spec) { + foreach ($goodsList as $goods) { + if ($goods['id'] === $spec['goods_id']) { + $specList[$key]['goods_title'] = $goods['goods_title']; + } + } + if ($spec['goods_spec'] === 'default:default') { + $specList[$key]['goods_spec_alias'] = '默认规格'; + } else { + $specList[$key]['goods_spec_alias'] = str_replace([':', ','], [': ', ', '], $spec['goods_spec']); + } + } + // 产品数据组装 + foreach ($goodsList as $key => $vo) { + // 产品内容处理 + $goodsList[$key]['goods_content'] = htmlspecialchars_decode($vo['goods_content']); + // 产品品牌处理 + $goodsList[$key]['brand'] = isset($brandList[$vo['brand_id']]) ? $brandList[$vo['brand_id']] : []; + // 产品分类关联 + $goodsList[$key]['cate'] = []; + if (isset($cateList[$vo['cate_id']])) { + $goodsList[$key]['cate'][] = ($tcate = $cateList[$vo['cate_id']]); + while (isset($tcate['pid']) && $tcate['pid'] > 0 && isset($cateList[$tcate['pid']])) { + $goodsList[$key]['cate'][] = ($tcate = $cateList[$tcate['pid']]); + } + $goodsList[$key]['cate'] = array_reverse($goodsList[$key]['cate']); + } + // 产品详细列表关联 + $goodsList[$key]['spec'] = []; + foreach ($specList as $spec) { + if ($vo['id'] === $spec['goods_id']) { + $goodsList[$key]['spec'][] = $spec; + } + } + } + return ['list' => $goodsList, 'cate' => $cateList, 'brand' => $brandList]; + } + + /** + * 同步更新产品库存及售出(@todo 需要重新做库存统计) + * @param int $goods_id 产品ID + * @return array + * @throws \think\Exception + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\ModelNotFoundException + * @throws \think\exception\DbException + * @throws \think\exception\PDOException + */ + public static function syncGoodsStock($goods_id) + { + // 检查产品是否需要更新库存 + $map = ['id' => $goods_id, 'is_deleted' => '0']; + if (!($goods = Db::name('Goods')->where($map)->find())) { + return ['code' => 0, 'msg' => '指定产品信息无法同步库存!']; + } + // 统计入库信息 + $stockField = 'goods_id,goods_spec,ifnull(sum(goods_stock), 0) goods_stock'; + $stockWhere = ['status' => '1', 'is_deleted' => '0', 'goods_id' => $goods_id, 'mch_id' => $mch_id]; + $stockList = (array)Db::name('GoodsStock')->field($stockField)->where($stockWhere)->group('goods_id,goods_spec')->select(); + // 统计销售信息 + $saleField = 'goods_id,goods_spec,ifnull(sum(number), 0) goods_sale'; + $saleWhere = ['status' => '1', 'is_deleted' => '0', 'goods_id' => $goods_id, 'mch_id' => $mch_id]; + $saleList = (array)Db::name('StoreOrderList')->field($saleField)->where($saleWhere)->group('goods_id,goods_spec')->select(); + // 库存置零 + list($where, $total_stock, $total_sale) = [['goods_id' => $goods_id], 0, 0]; + Db::name('GoodsList')->where($where)->update(['goods_stock' => 0, 'goods_sale' => 0, 'mch_id' => $mch_id]); + // 更新产品库存 + foreach ($stockList as $stock) { + $total_stock += intval($stock['goods_stock']); + $where = ['goods_id' => $goods_id, 'goods_spec' => $stock['goods_spec'], 'mch_id' => $mch_id]; + Db::name('GoodsList')->where($where)->update(['goods_stock' => $stock['goods_stock']]); + } + // 更新产品销量 + foreach ($saleList as $sale) { + $total_sale += intval($sale['goods_sale']); + $where = ['goods_id' => $goods_id, 'goods_spec' => $sale['goods_spec'], 'mch_id' => $mch_id]; + Db::name('GoodsList')->where($where)->update(['goods_sale' => $sale['goods_sale']]); + } + // 更新总库存及总销量 + $update = ['package_stock' => $total_stock, 'package_sale' => $total_sale, 'mch_id' => $mch_id]; + Db::name('Goods')->where(['id' => $goods_id])->update($update); + return ['code' => 1, 'msg' => '同步产品库存成功!']; + } + +} \ No newline at end of file diff --git a/application/goods/view/brand/form.html b/application/goods/view/brand/form.html new file mode 100644 index 000000000..86c25498b --- /dev/null +++ b/application/goods/view/brand/form.html @@ -0,0 +1,63 @@ +{extend name='admin@public/content'} + +{block name="content"} +
+ +
+ +
+ +
+
+ +
+ +
+

品牌Logo图片

+ + +
+
+ +
+
+

品牌封面图片

+ + +
+
+ +
+ +
+ +
+
+ +
+ +
+ +
+
+ +
+ +
+
+ {if !empty($vo.id)}{/if} + + +
+
+ + + + +
+{/block} \ No newline at end of file diff --git a/application/goods/view/brand/index.html b/application/goods/view/brand/index.html new file mode 100644 index 000000000..9f993f7d8 --- /dev/null +++ b/application/goods/view/brand/index.html @@ -0,0 +1,108 @@ +{extend name='admin@public/content'} + +{block name="button"} + + + + + + + + + +{/block} + +{block name="content"} + + + + + + + +
+ {if empty($list)} +

没 有 记 录 哦!

+ {else} + + + + + + + + + + + + + + {foreach $list as $key=>$vo} + + + + + + + + + {/foreach} + +
+ + + + 产品品牌添加时间品牌状态
+ + + + {$vo.brand_title}{$vo.create_at|format_datetime} + {if $vo.status eq 0}已禁用{elseif $vo.status eq 1}使用中{/if} + + + {if auth("$classuri/edit")} + | + 编辑 + {/if} + + {if $vo.status eq 1 and auth("$classuri/forbid")} + | + 禁用 + {elseif auth("$classuri/resume")} + | + 启用 + {/if} + + {if auth("$classuri/del")} + | + 删除 + {/if} + +
+ {if isset($page)}

{$page|raw}

{/if} + {/if} +
+{/block} \ No newline at end of file diff --git a/application/goods/view/cate/form.html b/application/goods/view/cate/form.html new file mode 100644 index 000000000..4e72da548 --- /dev/null +++ b/application/goods/view/cate/form.html @@ -0,0 +1,39 @@ +
+ +
+ +
+ +
+
+ +
+ +
+ +
+
+ +
+ +
+ +
+
+ +
+ +
+ {if isset($vo['id'])}{/if} + + +
+
diff --git a/application/goods/view/cate/index.html b/application/goods/view/cate/index.html new file mode 100644 index 000000000..3aec38b2e --- /dev/null +++ b/application/goods/view/cate/index.html @@ -0,0 +1,83 @@ +{extend name='admin@public/content'} + +{block name="button"} + + + + + + +{/block} + +{block name="content"} +
+ +

没 有 记 录 哦!

+ + + + + + + + + + + + + + + + {foreach $list as $key=>$vo} + + + + + + + + + + {/foreach} + +
+ + + + 产品分类分类描述添加时间分类状态
+ + + + + {$vo.spl|raw} {$vo.cate_title} + + {$vo.cate_desc|default='未设置分类描述'|raw} + + {$vo.create_at|format_datetime} + + {if $vo.status eq 0}已禁用{elseif $vo.status eq 1}使用中{/if} + + + {if auth("$classuri/edit")} + | + 编辑 + {/if} + + {if $vo.status eq 1 and auth("$classuri/forbid")} + | + 禁用 + {elseif auth("$classuri/resume")} + | + 启用 + {/if} + + {if auth("$classuri/del")} + | + 删除 + {/if} + +
+ {if isset($page)}

{$page|raw}

{/if} + +
+{/block} \ No newline at end of file diff --git a/application/goods/view/product/form.html b/application/goods/view/product/form.html new file mode 100644 index 000000000..ea3cab946 --- /dev/null +++ b/application/goods/view/product/form.html @@ -0,0 +1,349 @@ +{extend name='admin@public/content'} + +{block name="content"} +
+ + +
+ +
+ +
+
+ + + +
+ +
+ +
+
+ + +
+ +
+ +
+
+ + {if !empty($tags)} +
+ +
+
+ {foreach $tags as $tag} + + {/foreach} +
+
+
+ {/if} + +
+ +
+ + + + + + + + + +
产品LOGO产品图片
+ + + +
+
+
+ + +
+ +
+ + + + + + + + + + +
可选规格规格内容
+ +
+ + + + + + + + + + + + + + + + + +
产品规格市场价格销售价格规格状态
+ + + + + + + + + + + +
+
+
+ + +
+ +
+
+
+ +
+
+ +
+
+
+
+ +
+ +
+ +
+
+ +
+ +
+ +
+
+ +
+ +
+
+ {if !empty($vo.id)}{/if} + + +
+
+ +
+ + +{/block} + +{block name="style"} + +{/block} \ No newline at end of file diff --git a/application/goods/view/product/index.html b/application/goods/view/product/index.html new file mode 100644 index 000000000..f3cf5e250 --- /dev/null +++ b/application/goods/view/product/index.html @@ -0,0 +1,189 @@ +{extend name='admin@public/content'} + +{block name="button"} + + + + + + + + + + + + + + + + + +{/block} + +{block name="content"} + + + + + +
+ +

没 有 记 录 哦!

+ + + + + + + + + + + + + + + {foreach $list as $key=>$vo} + + + + + + + + + + {/foreach} + +
+ + + + 品牌分类 + + + + + + + +
产品信息售价 ( 标价 ) / 库存 ( 剩余, 已售 )
+
添加时间 / 状态
+ + + + + 品牌:{$vo.brand.brand_title|default='未配置品牌'|raw}
+ 分类:{if empty($vo.cate)}未配置分类{else} + {foreach $vo.cate as $k=>$cate}{$cate.cate_title} + {if $k+1 < count($vo.cate)}{/if} + {/foreach} + {/if} +
+ + + + + {foreach $vo.spec as $spec} + + + + + {/foreach} +
+ [{$spec.goods_id}] {$spec.goods_title|default=''|raw} + {$spec.goods_spec_alias|raw} + + 售 {$spec.selling_price} ( 市 {$spec.market_price} ) + 存 {$spec.goods_stock} ( 剩 {$spec.goods_stock-$spec.goods_sale}, 售 {$spec.goods_sale} ) +
+
+ {$vo.create_at|format_datetime|str_replace=' ','
',###|raw} + {if $vo.status eq '0'}已下架{elseif $vo.status eq '1'}销售中{/if} +
+ + + | + 编辑 + + + + | + 下架 + + | + 上架 + + + + | + 删除 + + +
+ {if isset($page)}

{$page|raw}

{/if} + +
+ +{/block} \ No newline at end of file diff --git a/application/goods/view/spec/form.html b/application/goods/view/spec/form.html new file mode 100644 index 000000000..c9b987594 --- /dev/null +++ b/application/goods/view/spec/form.html @@ -0,0 +1,194 @@ +
+ +
+ +
+ +
+
+ +
+ +
+ + + + + + + + + + + + +
+ + + + + 删除 + 删除 + 上移 + 上移 + 下移 + 下移 +
+ 添加属性 +
+

设置多个内容值时请使用空格键或英文逗号隔开。

+
+
+ +
+ +
+ +
+
+ +
+ +
+ {if isset($vo['id'])}{/if} + + +
+ + + + +
diff --git a/application/goods/view/spec/index.html b/application/goods/view/spec/index.html new file mode 100644 index 000000000..6993e14db --- /dev/null +++ b/application/goods/view/spec/index.html @@ -0,0 +1,115 @@ +{extend name='admin@public/content'} + +{block name="button"} + + + + + + + + + +{/block} + +{block name="content"} + + + + + + + +
+ {if empty($list)} +

没 有 记 录 哦!

+ {else} + + + + + + + + + + + + + + + {foreach $list as $key=>$vo} + + + + + + + + + + {/foreach} + +
+ + + + 规格分组规格内容添加时间标签状态
+ + + + {$vo.spec_title} + {foreach $vo.spec_param as $param} +

{$param.name} : {$param.value}

+ {/foreach} +
+ {$vo.create_at|format_datetime|raw} + + {if $vo.status eq 0}已禁用{elseif $vo.status eq 1}使用中{/if} + + + {if auth("$classuri/edit")} + | + 编辑 + {/if} + + {if $vo.status eq 1 and auth("$classuri/forbid")} + | + 禁用 + {elseif auth("$classuri/resume")} + | + 启用 + {/if} + + {if auth("$classuri/del")} + | + 删除 + {/if} +
+ {if isset($page)}

{$page|raw}

{/if} + {/if} +
+{/block} \ No newline at end of file