説明なし
選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。

Api.php 65KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499
  1. <?php
  2. /**
  3. * 易优CMS
  4. * ============================================================================
  5. * 版权所有 2016-2028 海口快推科技有限公司,并保留所有权利。
  6. * 网站地址: http://www.eyoucms.com
  7. * ----------------------------------------------------------------------------
  8. * 如果商业用途务必到官方购买正版授权, 以免引起不必要的法律纠纷.
  9. * ============================================================================
  10. * Author: 小虎哥 <1105415366@qq.com>
  11. * Date: 2018-4-3
  12. */
  13. namespace app\api\model\v1;
  14. use think\Db;
  15. /**
  16. * 小程序模型类
  17. */
  18. class Api extends Base
  19. {
  20. //初始化
  21. protected function initialize()
  22. {
  23. // 需要调用`Model`的`initialize`方法
  24. parent::initialize();
  25. }
  26. /**
  27. * 文档详情
  28. * @param int $aid 文档ID
  29. */
  30. public function getArchivesView($aid = '', $users = [])
  31. {
  32. $aid = intval($aid);
  33. $args = [$aid, $users];
  34. $cacheKey = 'api-'.md5(__CLASS__.__FUNCTION__.json_encode($args));
  35. $result = cache($cacheKey);
  36. if (true || empty($result)) {
  37. $detail = $this->getViewInfo($aid, $users);
  38. if (!empty($detail['detail'])) {
  39. if (0 <= $detail['detail']['arcrank']) { // 待审核稿件
  40. $detail['detail']['title'] = htmlspecialchars_decode($detail['detail']['title']);
  41. $add_time = $detail['detail']['add_time'];
  42. $detail['detail']['add_time_format'] = $this->time_format($add_time);
  43. $detail['detail']['add_time'] = date('Y-m-d H:i:s', $add_time); // 格式化发布时间,兼容早期开源小程序
  44. $detail['detail']['content'] = $this->html_httpimgurl($detail['detail']['content'], true); // 转换内容图片为http路径
  45. !empty($detail['product']) && $result['product'] = $detail['product']; // 推荐商品
  46. !empty($detail['coupon_list']) && $result['coupon_list'] = $detail['coupon_list']; // 优惠券列表
  47. } else {
  48. $detail['detail'] = [
  49. 'arcrank' => $detail['detail']['arcrank'],
  50. ];
  51. }
  52. }
  53. $result['detail']['data'] = !empty($detail['detail']) ? $detail['detail'] : false;
  54. cache($cacheKey, $result, null, 'archives');
  55. }
  56. if (!empty($result['detail']['data'])) {
  57. // 浏览量
  58. Db::name('archives')->where(['aid'=>$aid])->setInc('click');
  59. $result['detail']['data']['click'] += 1;
  60. eyou_statistics_data(1);
  61. }
  62. return $result;
  63. }
  64. /**
  65. * 获取单条文档记录
  66. * @author wengxianhu by 2017-7-26
  67. */
  68. private function getViewInfo($aid, $users = [])
  69. {
  70. $result = [];
  71. $detail = Db::name('archives')
  72. ->alias('a')
  73. ->field('a.aid,a.typeid,a.channel,a.title,a.litpic,a.author,a.click,a.arcrank,a.seo_title,a.seo_keywords,a.seo_description,a.users_price,a.users_discount_type,a.users_free,a.old_price,
  74. a.sales_num,a.virtual_sales,a.sales_all,a.stock_show,a.stock_count,a.prom_type,a.arc_level_id,a.downcount,a.add_time,a.attrlist_id,b.typename,a.restric_type,a.no_vip_pay')
  75. ->join('arctype b','a.typeid = b.id','left')
  76. ->where([
  77. 'a.aid' => $aid,
  78. 'a.status' => 1,
  79. 'a.is_del' => 0,
  80. ])
  81. ->find();
  82. if (!empty($detail)) {
  83. $detail['arc_level_value'] = 0;
  84. if (0 < $detail['arc_level_id']) {
  85. $detail['arc_level_value'] = Db::name('users_level')->where(['level_id'=>$detail['arc_level_id']])->value('level_value');
  86. }
  87. // 模型标题处理
  88. $channeltype_row = \think\Cache::get('extra_global_channeltype');
  89. $channeltypeInfo = !empty($channeltype_row[$detail['channel']]) ? $channeltype_row[$detail['channel']] : [];
  90. $detail['channel_ntitle'] = !empty($channeltypeInfo['ntitle']) ? $channeltypeInfo['ntitle'] : '文章';
  91. $detail['seo_title'] = $this->set_arcseotitle($detail['title'], $detail['seo_title']); // seo标题
  92. $detail['litpic'] = $this->get_default_pic($detail['litpic']); // 默认封面图
  93. // $detail['forward'] = Db::name('users_forward')->where('aid',$aid)->count(); // 转发记录,废弃了,有需要就通过请求传参进行判断是否要查表返回数据,或者写成标签apiForward
  94. $detail['content'] = '';
  95. if (1 == $detail['channel']) { // 文章模型
  96. unset($detail['sales_num']);
  97. unset($detail['virtual_sales']);
  98. unset($detail['sales_all']);
  99. unset($detail['stock_show']);
  100. unset($detail['stock_count']);
  101. unset($detail['prom_type']);
  102. unset($detail['downcount']);
  103. } else if (2 == $detail['channel']) { // 产品模型
  104. //检测是否安装秒杀插件
  105. if (is_dir('./weapp/Seckill/')) {
  106. $SeckillRow = model('Weapp')->getWeappList('Seckill');
  107. if (!empty($SeckillRow) && 1 != intval($SeckillRow['status'])) {
  108. $detail['seckill_goods_id'] = false;
  109. }else{
  110. $tagWeappSeckill = new \think\template\taglib\api\TagWeappSeckill;
  111. $detail['seckill_goods_id'] = $tagWeappSeckill->guideSeckill($aid);
  112. }
  113. } else {
  114. $detail['seckill_goods_id'] = false;
  115. }
  116. // 是否安装积分商城插件
  117. $detail['points_goods_id'] = false;
  118. $weappInfo = model('ShopPublicHandle')->getWeappPointsShop();
  119. if (!empty($weappInfo)) {
  120. // 调用积分商城逻辑层方法
  121. $pointsGoodsModel = new \app\plugins\model\PointsGoods();
  122. $detail['points_goods_id'] = $pointsGoodsModel->getPointsShopGoodsID($aid);
  123. }
  124. unset($detail['users_free']);
  125. unset($detail['downcount']);
  126. /*产品参数*/
  127. if (!empty($detail['attrlist_id'])) { // 新版参数
  128. $productAttrModel = new \app\home\model\ProductAttr;
  129. $attr_list = $productAttrModel->getProAttrNew($aid, 'a.attr_id,a.attr_name,b.attr_value,b.aid');
  130. } else { // 旧版参数
  131. $productAttrModel = new \app\home\model\ProductAttr;
  132. $attr_list = $productAttrModel->getProAttr($aid);
  133. }
  134. $attr_list = !empty($attr_list[$aid]) ? $attr_list[$aid] : [];
  135. foreach ($attr_list as $key => $val) {
  136. $val['attr_value'] = htmlspecialchars_decode($val['attr_value']);
  137. unset($val['aid']);
  138. $attr_list[$key] = $val;
  139. }
  140. $detail['attr_list'] = !empty($attr_list) ? $attr_list : false; //产品默认参数
  141. /*规格数据*/
  142. $detail['spec_attr'] = $this->getSpecAttr($aid, $users);
  143. /* END */
  144. // 产品相册
  145. $productImgModel = new \app\home\model\ProductImg;
  146. $image_list = $productImgModel->getProImg($aid, 'aid,image_url,intro');
  147. $image_list = !empty($image_list[$aid]) ? $image_list[$aid] : [];
  148. foreach ($image_list as $key => $val) {
  149. $val['image_url'] = $this->get_default_pic($val['image_url']);
  150. isset($val['intro']) && $val['intro'] = htmlspecialchars_decode($val['intro']);
  151. $image_list[$key] = $val;
  152. }
  153. $detail['image_list'] = !empty($image_list) ? $image_list : false;
  154. /*可控制的主表字段列表*/
  155. $detail['ifcontrolRow'] = Db::name('channelfield')->field('id,name')->where([
  156. 'channel_id' => $detail['channel'],
  157. 'ifmain' => 1,
  158. 'ifeditable' => 1,
  159. 'ifcontrol' => 0,
  160. 'status' => 1,
  161. ])->getAllWithIndex('name');
  162. // 设置默认原价
  163. $detail['old_price'] = $detail['users_price'];
  164. $detail['product_num'] = 1;
  165. $detail['spec_value_id'] = '';
  166. $result['product'] = $this->getRecomProduct();
  167. $result['coupon_list'] = $this->getCoupon($detail, $users);
  168. if ('v1.5.1' < getVersion()) {
  169. //总评论数
  170. $detail['comment_data_count'] = Db::name('shop_order_comment')->where(['product_id' => $aid, 'is_show' => 1])->count();
  171. $good_count = Db::name('shop_order_comment')->where(['product_id' => $aid, 'is_show' => 1, 'total_score' => ['egt', 4]])->count();
  172. //好评率
  173. $detail['comment_good_count'] = $good_count;
  174. $detail['comment_good_per'] = !empty($detail['comment_data_count']) ? round($good_count / $detail['comment_data_count'], 2) : 1;
  175. if ($detail['comment_data_count'] > 0) {
  176. $detail['comment_data'] = Db::name('shop_order_comment')
  177. ->alias('a')
  178. ->field('a.*,b.nickname,b.head_pic')
  179. ->join('users b', 'a.users_id = b.users_id', 'left')
  180. ->where(['a.product_id' => $aid, 'a.is_show' => 1])
  181. ->order('a.total_score asc')
  182. ->limit(2)
  183. ->select();
  184. foreach ($detail['comment_data'] as $k => $v) {
  185. if (1 == $v['is_anonymous']) {
  186. $v['nickname'] = '匿名用户';
  187. }
  188. $v['add_time'] = date('Y-m-d H:i:s', $v['add_time']);
  189. $v['content'] = unserialize($v['content']);
  190. $v['head_pic'] = get_default_pic($v['head_pic'], true);
  191. $detail['comment_data'][$k] = $v;
  192. }
  193. }
  194. }
  195. $detail['cart_total_num'] = 0;
  196. if (!empty($users['users_id'])) {
  197. //购物车数量
  198. $detail['cart_total_num'] = Db::name('shop_cart')->where(['users_id' => $users['users_id']])->sum('product_num');
  199. }
  200. } else if (3 == $detail['channel']) { // 图集模型
  201. unset($detail['users_price']);
  202. unset($detail['users_free']);
  203. unset($detail['old_price']);
  204. unset($detail['sales_num']);
  205. unset($detail['virtual_sales']);
  206. unset($detail['sales_all']);
  207. unset($detail['stock_show']);
  208. unset($detail['stock_count']);
  209. unset($detail['prom_type']);
  210. unset($detail['downcount']);
  211. // 图集相册
  212. $imagesUploadModel = new \app\home\model\ImagesUpload;
  213. $image_list = $imagesUploadModel->getImgUpload($aid, 'aid,image_url,intro');
  214. $image_list = !empty($image_list[$aid]) ? $image_list[$aid] : [];
  215. foreach ($image_list as $key => $val) {
  216. $val['image_url'] = $this->get_default_pic($val['image_url']);
  217. isset($val['intro']) && $val['intro'] = htmlspecialchars_decode($val['intro']);
  218. $image_list[$key] = $val;
  219. }
  220. $detail['image_list'] = !empty($image_list) ? $image_list : false;
  221. } else if (4 == $detail['channel']) { // 下载模型
  222. unset($detail['users_price']);
  223. unset($detail['users_free']);
  224. unset($detail['old_price']);
  225. unset($detail['sales_num']);
  226. unset($detail['virtual_sales']);
  227. unset($detail['sales_all']);
  228. unset($detail['stock_show']);
  229. unset($detail['stock_count']);
  230. unset($detail['prom_type']);
  231. if (0 == $detail['arc_level_id']) {
  232. $downloadFileModel = new \app\home\model\DownloadFile;
  233. $down_list = $downloadFileModel->getDownFile($aid);
  234. $down_list = !empty($down_list[$aid]) ? $down_list[$aid] : [];
  235. foreach ($down_list as $key => $val) {
  236. $val['file_url'] = handle_subdir_pic($val['file_url']);
  237. $down_list[$key] = $val;
  238. }
  239. } else {
  240. $down_list = false;
  241. }
  242. $detail['down_list'] = !empty($down_list) ? $down_list : false;
  243. } else if (5 == $detail['channel']) { // 视频模型
  244. unset($detail['sales_num']);
  245. unset($detail['virtual_sales']);
  246. unset($detail['sales_all']);
  247. unset($detail['stock_show']);
  248. unset($detail['stock_count']);
  249. unset($detail['prom_type']);
  250. unset($detail['downcount']);
  251. // 视频附件列表
  252. $mediaFileModel = new \app\home\model\MediaFile;
  253. $media_list = $mediaFileModel->getMediaFile($aid);
  254. foreach ($media_list as $key => $val) {
  255. $val['file_url'] = handle_subdir_pic($val['file_url'], 'media');
  256. $media_list[$key] = $val;
  257. }
  258. $detail['media_list'] = !empty($media_list) ? $media_list : false;
  259. //视频权限
  260. $media_info = [
  261. 'auth_type' => 1, // 所有人免费
  262. 'level_auth' => 1, // 1-会员级别达到要求 2-需要升级会员级别
  263. 'buy_auth' => 1, // 1-已购买 2-需要购买
  264. 'level_name' => '', // 会员等级名称
  265. ];
  266. if (0 < $detail['users_price'] && empty($detail['users_free'])) {
  267. if (empty($detail['arc_level_id'])) {
  268. //不限会员 付费
  269. $media_info['auth_type'] = 2;
  270. $media_info['buy_auth'] = 2;
  271. } else {
  272. //3-限制会员 付费
  273. $media_info['auth_type'] = 3;
  274. $media_info['level_auth'] = 2;
  275. $media_info['buy_auth'] = 2;
  276. }
  277. } else {
  278. if (0 < intval($detail['arc_level_id'])) { // 会员免费
  279. $media_info['auth_type'] = 4;
  280. $media_info['level_auth'] = 2;
  281. $media_info['buy_auth'] = 2;
  282. }
  283. }
  284. //指定会员情况下开启非会员付费
  285. if (!empty($detail['no_vip_pay'])){
  286. $media_info['buy_auth'] = 2;
  287. }
  288. if (in_array($media_info['auth_type'],[3,4])){
  289. $media_info['level_name'] = Db::name('users_level')->where('level_id',$detail['arc_level_id'])->value('level_name');
  290. }
  291. //如果权限里涉及会员的 判断一下用户是否达到会员权限
  292. if (1 < $media_info['level_auth'] && !empty($users['users_id'])){
  293. if ($detail['arc_level_value'] <= $users['level_value']){
  294. $media_info['level_auth'] = 1;
  295. }
  296. }
  297. //如果权限里涉及购买的 判断一下用户是否已经购买
  298. if (1 < $media_info['buy_auth'] && !empty($users['users_id'])){
  299. $where = [
  300. 'users_id' => $users['users_id'],
  301. 'product_id' => $aid,
  302. 'order_status' => 1
  303. ];
  304. // 存在数据则已付费
  305. $is_buy = Db::name('media_order')->where($where)->count();
  306. if (!empty($is_buy)){
  307. $media_info['buy_auth'] = 1;
  308. }
  309. }
  310. $detail['media_info'] = $media_info;
  311. }
  312. else {
  313. unset($detail['users_price']);
  314. unset($detail['users_free']);
  315. unset($detail['old_price']);
  316. unset($detail['sales_num']);
  317. unset($detail['virtual_sales']);
  318. unset($detail['sales_all']);
  319. unset($detail['stock_show']);
  320. unset($detail['stock_count']);
  321. unset($detail['prom_type']);
  322. unset($detail['downcount']);
  323. }
  324. // 获取自定义字段的数据
  325. $customField = [];
  326. $addtableName = $channeltypeInfo['table'].'_content';
  327. $detailRow = Db::name($addtableName)->field('id,aid,add_time,update_time', true)->where('aid', $aid)->find();
  328. if (!empty($detailRow)) {
  329. $fieldLogic = new \app\home\logic\FieldLogic();
  330. $detailExt = $fieldLogic->getChannelFieldList($detailRow, $detail['channel']); // 自定义字段的数据格式处理
  331. if (empty($channeltypeInfo['ifsystem'])) { // 自定义模型
  332. // 如果存在自定义字段content,默认作为文档的正文内容。
  333. // 如果不存在,将获取第一个html类型的内容,作为文档的正文内容。
  334. if (!isset($detailExt['content'])) {
  335. $contentField = Db::name('channelfield')->where([
  336. 'channel_id' => $detail['channel'],
  337. 'dtype' => 'htmltext',
  338. ])->getField('name');
  339. $detailExt['content'] = !empty($detailExt[$contentField]) ? $detailExt[$contentField] : '';
  340. }
  341. }
  342. $detail['content'] = $detailExt['content'];
  343. unset($detailExt['content']);
  344. // 手机端详情内容
  345. if (isset($detailExt['content_ey_m'])) {
  346. $detail['content'] = empty($detailExt['content_ey_m']) ? $detail['content'] : $detailExt['content_ey_m'];
  347. unset($detailExt['content_ey_m']);
  348. }
  349. if (!empty($detailExt)) {
  350. $field = 'name, title, dtype';
  351. $customField = Db::name('channelfield')->field($field)->where([
  352. 'name' => ['IN', array_keys($detailExt)],
  353. 'channel_id' => $detail['channel'],
  354. 'ifeditable' => 1
  355. ])->getAllWithIndex('name');
  356. if (!empty($customField)) {
  357. foreach ($customField as $key => $value) {
  358. if ('img' == $value['dtype']) {
  359. $customField[$key]['value'] = $this->get_default_pic($detailExt[$key]);
  360. } else if ('media' == $value['dtype']) {
  361. $customField[$key]['value'] = $this->get_default_pic($detailExt[$key]);
  362. } else if ('imgs' == $value['dtype']) {
  363. foreach ($detailExt[$key] as $kk => $vv) {
  364. $detailExt[$key][$kk]['image_url'] = $this->get_default_pic($vv['image_url']);
  365. }
  366. $customField[$key]['value'] = $detailExt[$key];
  367. } else {
  368. $customField[$key]['value'] = $detailExt[$key];
  369. }
  370. }
  371. }
  372. }
  373. $customField = array_values($customField);
  374. }
  375. $detail['customField'] = !empty($customField) ? $customField : false;
  376. $detail['users_discount_price'] = !empty($detail['users_price']) ? $detail['users_price'] : 0;
  377. if (!empty($detail['users_price']) && !empty($users['users_id']) && 0 === intval($detail['users_discount_type'])) {
  378. $discount = Db::name('users_level')->where('level_id',$users['level'])->value('discount');
  379. if (100 > $discount) {
  380. $detail['users_discount_price'] = sprintf("%.2f",$detail['users_price']*($discount/100));
  381. }
  382. }
  383. // 商品是 单规格 且 设置会员折扣价 则处理
  384. if (empty($detail['spec_attr']['spec_data']) && 1 === intval($detail['users_discount_type']) && !empty($users['level_id'])) {
  385. $detail['users_discount_price'] = model('ShopPublicHandle')->handleUsersDiscountPrice($aid, $users['level_id']);
  386. }
  387. $result['detail'] = $detail;
  388. }
  389. return $result;
  390. }
  391. /**
  392. * 留言栏目表单
  393. * @param int $typeid 栏目ID
  394. */
  395. public function getGuestbookForm($typeid)
  396. {
  397. $typeid = intval($typeid);
  398. if (empty($typeid)) {
  399. $typeid = Db::name('arctype')->where([
  400. 'current_channel' => 8,
  401. 'is_del' => 0,
  402. 'status' => 1,
  403. 'lang' => parent::$lang,
  404. ])->getField('id');
  405. }
  406. $attr_list = array();
  407. $typename = '';
  408. if (0 < $typeid) {
  409. $detail = Db::name('arctype')->field('id,id as typeid,typename,seo_title,seo_keywords,seo_description')->where([
  410. 'id' => $typeid,
  411. 'lang' => parent::$lang,
  412. ])->find();
  413. $detail['seo_title'] = $this->set_arcseotitle($detail['typename'], $detail['seo_title']);
  414. $attr_list = Db::name('GuestbookAttribute')->field('attr_id,attr_name,attr_input_type,attr_values')
  415. ->where([
  416. 'typeid' => $typeid,
  417. 'is_del' => 0,
  418. ])
  419. ->order('sort_order asc, attr_id asc')
  420. ->select();
  421. foreach ($attr_list as $key => $val) {
  422. if (in_array($val['attr_input_type'], array(1,3,4))) {
  423. $val['attr_values'] = explode(PHP_EOL, $val['attr_values']);
  424. $attr_list[$key] = $val;
  425. }
  426. }
  427. }
  428. /*表单令牌*/
  429. $token_name = md5('guestbookform_token_'.$typeid.md5(getTime().uniqid(mt_rand(), TRUE)));
  430. $token_value = md5($_SERVER['REQUEST_TIME_FLOAT']);
  431. $session_path = \think\Config::get('session.path');
  432. $session_file = ROOT_PATH . $session_path . "/sess_".$token_name;
  433. $fp = fopen($session_file, "w+");
  434. if (!empty($fp)) {
  435. if (fwrite($fp, $token_value)) {
  436. fclose($fp);
  437. }
  438. } else {
  439. file_put_contents ( $session_file, $token_value);
  440. }
  441. /*end*/
  442. $result = array(
  443. 'detail' => $detail,
  444. 'attr_list' => $attr_list,
  445. 'token' => [
  446. 'name' => '__token__'.$token_name,
  447. 'value' => $token_value,
  448. ],
  449. );
  450. return $result;
  451. }
  452. /**
  453. * 给指定留言添加表单值到 guestbook_attr
  454. * @param int $aid 留言id
  455. * @param int $typeid 留言栏目id
  456. */
  457. public function saveGuestbookAttr($post, $aid, $typeid, $form_type)
  458. {
  459. $attrArr = [];
  460. foreach ($post as $k => $v) {
  461. if (!strstr($k, 'attr_')) {
  462. continue;
  463. }
  464. $attr_id = str_replace('attr_', '', $k);
  465. is_array($v) && $v = implode(PHP_EOL, $v);
  466. $v = trim($v);
  467. $adddata = array(
  468. 'aid' => $aid,
  469. 'form_type' => $form_type,
  470. 'attr_id' => $attr_id,
  471. 'attr_value' => $v,
  472. 'lang' => parent::$lang,
  473. 'add_time' => getTime(),
  474. 'update_time' => getTime(),
  475. );
  476. Db::name('GuestbookAttr')->add($adddata);
  477. }
  478. }
  479. // 获取指定商品的规格数据
  480. public function getSpecAttr($aid = null, $users = [], $order2 = '')
  481. {
  482. $ReturnData = $SpecData = $SpecValue = $SelectSpecData = [];
  483. if (empty($aid)) return $ReturnData;
  484. if (empty($users['level_discount'])) $users['level_discount'] = 100;
  485. // 会员折扣率
  486. $SelectSpecData['users_discount'] = (intval($users['level_discount']) / 100);
  487. $SpecWhere = [
  488. 'aid' => $aid,
  489. 'lang' => get_home_lang(),
  490. 'spec_is_select' => 1,
  491. ];
  492. $order = 'spec_value_id asc, spec_id asc';
  493. $field = '*, false checked';
  494. $ProductSpecData = Db::name('product_spec_data')->field($field)->where($SpecWhere)->order($order)->select();
  495. foreach ($ProductSpecData as $key => $value) {
  496. $ProductSpecData[$key]['spec_image'] = !empty($value['spec_image']) ? get_default_pic($value['spec_image'], true) : '';
  497. }
  498. if (!empty($ProductSpecData)) {
  499. $ProductSpecData = group_same_key($ProductSpecData, 'spec_mark_id');
  500. foreach ($ProductSpecData as $key => $value) {
  501. // $value[0]['checked'] = true;
  502. $SpecData[] = [
  503. 'spec_value_id' => $value[0]['spec_value_id'],
  504. 'spec_mark_id' => $value[0]['spec_mark_id'],
  505. 'spec_name' => $value[0]['spec_name'],
  506. 'spec_data_new' => $value,
  507. ];
  508. }
  509. unset($SpecWhere['spec_is_select']);
  510. $order2 = !empty($order2) ? $order2 : 'seckill_price asc,spec_price asc';
  511. $ProductSpecValue = Db::name('product_spec_value')->where($SpecWhere)->order($order2)->select();
  512. if (!empty($ProductSpecValue)) {
  513. // 默认的规格值,取价格最低者
  514. $SelectSpecData['spec_value_id'] = explode('_', $ProductSpecValue[0]['spec_value_id']);
  515. // 若存在规格并且价格存在则覆盖原有价格
  516. $SelectSpecData['users_price'] = sprintf("%.2f", strval($ProductSpecValue[0]['spec_price']) * strval($SelectSpecData['users_discount']));
  517. // 产品原价
  518. $SelectSpecData['old_price'] = $ProductSpecValue[0]['spec_price'];
  519. // 若存在规格并且库存存在则覆盖原有库存
  520. $SelectSpecData['stock_count'] = $ProductSpecValue[0]['spec_stock'];
  521. // 价格及库存
  522. $SpecValue = $ProductSpecValue;
  523. }
  524. foreach ($SpecData as $key => $value) {
  525. foreach ($value['spec_data_new'] as $kk => $vv) {
  526. // 追加默认规格class
  527. if (in_array($vv['spec_value_id'], $SelectSpecData['spec_value_id'])) {
  528. $SpecData[$key]['spec_data_new'][$kk]['checked'] = true;
  529. }
  530. }
  531. }
  532. }
  533. $ReturnData = [
  534. 'spec_data' => $SpecData,
  535. 'spec_value' => $SpecValue,
  536. 'select_spec_data' => $SelectSpecData
  537. ];
  538. return $ReturnData;
  539. }
  540. /**
  541. * 购物车/用户中心/商品详情获取 可能你还想要
  542. */
  543. public function getRecomProduct($type = 1){
  544. // 查询条件
  545. $where = [
  546. 'channel' => 2,
  547. 'arcrank' => ['>', -1],
  548. 'lang' => self::$lang,
  549. 'status' => 1,
  550. 'is_del' => 0,
  551. ];
  552. // 数据排序
  553. if (1 == $type){
  554. //最新
  555. $order = [
  556. 'aid' => 'desc',
  557. ];
  558. }elseif (2 == $type){
  559. //推荐
  560. $where['is_recom'] = 1;
  561. $order = [
  562. 'sort_order' => 'asc',
  563. 'aid' => 'desc',
  564. ];
  565. }elseif (3 == $type){
  566. //最热
  567. $order = [
  568. 'click' => 'desc',
  569. 'aid' => 'desc',
  570. ];
  571. }else {
  572. $order = [
  573. 'sort_order' => 'asc',
  574. 'aid' => 'desc',
  575. ];
  576. }
  577. // 分页处理
  578. $limit ='0,6';
  579. // 查询数据
  580. $ArchivesData = Db::name('archives')->where($where)->order($order)->limit($limit)->select();
  581. if (!empty($ArchivesData)) {
  582. foreach ($ArchivesData as $key => $value) {
  583. $ArchivesData[$key]['litpic'] = $this->get_default_pic($value['litpic'], true); // 默认封面图
  584. }
  585. } else {
  586. $ArchivesData = [];
  587. }
  588. return [
  589. 'title' => '可能你还想要',
  590. 'list' => $ArchivesData,
  591. ];
  592. }
  593. //获取当前商品可用优惠券
  594. public function getCoupon($arcData = [],$users = [])
  595. {
  596. // 筛选条件
  597. $filter = [
  598. 'status' => 1
  599. ];
  600. $filter['start_date'] = ['<=',getTime()];
  601. $filter['end_date'] = ['>=',getTime()];
  602. $result = Db::name('shop_coupon')
  603. ->where($filter)
  604. ->where("coupon_type = '1' OR FIND_IN_SET('{$arcData['typeid']}', arctype_id) OR FIND_IN_SET('{$arcData['aid']}', product_id)")
  605. ->order('sort_order asc,coupon_id desc')
  606. ->select();
  607. $coupon_ids = [];
  608. if (!empty($result)) {
  609. foreach ($result as $k => $v) {
  610. $coupon_ids[] = $v['coupon_id'];
  611. $v['coupon_form_name'] = '满减券';
  612. switch ($v['use_type']) {
  613. case '1': // 固定期限有效
  614. $v['start_date'] = date('Y/m/d', $v['start_date']);
  615. $v['end_date'] = date('Y/m/d', $v['end_date']);
  616. $v['use_type_name'] = $v['start_date'] . '-' . $v['end_date'];
  617. break;
  618. case '2'; // 领取当天开始有效
  619. $v['use_type_name'] = '领取' . $v['valid_days'] . '天内有效';
  620. break;
  621. case '3'; // 领取次日开始有效
  622. $v['use_type_name'] = '领取次日开始' . $v['valid_days'] . '天内有效';
  623. break;
  624. }
  625. switch ($v['coupon_type']) {
  626. case '1':
  627. $v['coupon_type_name'] = '全部商品';
  628. break;
  629. case '2';
  630. $v['coupon_type_name'] = '指定商品';
  631. break;
  632. case '3';
  633. $v['coupon_type_name'] = '指定分类';
  634. break;
  635. }
  636. $result[$k] = $v;
  637. }
  638. }
  639. if (!empty($users)){
  640. $have_where['users_id'] = $users['users_id'];
  641. if (!empty($coupon_ids)) {
  642. $have_where['coupon_id'] = ['in',$coupon_ids];
  643. }
  644. $have_where['use_status'] = 0;
  645. $have = Db::name('shop_coupon_use')->where($have_where)->column('coupon_id');
  646. if (!empty($have)){
  647. foreach ($result as $k => $v ) {
  648. if (in_array($v['coupon_id'],$have)){
  649. $result[$k]['geted'] = 1;//当前还有已领取未使用的
  650. }
  651. }
  652. }
  653. }
  654. return $result;
  655. }
  656. /**
  657. * 获取秒杀商品详情
  658. */
  659. public function GetSharpGoods($aid = 0){
  660. $users = [];
  661. $aid = intval($aid);
  662. $args = [$aid, $users];
  663. $cacheKey = 'api-'.md5(__CLASS__.__FUNCTION__.json_encode($args));
  664. $result = cache($cacheKey);
  665. if (true || empty($result)) {
  666. $status = 0;
  667. $msg = 'OK';
  668. $detail = array();
  669. if (0 < $aid) {
  670. $detail = $this->getSharpInfo($aid, $users);
  671. if (!empty($detail)) {
  672. if (0 == $detail['arcrank']) {
  673. $status = 1;
  674. } else if (0 > $detail['arcrank']) {
  675. $msg = '文档审核中,无权查看!';
  676. } else if (0 < $detail['arcrank']) {
  677. $msg = '当前是游客,无权查看!';
  678. }
  679. $detail['add_time'] = date('Y-m-d H:i:s', $detail['add_time']); // 格式化发布时间
  680. $detail['update_time'] = date('Y-m-d H:i:s', $detail['update_time']); // 格式化更新时间
  681. $detail['content_1586404997'] = $this->html_httpimgurl($detail['content_1586404997'], true); // 转换内容图片为http路径
  682. $detail['seo_description'] = '';
  683. } else {
  684. $msg = '文档已删除!';
  685. }
  686. }
  687. $result = [
  688. 'conf' => [
  689. 'status' => $status,
  690. 'msg' => $msg,
  691. ],
  692. 'detail' => !empty($detail) ? $detail : [],
  693. ];
  694. cache($cacheKey, $result, null, 'archives');
  695. }
  696. return $result;
  697. }
  698. /**
  699. * 获取秒杀商品记录
  700. * @author wengxianhu by 2017-7-26
  701. */
  702. private function getSharpInfo($aid,$users = [])
  703. {
  704. $where = [
  705. 'a.status' => 1,
  706. 'a.is_del' => 0,
  707. 'a.aid' => $aid,
  708. ];
  709. $result = array();
  710. $row = Db::name('sharp_goods')
  711. ->alias('a')
  712. ->field('a.sharp_goods_id,a.limit,a.seckill_stock,a.seckill_price,a.sales,a.virtual_sales,a.is_sku,b.sales_actual,c.*')
  713. ->where($where)
  714. ->join('sharp_active_goods b','a.aid = b.aid','left')
  715. ->join('archives c','a.aid = c.aid','left')
  716. ->find();
  717. $channeltype_row = \think\Cache::get('extra_global_channeltype');
  718. if (!empty($row)) {
  719. if (0 < $row['virtual_sales']){
  720. $row['sales_actual'] = $row['sales_actual']+$row['virtual_sales'];
  721. }
  722. $row['seo_title'] = $this->set_arcseotitle($row['title'], $row['seo_title']);
  723. /*封面图*/
  724. if (empty($row['litpic'])) {
  725. $row['is_litpic'] = 0; // 无封面图
  726. } else {
  727. $row['is_litpic'] = 1; // 有封面图
  728. }
  729. $row['litpic'] = $this->get_default_pic($row['litpic'], true); // 默认封面图
  730. // 模型处理
  731. $channeltypeInfo = !empty($channeltype_row[$row['channel']]) ? $channeltype_row[$row['channel']] : [];
  732. /*产品参数*/
  733. if (!empty($row['attrlist_id'])){
  734. $productAttrModel = new \app\home\model\ProductAttr();
  735. $attr_list = $productAttrModel->getProAttrNew($aid);
  736. }else{
  737. $productAttrModel = new \app\home\model\ProductAttr();
  738. $attr_list = $productAttrModel->getProAttr($aid);
  739. }
  740. $attr_list = !empty($attr_list[$aid]) ? $attr_list[$aid] : [];
  741. foreach ($attr_list as $key => $val) {
  742. $attr_list[$key]['attr_value'] = htmlspecialchars_decode($val['attr_value']);
  743. }
  744. $row['attr_list'] = $attr_list;
  745. /*规格数据*/
  746. $row['spec_attr'] = $this->getSpecAttr($aid, $users);
  747. /* END */
  748. // 产品相册
  749. $image_list = [];
  750. $productImgModel = new \app\home\model\ProductImg();
  751. $image_list_tmp = $productImgModel->getProImg($aid);
  752. if (!empty($image_list_tmp[$aid])) {
  753. foreach ($image_list_tmp[$aid] as $key => $val) {
  754. $val['image_url'] = $this->get_default_pic($val['image_url'], true);
  755. $image_list[$key] = $val;
  756. }
  757. }
  758. $row['image_list'] = $image_list;
  759. /*可控制的主表字段列表*/
  760. $row['ifcontrolRow_1586404997'] = Db::name('channelfield')->field('id,name')->where([
  761. 'channel_id' => $row['channel'],
  762. 'ifmain' => 1,
  763. 'ifeditable' => 1,
  764. 'ifcontrol' => 0,
  765. 'status' => 1,
  766. ])->getAllWithIndex('name');
  767. // 设置默认原价
  768. $row['old_price'] = $row['users_price'];
  769. $row['product_num'] = 1;
  770. $row['spec_value_id'] = '';
  771. // 获取自定义字段的数据
  772. $class = '\app\home\model\\'.$channeltypeInfo['ctl_name'];
  773. $model = new $class();
  774. $extFields = Db::name($channeltypeInfo['table'].'_content')->getTableFields();
  775. $extFields = array_flip($extFields);
  776. unset($extFields['id']);
  777. $rowExt = $model->getInfo($aid);
  778. $rowExt_new = [];
  779. foreach ($extFields as $key => $val) {
  780. $rowExt_new[$key] = $rowExt[$key];
  781. }
  782. $rowExt = array_diff_key($rowExt_new, $row);
  783. if (!empty($rowExt)) {
  784. $fieldLogic = new \app\home\logic\FieldLogic();
  785. $rowExt = $fieldLogic->getChannelFieldList($rowExt, $row['channel']); // 自定义字段的数据格式处理
  786. }
  787. /*--end*/
  788. // 浏览量
  789. Db::name('archives')->where(['aid'=>$aid])->setInc('click');
  790. $row['click'] = $row['click'] + 1;
  791. $result = array_merge($rowExt, $row);
  792. if (!empty($channeltypeInfo['ifsystem']) && 1 == $channeltypeInfo['ifsystem']) {
  793. $content = $result['content'];
  794. unset($result['content']);
  795. } else {
  796. /*获取第一个html类型的内容,作为文档的内容*/
  797. $contentField = Db::name('channelfield')->where([
  798. 'channel_id' => $row['channel'],
  799. 'dtype' => 'htmltext',
  800. ])->getField('name');
  801. $content = !empty($rowExt[$contentField]) ? $rowExt[$contentField] : '';
  802. /*--end*/
  803. }
  804. $result['content_1586404997'] = $content;
  805. }
  806. return $result;
  807. }
  808. /**
  809. * 获取某个场次的秒杀信息
  810. * @param $item
  811. */
  812. public function getSharp($active_time_id=0,$aid=0)
  813. {
  814. //获取当时时间的整点时间戳
  815. $time = strtotime(date('Y-m-d'));
  816. $hour = date('H');
  817. $where['a.status'] = 1;
  818. $where['a.active_time_id'] = $active_time_id;
  819. $where['b.is_del'] = 0;
  820. $where['b.status'] = 1;
  821. //获取秒杀场次信息
  822. $active = Db::name('sharp_active_time')
  823. ->alias('a')
  824. ->where($where)
  825. ->field('a.active_time_id,a.active_id,a.active_time,b.active_date,c.sales_actual')
  826. ->join('sharp_active b','a.active_id = b.active_id')
  827. ->join('sharp_active_goods c','a.active_time_id = c.active_time_id')
  828. ->find();
  829. //正在进行时
  830. if ($active['active_date'] == $time && $active['active_time'] == $hour){
  831. $active['status'] = 10;
  832. } else if (($active['active_date'] == $time && $active['active_time'] > $hour) || $active['active_date'] > $time){
  833. //预告
  834. $active['status'] = 20;
  835. }else if (($active['active_date'] == $time && $active['active_time'] < $hour) || $active['active_date'] < $time){
  836. //过期
  837. $active['status'] = 30;
  838. }
  839. $date = date('Y-m-d'); //2020-12-15
  840. if ($active['active_time'] < 10){
  841. $active['active_time'] = '0'.$active['active_time'].':00';
  842. }else{
  843. $active['active_time'] = $active['active_time'].':00'; // 00:00
  844. }
  845. $data_time = $date.' '.$active['active_time']; //2020-15-2-15 15:00
  846. $time_plus = strtotime($data_time . "+1 hours"); //2020-15-2-15 15:00
  847. $time_plus = date('Y-m-d H:i',$time_plus); //2020-15-2-15 16:00
  848. if ( 10 == $active['status'] ){
  849. $arr = [
  850. $time_plus,$time_plus,$data_time];
  851. }else if ( 20 == $active['status'] ){
  852. $arr = [$data_time,$time_plus,$data_time];
  853. }else if ( 30 == $active['status'] ){
  854. $arr = [false,$time_plus,$data_time];
  855. }
  856. $active['count_down_time'] = $arr[0];
  857. $active['end_time'] = $arr[1];
  858. $active['start_time'] = $arr[2];
  859. return $active;
  860. }
  861. /**
  862. * 获取优惠券数量
  863. */
  864. public function getCouponCount($user)
  865. {
  866. if (false == $user){
  867. return 0;
  868. }else{
  869. $where['users_id'] = $user['users_id'];
  870. $where['use_status'] = 0;
  871. $where['end_time'] = ['>',getTime()];
  872. $count = Db::name('shop_coupon_use')->where($where)->count();
  873. return $count;
  874. }
  875. }
  876. /**
  877. * 获取评论列表
  878. */
  879. public function getGoodsCommentList($param = [] )
  880. {
  881. $page = empty($param['page']) ? 1: $param['page'];
  882. $type = empty($param['type']) ? 'all': $param['type'];
  883. $pagesize = empty($param['pagesize']) ? config('paginate.list_rows'): $param['pagesize'];
  884. $total_score = empty($param['total_score']) ? '': $param['total_score'];
  885. $goods_id = $param['aid'];
  886. $field='a.*,u.nickname,u.head_pic';
  887. // 筛选条件
  888. $condition = [
  889. 'a.product_id' => $goods_id,
  890. 'a.is_show' => 1,
  891. ];
  892. if (!empty($param['total_score'])) $condition['a.total_score'] = $param['total_score'];
  893. if ('img' == $type) $condition['a.upload_img'] = ['neq',''];
  894. $args = [$goods_id,$page,$pagesize,$total_score];
  895. $cacheKey = 'api-'.md5(__CLASS__.__FUNCTION__.json_encode($args));
  896. if (true || empty($result)) {
  897. $paginate = array(
  898. 'page' => $page,
  899. );
  900. $pages = Db::name('shop_order_comment')
  901. ->field($field)
  902. ->alias('a')
  903. ->join('users u','a.users_id = u.users_id','left')
  904. ->where($condition)
  905. ->order('a.add_time desc')
  906. ->paginate($pagesize, false, $paginate);
  907. $result = $pages->toArray();
  908. foreach ($result['data'] as $key => $val) {
  909. $val['head_pic'] = get_default_pic($val['head_pic'],true);
  910. if (isset($val['add_time'])) {
  911. $val['add_time'] = date('Y-m-d H:i:s', $val['add_time']);
  912. }
  913. if (!empty($val['upload_img'])){
  914. $val['upload_img'] = unserialize($val['upload_img']);
  915. $val['upload_img'] = explode(',',$val['upload_img']);
  916. foreach ($val['upload_img'] as $k => $v){
  917. $val['upload_img'][$k] = get_default_pic($v,true);
  918. }
  919. }
  920. if (!empty($val['content'])){
  921. $val['content'] = unserialize($val['content']);
  922. }
  923. $result['data'][$key] = $val;
  924. }
  925. $score_type = Db::name('shop_order_comment')
  926. ->where([
  927. 'product_id' => $goods_id,
  928. 'is_show' => 1,
  929. 'lang' => get_home_lang(),
  930. ])->field('count(*) as count,total_score')
  931. ->group('total_score')
  932. ->getAllWithIndex('total_score');
  933. $result['count']['goods'] = isset($score_type[1]) ? $score_type[1]['count'] : 0;
  934. $result['count']['middle'] = isset($score_type[2]) ? $score_type[2]['count'] : 0;
  935. $result['count']['bad'] = isset($score_type[3]) ? $score_type[3]['count'] : 0;
  936. $result['count']['all'] = $result['count']['goods']+$result['count']['middle']+$result['count']['bad'] ;
  937. cache($cacheKey, $result, null, 'getGoodsCommentList');
  938. $condition['a.upload_img'] = ['neq',''];
  939. $result['have_img_count'] = Db::name('shop_order_comment')->alias('a')->where($condition)->count();
  940. }
  941. return $result;
  942. }
  943. /**
  944. * 获取限时折扣商品详情
  945. */
  946. public function GetDiscountGoods($aid = 0,$users = []){
  947. $aid = intval($aid);
  948. $args = [$aid, $users];
  949. $cacheKey = 'api-'.md5(__CLASS__.__FUNCTION__.json_encode($args));
  950. $result = cache($cacheKey);
  951. if (true || empty($result)) {
  952. $detail = $this->getDiscountInfo($aid, $users);
  953. if (!empty($detail['detail'])) {
  954. if (0 <= $detail['detail']['arcrank']) { // 待审核稿件
  955. $detail['detail']['title'] = htmlspecialchars_decode($detail['detail']['title']);
  956. $detail['detail']['add_time'] = date('Y-m-d H:i:s', $detail['detail']['add_time']); // 格式化发布时间
  957. $detail['detail']['content'] = $this->html_httpimgurl($detail['detail']['content'], true); // 转换内容图片为http路径
  958. } else {
  959. $detail['detail'] = [
  960. 'arcrank' => $detail['detail']['arcrank'],
  961. ];
  962. }
  963. }
  964. $result = [
  965. 'detail' => [
  966. 'data' => !empty($detail['detail']) ? $detail['detail'] : false,
  967. ],
  968. 'product'=> !empty($detail['product']) ? $detail['product'] : [],
  969. 'coupon_list'=> !empty($detail['coupon_list']) ? $detail['coupon_list'] : [],
  970. ];
  971. cache($cacheKey, $result, null, 'archives');
  972. }
  973. return $result;
  974. }
  975. /**
  976. * 获取限时折扣商品记录
  977. */
  978. private function getDiscountInfo($aid,$users = [])
  979. {
  980. $where = [
  981. 'a.status' => 1,
  982. 'a.is_del' => 0,
  983. 'a.aid' => $aid,
  984. ];
  985. $result = [];
  986. $detail = Db::name('discount_goods')
  987. ->alias('a')
  988. ->field('a.discount_gid,a.discount_stock,a.discount_price,a.sales,a.virtual_sales,a.is_sku,b.sales_actual,c.*')
  989. ->where($where)
  990. ->join('discount_active_goods b','a.aid = b.aid','left')
  991. ->join('archives c','a.aid = c.aid','left')
  992. ->find();
  993. if (!empty($detail)) {
  994. // 模型标题处理
  995. $channeltype_row = \think\Cache::get('extra_global_channeltype');
  996. $channeltypeInfo = !empty($channeltype_row[$detail['channel']]) ? $channeltype_row[$detail['channel']] : [];
  997. $detail['channel_ntitle'] = !empty($channeltypeInfo['ntitle']) ? $channeltypeInfo['ntitle'] : '文章';
  998. $detail['seo_title'] = $this->set_arcseotitle($detail['title'], $detail['seo_title']); // seo标题
  999. $detail['litpic'] = $this->get_default_pic($detail['litpic']); // 默认封面图
  1000. $detail['content'] = '';
  1001. unset($detail['users_free']);
  1002. unset($detail['downcount']);
  1003. /*产品参数*/
  1004. if (!empty($detail['attrlist_id'])){ // 新版参数
  1005. $productAttrModel = new \app\home\model\ProductAttr;
  1006. $attr_list = $productAttrModel->getProAttrNew($aid, 'a.attr_id,a.attr_name,b.attr_value,b.aid');
  1007. }else{ // 旧版参数
  1008. $productAttrModel = new \app\home\model\ProductAttr;
  1009. $attr_list = $productAttrModel->getProAttr($aid);
  1010. }
  1011. $attr_list = !empty($attr_list[$aid]) ? $attr_list[$aid] : [];
  1012. foreach ($attr_list as $key => $val) {
  1013. $val['attr_value'] = htmlspecialchars_decode($val['attr_value']);
  1014. unset($val['aid']);
  1015. $attr_list[$key] = $val;
  1016. }
  1017. $detail['attr_list'] = !empty($attr_list) ? $attr_list : false;
  1018. /*规格数据*/
  1019. $detail['spec_attr'] = $this->getSpecAttr($aid, $users,'discount_price asc,spec_price asc');
  1020. /* END */
  1021. // 产品相册
  1022. $productImgModel = new \app\home\model\ProductImg;
  1023. $image_list = $productImgModel->getProImg($aid, 'aid,image_url,intro');
  1024. $image_list = !empty($image_list[$aid]) ? $image_list[$aid] : [];
  1025. foreach ($image_list as $key => $val) {
  1026. $val['image_url'] = $this->get_default_pic($val['image_url']);
  1027. isset($val['intro']) && $val['intro'] = htmlspecialchars_decode($val['intro']);
  1028. $image_list[$key] = $val;
  1029. }
  1030. $detail['image_list'] = !empty($image_list) ? $image_list : false;
  1031. /*可控制的主表字段列表*/
  1032. $detail['ifcontrolRow'] = Db::name('channelfield')->field('id,name')->where([
  1033. 'channel_id' => $detail['channel'],
  1034. 'ifmain' => 1,
  1035. 'ifeditable' => 1,
  1036. 'ifcontrol' => 0,
  1037. 'status' => 1,
  1038. ])->getAllWithIndex('name');
  1039. // 设置默认原价
  1040. $detail['old_price'] = $detail['users_price'];
  1041. $detail['product_num'] = 1;
  1042. $detail['spec_value_id'] = '';
  1043. $result['product'] = $this->getRecomProduct();
  1044. $result['coupon_list'] = $this->getCoupon($detail,$users);
  1045. if ('v1.5.1' < getVersion()) {
  1046. //总评论数
  1047. $detail['comment_data_count'] = Db::name('shop_order_comment')->where(['product_id'=>$aid,'is_show'=>1])->count();
  1048. $good_count = Db::name('shop_order_comment')->where(['product_id'=>$aid,'is_show'=>1,'total_score'=>1])->count();
  1049. //好评率
  1050. $detail['comment_good_per'] = !empty($detail['comment_data_count']) ? round($good_count/$detail['comment_data_count'],2)*100 : 0;
  1051. $detail['comment_good_per'] = $detail['comment_good_per'].'%';
  1052. if ($detail['comment_data_count'] > 0){
  1053. $detail['comment_data'] = Db::name('shop_order_comment')
  1054. ->alias('a')
  1055. ->field('a.*,b.nickname,b.head_pic')
  1056. ->join('users b','a.users_id = b.users_id')
  1057. ->where(['a.product_id'=>$aid,'a.is_show'=>1])
  1058. ->order('a.total_score asc')
  1059. ->limit(2)
  1060. ->select();
  1061. foreach ($detail['comment_data'] as $k => $v){
  1062. if (1 == $v['is_anonymous']){
  1063. $v['nickname'] = '匿名用户';
  1064. }
  1065. $v['add_time'] = date('Y-m-d H:i:s',$v['add_time']);
  1066. $v['content'] = unserialize($v['content']);
  1067. $v['head_pic'] = get_default_pic($v['head_pic'],true);
  1068. $detail['comment_data'][$k] = $v;
  1069. }
  1070. }
  1071. }
  1072. $detail['cart_total_num'] = 0;
  1073. if (!empty($users['users_id'])){
  1074. //购物车数量
  1075. $detail['cart_total_num'] = Db::name('shop_cart')->where(['users_id' => $users['users_id']])->sum('product_num');
  1076. }
  1077. // 获取自定义字段的数据
  1078. $customField = [];
  1079. $detailExt = Db::name($channeltypeInfo['table'].'_content')->field('id,aid,add_time,update_time', true)->where('aid', $aid)->find();
  1080. if (!empty($detailExt)) {
  1081. $fieldLogic = new \app\home\logic\FieldLogic();
  1082. $detailExt = $fieldLogic->getChannelFieldList($detailExt, $detail['channel']); // 自定义字段的数据格式处理
  1083. if (empty($channeltypeInfo['ifsystem'])) { // 自定义模型
  1084. // 如果存在自定义字段content,默认作为文档的正文内容。
  1085. // 如果不存在,将获取第一个html类型的内容,作为文档的正文内容。
  1086. if (!isset($detailExt['content'])) {
  1087. $contentField = Db::name('channelfield')->where([
  1088. 'channel_id' => $detail['channel'],
  1089. 'dtype' => 'htmltext',
  1090. ])->getField('name');
  1091. $detailExt['content'] = !empty($detailExt[$contentField]) ? $detailExt[$contentField] : '';
  1092. }
  1093. }
  1094. $detail['content'] = $detailExt['content'];
  1095. unset($detailExt['content']);
  1096. if (!empty($detailExt)) {
  1097. $field = 'name, title, dtype';
  1098. $customField = Db::name('channelfield')->field($field)->where([
  1099. 'name' => ['IN', array_keys($detailExt)],
  1100. 'channel_id' => $detail['channel'],
  1101. 'ifeditable' => 1
  1102. ])->getAllWithIndex('name');
  1103. if (!empty($customField)) {
  1104. foreach ($customField as $key => $value) {
  1105. if ('img' == $value['dtype']) {
  1106. $customField[$key]['value'] = $this->get_default_pic($detailExt[$key]);
  1107. } else if ('media' == $value['dtype']) {
  1108. $customField[$key]['value'] = $this->get_default_pic($detailExt[$key]);
  1109. } else if ('imgs' == $value['dtype']) {
  1110. foreach ($detailExt[$key] as $kk => $vv) {
  1111. $detailExt[$key][$kk]['image_url'] = $this->get_default_pic($vv['image_url']);
  1112. }
  1113. $customField[$key]['value'] = $detailExt[$key];
  1114. } else {
  1115. $customField[$key]['value'] = $detailExt[$key];
  1116. }
  1117. }
  1118. }
  1119. }
  1120. $customField = array_values($customField);
  1121. }
  1122. $detail['customField'] = !empty($customField) ? $customField : false;
  1123. // 浏览量
  1124. Db::name('archives')->where(['aid'=>$aid])->setInc('click');
  1125. $detail['click'] += 1;
  1126. $result['detail'] = $detail;
  1127. }
  1128. return $result;
  1129. }
  1130. //添加文章评论
  1131. public function addArticleComment($param = [],$users = [])
  1132. {
  1133. $users_level_id = !empty($users['level_id']) ? $users['level_id'] : 0;//0-游客
  1134. $comment_level_data = Db::name('weapp_comment_level')->where('users_level_id',$users_level_id)->find();
  1135. if (empty($comment_level_data['is_comment'])){
  1136. return ['code'=>0, 'msg'=>'您没有评论权限'];
  1137. }
  1138. if (!empty($comment_level_data['is_review'])){
  1139. $param['is_review'] = 0;
  1140. }else{
  1141. $param['is_review'] = 1;
  1142. }
  1143. $param['add_time'] = getTime();
  1144. $param['update_time'] = getTime();
  1145. $param['users_id'] = !empty($users['users_id']) ? $users['users_id'] : 0;
  1146. $param['username'] = !empty($users['username']) ? $users['username'] : '游客';
  1147. $param['provider'] = !empty($param['provider']) ? $param['provider'] : 'weixin';
  1148. $param['users_ip'] = clientIP();
  1149. if (!empty($param['content'])) {
  1150. $content_tmp = filterNickname($param['content']);
  1151. if ($content_tmp != $param['content']) {
  1152. return ['code'=>0, 'msg'=>'不支持表情等特殊符号'];
  1153. }
  1154. }
  1155. $comment_id = Db::name('weapp_comment')->insertGetId($param);
  1156. if (false !== $comment_id){
  1157. $data = ['code'=>1, 'is_show'=>0];
  1158. if (empty($param['is_review'])) {
  1159. $msg = '评论成功,进入待审核中';
  1160. } else {
  1161. $msg = '评论成功';
  1162. $comment = Db::name('weapp_comment')
  1163. ->alias('a')
  1164. ->field('a.*,b.head_pic,b.nickname,b.sex')
  1165. ->join('users b','a.users_id = b.users_id','left')
  1166. ->find($comment_id);
  1167. $comment['head_pic'] = $this->get_head_pic($comment['head_pic'], false, $comment['sex']);
  1168. $comment['add_time_format'] = $this->time_format($comment['add_time']);
  1169. $comment['add_time'] = date('Y-m-d', $comment['add_time']);
  1170. $data['comment'] = $comment;
  1171. $data['is_show'] = 1;
  1172. }
  1173. $data['msg'] = $msg;
  1174. return $data;
  1175. }
  1176. return ['code'=>0, 'msg'=>'添加评论失败'];
  1177. }
  1178. /**
  1179. * 索引页
  1180. */
  1181. public function getRepertory($page=1)
  1182. {
  1183. $paginate = array(
  1184. 'page' => $page,
  1185. );
  1186. $allow_release_channel = config('global.allow_release_channel');
  1187. $pages = Db::name('archives')->field('aid,typeid,title,channel,update_time')
  1188. ->where([
  1189. 'channel' => ['IN', $allow_release_channel],
  1190. 'arcrank' => ['EGT', 0],
  1191. 'is_del' => 0,
  1192. 'lang' => parent::$lang,
  1193. ])
  1194. ->order('update_time desc')
  1195. ->paginate(100, false, $paginate);
  1196. $list = array();
  1197. foreach ($pages->items() as $key => $val) {
  1198. $path = '';
  1199. if (1 == $val['channel']) {
  1200. $path = '/pages/archives/article/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1201. } else if (2 == $val['channel']) {
  1202. $path = '/pages/archives/product/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1203. } else if (3 == $val['channel']) {
  1204. $path = '/pages/archives/images/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1205. } else if (4 == $val['channel']) {
  1206. $path = '/pages/archives/download/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1207. } else if (5 == $val['channel']) {
  1208. $path = '/pages/archives/media/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1209. } else if (6 == $val['channel']) {
  1210. $path = '/pages/archives/single/view?typeid='.$val['typeid'];
  1211. } else if (7 == $val['channel']) {
  1212. $path = '/pages/archives/special/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1213. } else {
  1214. $path = '/pages/archives/custom/view?aid='.$val['aid'].'&typeid='.$val['typeid'];
  1215. }
  1216. $arr = [
  1217. 'title' => $val['title'],
  1218. 'path' => $path,
  1219. 'releaseDate' => MyDate('Y-m-d H:i:s', $val['update_time']),
  1220. ];
  1221. $list[] = $arr;
  1222. }
  1223. $result = array(
  1224. 'totalPage' => $pages->lastPage(),
  1225. 'current' => $page,
  1226. 'list' => $list,
  1227. );
  1228. return $result;
  1229. }
  1230. /**
  1231. * 发送微信小程序订阅消息
  1232. */
  1233. public function sendAppletsNotice($result_id, $send_scene = 0, $users = [])
  1234. {
  1235. $template_info = Db::name('applets_template')->where(['send_scene' => $send_scene])->find();
  1236. if (empty($template_info)) {
  1237. return ['code' => 0, 'msg' => '小程序消息模板不存在'];
  1238. } else if (empty($template_info['is_open'])) {
  1239. return ['code' => 0, 'msg' => '小程序消息模板没开启'];
  1240. }
  1241. $tokenData = get_weixin_access_token(true);
  1242. if (!empty($tokenData['code'])) {
  1243. $redata = $this->weixinTemplateKeywords($result_id, $template_info);
  1244. $openid = Db::name('wx_users')->where(['users_id' => $users['users_id']])->value('openid');
  1245. $post_data = [
  1246. 'template_id' => $template_info['template_id'],
  1247. 'page' => !empty($redata['pagepath']) ? $redata['pagepath'] : '/pages/index/index',
  1248. 'touser' => $openid,
  1249. 'data' => $redata['data'],
  1250. 'miniprogram_state' => 'formal',
  1251. 'lang' => 'zh_CN',
  1252. ];
  1253. $url = "https://api.weixin.qq.com/cgi-bin/message/subscribe/send?access_token=" . $tokenData['access_token'];
  1254. $response = httpRequest($url, 'POST', json_encode($post_data, JSON_UNESCAPED_UNICODE));
  1255. $params = json_decode($response, true);
  1256. if (isset($params['errcode']) && empty($params['errcode'])) {
  1257. return ['code' => 1, 'msg' => '发送成功'];
  1258. }
  1259. }
  1260. $msg = !empty($params['errmsg']) ? $params['errmsg'] : $tokenData['msg'];
  1261. return ['code' => 0, 'msg' => $msg];
  1262. }
  1263. private function weixinTemplateKeywords($result_id, $template_info)
  1264. {
  1265. $data = [];
  1266. $pagepath = '';
  1267. $tpl_data = json_decode($template_info['tpl_data'], true);
  1268. $keywordsList = [];
  1269. if (!empty($tpl_data['keywordsList'])) {
  1270. foreach ($tpl_data['keywordsList'] as $key => $val) {
  1271. $keywordsList[$val['rule'] . $val['kid']] = $val;
  1272. }
  1273. }
  1274. $send_scene = $template_info['send_scene'];
  1275. // 订单发货通知
  1276. if (7 == $send_scene) {
  1277. $pagepath = "/pages/order/detail?order_id={$result_id}";
  1278. $orderInfo = Db::name('shop_order')->where(['order_id' => $result_id])->find();
  1279. $orderDetailsInfo = Db::name('shop_order_details')->where(['order_id' => $orderInfo['order_id']])->order('details_id asc')->select();
  1280. $product_name_new = msubstr($orderDetailsInfo[0]['product_name'], 0, 15);
  1281. if ($product_name_new == $orderDetailsInfo[0]['product_name'] && count($orderDetailsInfo) == 1) {
  1282. $product_name = $product_name_new;
  1283. } else {
  1284. $product_name = $product_name_new . '等...';
  1285. }
  1286. $data = [
  1287. 'character_string1' => [
  1288. 'value' => $orderInfo['order_code'],
  1289. ],
  1290. 'thing2' => [
  1291. 'value' => $product_name,
  1292. ],
  1293. 'thing7' => [
  1294. 'value' => $orderInfo['express_name'],
  1295. ],
  1296. 'character_string4' => [
  1297. 'value' => $orderInfo['express_order'],
  1298. ],
  1299. 'date5' => [
  1300. 'value' => MyDate('Y-m-d H:i:s', $orderInfo['express_time']),
  1301. ],
  1302. ];
  1303. }
  1304. return [
  1305. 'data' => $data,
  1306. 'pagepath' => $pagepath,
  1307. ];
  1308. }
  1309. /**
  1310. * 发送微信公众号模板消息
  1311. */
  1312. public function sendWechatNotice($result_id, $send_scene = 0, $users = [])
  1313. {
  1314. // 用户关注公众号则发送
  1315. if (!empty($users['wechat_followed']) && !empty($users['wechat_open_id'])) {
  1316. // 存在指定模板并已开启则发送
  1317. $template_info = Db::name('wechat_template')->where(['send_scene' => $send_scene])->find();
  1318. if (!empty($result_id) && !empty($template_info['is_open'])) {
  1319. $tokenData = get_wechat_access_token();
  1320. if (!empty($tokenData['code'])) {
  1321. $redata = $this->wechatTemplateKeywords($result_id, $template_info);
  1322. $post_data = [
  1323. 'template_id' => $template_info['template_id'],
  1324. 'touser' => $users['wechat_open_id'],
  1325. 'data' => $redata['data'],
  1326. 'client_msg_id' => md5(json_encode($redata)),
  1327. ];
  1328. $web_status = tpCache('web.web_status');
  1329. if (!empty($web_status)) {
  1330. $post_data['url'] = !empty($redata['gourl']) ? $redata['gourl'] : request()->domain() . $this->root_dir;
  1331. }
  1332. $url = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=" . $tokenData['access_token'];
  1333. httpRequest($url, 'POST', json_encode($post_data, JSON_UNESCAPED_UNICODE));
  1334. }
  1335. }
  1336. }
  1337. }
  1338. private function wechatTemplateKeywords($result_id, $template_info)
  1339. {
  1340. $data = [];
  1341. $gourl = $pagepath = "";
  1342. $tpl_data = json_decode($template_info['tpl_data'], true);
  1343. $keywordsList = [];
  1344. if (!empty($tpl_data['keywordsList'])) {
  1345. foreach ($tpl_data['keywordsList'] as $key => $val) {
  1346. $keywordsList[$val['rule']] = $val;
  1347. }
  1348. }
  1349. $send_scene = $template_info['send_scene'];
  1350. // 订单支付成功通知
  1351. if (9 == $send_scene) {
  1352. $pagepath = "/pages/order/detail?order_id={$result_id}";
  1353. $gourl = url('user/Shop/shop_order_details', ['order_id' => $result_id]);
  1354. $orderInfo = Db::name('shop_order')->where(['order_id' => $result_id])->find();
  1355. $orderDetailsInfo = Db::name('shop_order_details')->where(['order_id' => $orderInfo['order_id']])->order('details_id asc')->select();
  1356. $product_name_new = msubstr($orderDetailsInfo[0]['product_name'], 0, 22);
  1357. if ($product_name_new == $orderDetailsInfo[0]['product_name'] && count($orderDetailsInfo) == 1) {
  1358. $product_name = $product_name_new;
  1359. } else {
  1360. $product_name = $product_name_new . '等...';
  1361. }
  1362. $data = [
  1363. 'first' => [
  1364. 'value' => $template_info['template_title'],
  1365. ],
  1366. 'keyword1' => [
  1367. 'value' => $orderInfo['order_code'],
  1368. ],
  1369. 'keyword2' => [
  1370. 'value' => $product_name,
  1371. ],
  1372. 'keyword3' => [
  1373. 'value' => count($orderDetailsInfo),
  1374. ],
  1375. 'keyword4' => [
  1376. 'value' => "¥{$orderInfo['order_amount']}",
  1377. ],
  1378. 'keyword5' => [
  1379. 'value' => MyDate('Y-m-d H:i:s', $orderInfo['pay_time']),
  1380. ],
  1381. 'remark' => [
  1382. 'value' => !empty($keywordsList['remark']) ? $keywordsList['remark']['example'] : '请您登录商城查看订单并发货!',
  1383. ],
  1384. ];
  1385. }
  1386. return [
  1387. 'data' => $data,
  1388. 'pagepath' => $pagepath,
  1389. 'gourl' => $gourl,
  1390. ];
  1391. }
  1392. }