form.html 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. {extend name="../../admin/view/main"}
  2. {block name="content"}
  3. {include file='shop_goods/formstyle'}
  4. <form onsubmit="return false;" id="GoodsForm" data-auto="true" method="post" class='layui-form layui-card table-block' autocomplete="off">
  5. <div class="layui-card-body padding-40 padding-bottom-20">
  6. {notempty name='marks'}
  7. <div class="layui-form-item relative block">
  8. <span class="color-green font-w7 label-required-prev">商品标签</span>
  9. <span class="color-desc margin-left-5">Mark Name</span>
  10. <div class="tags-container layui-textarea">
  11. {foreach $marks as $mark}{if isset($vo.mark) && is_array($vo.mark) && in_array($mark, $vo.mark)}
  12. <label class="think-checkbox notselect"><input checked type="checkbox" name="mark[]" value="{$mark}" lay-ignore> {$mark}</label>
  13. {else}
  14. <label class="think-checkbox notselect"><input type="checkbox" name="mark[]" value="{$mark}" lay-ignore> {$mark}</label>
  15. {/if}{/foreach}
  16. </div>
  17. </div>
  18. {/notempty}
  19. {notempty name='cates'}
  20. <label class="layui-form-item block relative">
  21. <span class="color-green font-w7 label-required-prev">所属分类</span>
  22. <span class="color-desc margin-left-5">Category Name</span>
  23. <select class="layui-select" name="cate" lay-search>
  24. {foreach $cates as $cate}{if isset($vo.cate) and $vo.cate eq $cate.id}
  25. <option {$cate.sat} selected value="{$cate.id}">{$cate.spl}{$cate.name|default=''}</option>
  26. {else}
  27. <option {$cate.sat} value="{$cate.id}">{$cate.spl}{$cate.name|default=''}</option>
  28. {/if}{/foreach}
  29. </select>
  30. </label>
  31. {/notempty}
  32. <label class="layui-form-item block relative">
  33. <span class="color-green font-w7">商品名称</span>
  34. <span class="color-desc margin-left-5">Goods Name</span>
  35. <input name="name" required class="layui-input" placeholder="请输入商品名称" value="{$vo.name|default=''}">
  36. </label>
  37. <label class="layui-form-item block relative">
  38. <span class="color-green font-w7 label-required-prev">邮费模板</span>
  39. <span class="color-desc margin-left-5">Truck Template</span>
  40. <select class="layui-select" name="truck_tcode" lay-search>
  41. <option value="">--- 包 邮 ---</option>
  42. {foreach $vo.truck_items as $truck}{if isset($vo.truck_tcode) and $vo.truck_tcode eq $truck.code}
  43. <option selected value="{$truck.code}">{$truck.code} - {$truck.name|default=''}</option>
  44. {else}
  45. <option value="{$truck.code}">{$truck.code} - {$truck.name|default=''}</option>
  46. {/if}{/foreach}
  47. </select>
  48. </label>
  49. <div class="layui-form-item">
  50. <span class="color-green font-w7 label-required-prev">商品图片及轮播展示图片</span>
  51. <table class="layui-table">
  52. <thead>
  53. <tr>
  54. <th style="width:90px" class="text-center">LOGO</th>
  55. <th class="text-left">展示图片</th>
  56. </tr>
  57. <tr>
  58. <td class="text-center">
  59. <input name="cover" type="hidden" value="{$vo.cover|default=''}">
  60. <script>$('[name="cover"]').uploadOneImage();</script>
  61. </td>
  62. <td class="text-left">
  63. <input name="slider" type="hidden" value="{$vo.slider|default=''}">
  64. <script>$('[name="slider"]').uploadMultipleImage();</script>
  65. </td>
  66. </tr>
  67. </thead>
  68. </table>
  69. </div>
  70. <div class="layui-form-item">
  71. <span class="color-green font-w7 label-required-prev">商品规格及商品SKU绑定<span class="color-red font-s12">(规格填写后不允许再次修改)</span></span>
  72. <div ng-repeat="x in specs track by $index" style="display:none" class="margin-bottom-10" ng-class="{true:'layui-show'}[mode==='add'&&specs.length>0]">
  73. <div class="goods-spec-box padding-10 margin-0 relative" style="background:#ddd">
  74. <span class="text-center goods-spec-title">分组</span>
  75. <label class="label-required-null inline-block"><input ng-blur="x.name=trimSpace(x.name)" ng-model="x.name" required placeholder="请输入分组名称"></label>
  76. <div class="pull-right">
  77. <a class="layui-btn layui-btn-sm layui-btn-primary goods-spec-btn" ng-click="addSpecVal(x.list)">增加</a>
  78. <a class="layui-btn layui-btn-sm layui-btn-primary goods-spec-btn" ng-class="{false:'layui-bg-gray'}[$index>0]" ng-click="upSpecRow(specs,$index)">上移</a>
  79. <a class="layui-btn layui-btn-sm layui-btn-primary goods-spec-btn" ng-class="{false:'layui-bg-gray'}[$index<specs.length-1]" ng-click="dnSpecRow(specs,$index)">下移</a>
  80. <a class="layui-btn layui-btn-sm layui-btn-primary goods-spec-btn" ng-click="delSpecRow(specs,$index)" ng-if="specs.length>1">删除</a>
  81. </div>
  82. </div>
  83. <div class="goods-spec-box padding-10 margin-0 layui-bg-gray block relative" ng-if="x.list && x.list.length > 0">
  84. <label class="label-required-null inline-block margin-right-10 margin-bottom-5 relative nowrap" ng-repeat="xx in x.list">
  85. <input type="checkbox" lay-ignore ng-model="xx.check" ng-click="xx.check=checkListChecked(x.list,$event.target.checked)">
  86. <input type="text" ng-blur="xx.name=trimSpace(xx.name)" ng-model="xx.name" ng-keyup="xx.name=$event.target.value" required placeholder="请输入规格">
  87. <a ng-if="x.list.length>1" ng-click="x.list=delSpecVal(x.list,$index)" class="layui-icon layui-icon-close font-s12 goods-spec-close"></a>
  88. </label>
  89. </div>
  90. </div>
  91. <div ng-if="mode==='add'">
  92. <a ng-if="specs.length<3" class="layui-btn layui-btn-sm layui-btn-primary" ng-click="addSpecRow(specs)">增加分组</a>
  93. <p>请完成属性修改后再编辑下面的规格信息,否则规格数据会丢失!!!</p>
  94. </div>
  95. <table class="layui-table margin-top-10">
  96. <thead>
  97. <tr>
  98. <th ng-repeat="x in navas track by $index" class="nowrap" ng-bind="x"></th>
  99. <th width="12%" class="text-center nowrap">商品SKU <a ng-click="batchSet('sku',null)" data-tips-text="批量设置" class="layui-icon">&#xe63c;</a></th>
  100. <th width="10%" class="text-center nowrap">市场价格 <a ng-click="batchSet('market',2)" data-tips-text="批量设置" class="layui-icon">&#xe63c;</a></th>
  101. <th width="10%" class="text-center nowrap">销售价格 <a ng-click="batchSet('selling',2)" data-tips-text="批量设置" class="layui-icon">&#xe63c;</a></th>
  102. <th width="10%" class="text-center nowrap">虚拟销量 <a ng-click="batchSet('virtual',0)" data-tips-text="批量设置" class="layui-icon">&#xe63c;</a></th>
  103. <th width="10%" class="text-center nowrap">快递计件 <a ng-click="batchSet('express',0)" data-tips-text="批量设置" class="layui-icon">&#xe63c;</a></th>
  104. <th width="08%" class="text-center nowrap">销售状态</th>
  105. </tr>
  106. </thead>
  107. <tbody>
  108. <tr ng-repeat="rows in items track by $index">
  109. <td class="layui-bg-gray nowrap" ng-if="td.show" ng-repeat="td in rows" ng-bind="td.name"></td>
  110. <td class="padding-0">
  111. <label class="padding-0 margin-0">
  112. <input ng-blur="rows[0].sku=setValue(rows[0].key,'sku',$event.target.value)" class="layui-input border-0 padding-left-0 text-center" ng-model="rows[0].sku">
  113. </label>
  114. </td>
  115. <td class="padding-0">
  116. <label class="padding-0 margin-0">
  117. <input ng-blur="rows[0].market=setValue(rows[0].key,'market',$event.target.value,'(parseFloat(_)||0).toFixed(2)')" class="layui-input border-0 padding-left-0 text-center" ng-model="rows[0].market">
  118. </label>
  119. </td>
  120. <td class="padding-0">
  121. <label class="padding-0 margin-0">
  122. <input ng-blur="rows[0].selling=setValue(rows[0].key,'selling',$event.target.value,'(parseFloat(_)||0).toFixed(2)')" class="layui-input border-0 padding-left-0 text-center" ng-model="rows[0].selling">
  123. </label>
  124. </td>
  125. <td class="padding-0">
  126. <label class="padding-0 margin-0">
  127. <input ng-blur="rows[0].virtual=setValue(rows[0].key,'virtual',$event.target.value,'(parseInt(_)||0)')" class="layui-input border-0 padding-left-0 text-center" ng-model="rows[0].virtual">
  128. </label>
  129. </td>
  130. <td class="padding-0">
  131. <label class="padding-0 margin-0">
  132. <input ng-blur="rows[0].express=setValue(rows[0].key,'express',$event.target.value,'(parseInt(_)||0)')" class="layui-input border-0 padding-left-0 text-center" ng-model="rows[0].express">
  133. </label>
  134. </td>
  135. <td class="text-center layui-bg-gray">
  136. <label class="think-checkbox margin-0 full-width full-height block"><input lay-ignore type="checkbox" ng-model="rows[0].status"></label>
  137. </td>
  138. </tr>
  139. </tbody>
  140. </table>
  141. <p class="color-desc">请注意商品的SKU尽量不要重复,也不能产生订单后再修改,否则会造成订单数据无法关联!</p>
  142. <label class="layui-hide">
  143. <textarea class="layui-textarea" name="data_specs">{{specs}}</textarea>
  144. <textarea class="layui-textarea" name="data_items">{{items}}</textarea>
  145. </label>
  146. </div>
  147. <label class="layui-form-item block">
  148. <span class="font-w7 color-green">商品简介描述</span>
  149. <textarea class="layui-textarea" placeholder="请输入商品简介描述" name="remark">{$vo.remark|default=''|raw}</textarea>
  150. </label>
  151. <div class="layui-form-item block">
  152. <span class="label-required-prev font-w7 color-green">商品富文本详情</span>
  153. <textarea name="content">{$vo.content|default=''|raw}</textarea>
  154. </div>
  155. <div class="hr-line-dashed margin-top-40"></div>
  156. {notempty name='vo.code'}<input type="hidden" name="code" value="{$vo.code}">{/notempty}
  157. <div class="layui-form-item text-center">
  158. <button class="layui-btn layui-btn-danger" ng-click="pageBack()" type="button">取消编辑</button>
  159. <button class="layui-btn" type="submit">保存商品</button>
  160. </div>
  161. </div>
  162. </form>
  163. {/block}
  164. {block name='script'}
  165. <label class="layui-hide">
  166. <textarea id="GoodsSpecs">{$vo.data_specs|raw|default=''}</textarea>
  167. <textarea id="GoodsItems">{$vo.data_items|raw|default=''}</textarea>
  168. </label>
  169. <script>
  170. /*! 表单初始化 */
  171. window.form.render();
  172. /*! 加载扩展插件 */
  173. require(['ckeditor', 'angular'], function () {
  174. window.createEditor('[name="content"]', {height: 500});
  175. var app = angular.module("GoodsForm", []).run(callback);
  176. angular.bootstrap(document.getElementById(app.name), [app.name]);
  177. function getRand(length, prefix) {
  178. return (function (time, code) {
  179. code += parseInt(time.substr(0, 1)) + parseInt(time.substr(1, 1)) + time.substr(2, 8);
  180. while (code.length < length) code += Math.round(Math.random() * 10);
  181. return code;
  182. })(Date.now().toString(), prefix || '' + '')
  183. }
  184. function callback($rootScope) {
  185. $rootScope.mode = '{$mode|default="add"}', $rootScope.navas = [];
  186. $rootScope.items = angular.fromJson(angular.element('#GoodsItems').val() || '[]') || {};
  187. $rootScope.cache = angular.fromJson(angular.element('#GoodsItems').val() || '[]') || {};
  188. $rootScope.specs = angular.fromJson(angular.element('#GoodsSpecs').val() || '[{"name":"默认分组","list":[{"name":"默认规格","check":true}]}]');
  189. /*! 批量设置数值 */
  190. $rootScope.batchSet = function (name, fixed) {
  191. layer.prompt({title: '请输入数值', formType: 0}, function (value, index) {
  192. layer.close(index), $rootScope.$apply(function () {
  193. if (fixed !== null) value = (parseFloat(value) || 0).toFixed(fixed);
  194. $rootScope.items.forEach(function (rows) {
  195. rows.forEach(function (item) {
  196. item[name] = value;
  197. });
  198. });
  199. });
  200. });
  201. };
  202. $rootScope.pageBack = function () {
  203. $.msg.confirm('确定要取消编辑吗?', function (index) {
  204. history.back(), $.msg.close(index);
  205. });
  206. };
  207. $rootScope.setValue = function (key, name, value, callback) {
  208. $rootScope.items[key] = $rootScope.items[key] || {};
  209. $rootScope.cache[key] = $rootScope.cache[key] || {};
  210. if (typeof callback === 'string' && callback.indexOf('_') > -1) {
  211. value = eval(callback.replace('_', "'" + value + "'"));
  212. }
  213. return $rootScope.cache[key][name] = $rootScope.items[key][name] = value;
  214. };
  215. $rootScope.getValue = function (key, name, value) {
  216. var cache = $rootScope.cache[key] || {};
  217. if (typeof cache[name] === 'undefined') {
  218. $rootScope.setValue(key, name, value, '_')
  219. cache = $rootScope.cache[key] || {};
  220. }
  221. return cache[name];
  222. };
  223. /*! 去除空白字符 */
  224. $rootScope.trimSpace = function (value) {
  225. return (value + '').replace(/\s*/ig, '');
  226. };
  227. /*! 当前商品规格发生变化时重新计算规格列表 */
  228. $rootScope.$watch('specs', function () {
  229. var data = [], navs = [], table = [[]];
  230. $rootScope.specs.forEach(function (spec) {
  231. var temp = [];
  232. spec.list.forEach(function (item) {
  233. if (item.check && item.name.length > 0) {
  234. item.show = true, item.group = spec.name;
  235. temp.push(item);
  236. }
  237. });
  238. data.push(temp), navs.push(spec.name);
  239. });
  240. $rootScope.navas = navs;
  241. /*! 表格交叉 */
  242. data.forEach(function (rows) {
  243. var temp = [];
  244. table.forEach(function (line) {
  245. rows.forEach(function (item) {
  246. temp.push(line.concat(item));
  247. });
  248. });
  249. table = temp;
  250. });
  251. /*! 表格数据 */
  252. data = angular.fromJson(angular.toJson(table));
  253. data.forEach(function (rows) {
  254. var keys = [];
  255. rows.forEach(function (item) {
  256. keys.push(item.group + '::' + item.name);
  257. }), rows.every(function (item) {
  258. item.key = keys.join(';;');
  259. item.sku = $rootScope.getValue(item.key, 'sku', getRand(14, 'S'));
  260. item.status = !!$rootScope.getValue(item.key, 'status', 1);
  261. item.market = $rootScope.getValue(item.key, 'market', '0.00');
  262. item.selling = $rootScope.getValue(item.key, 'selling', '0.00');
  263. item.express = $rootScope.getValue(item.key, 'express', '1');
  264. item.virtual = $rootScope.getValue(item.key, 'virtual', '0');
  265. return false;
  266. });
  267. });
  268. $rootScope.items = data;
  269. }, true);
  270. /*! 判断规则是否能取消选择 */
  271. $rootScope.checkListChecked = function (data, check) {
  272. for (var i in data) if (data[i].check) return check;
  273. return true;
  274. };
  275. /*! 增加整行规格分组 */
  276. $rootScope.addSpecRow = function (data) {
  277. data.push({name: '规格分组' + data.length, list: [{name: '规格属性', check: true}]})
  278. };
  279. /*! 下移整行规格分组 */
  280. $rootScope.dnSpecRow = function (data, $index) {
  281. var temp = [], self = data[$index];
  282. if ($index > data.length - 2) return false;
  283. data.forEach(function (item, index) {
  284. if (parseInt(index) !== parseInt($index)) temp.push(item);
  285. if (parseInt(index) === parseInt($index) + 1) temp.push(self);
  286. });
  287. return $rootScope.specs = temp;
  288. };
  289. /*! 上移整行规格分组 */
  290. $rootScope.upSpecRow = function (data, $index) {
  291. var temp = [], self = data[$index];
  292. if ($index < 1) return false;
  293. data.forEach(function (item, index) {
  294. if (parseInt(index) === parseInt($index) - 1) temp.push(self);
  295. if (parseInt(index) !== parseInt($index)) temp.push(item);
  296. });
  297. return $rootScope.specs = temp;
  298. };
  299. /*! 移除整行规格分组 */
  300. $rootScope.delSpecRow = function (data, $index) {
  301. var temp = [];
  302. data.forEach(function (item, index) {
  303. if (parseInt(index) !== parseInt($index)) temp.push(item);
  304. });
  305. return $rootScope.specs = temp;
  306. };
  307. /*! 增加分组的属性 */
  308. $rootScope.addSpecVal = function (data) {
  309. data.push({name: '规格属性' + data.length, check: true});
  310. };
  311. /*! 移除分组的属性 */
  312. $rootScope.delSpecVal = function (data, $index) {
  313. var temp = [];
  314. data.forEach(function (item, index) {
  315. if (parseInt(index) !== parseInt($index)) temp.push(item);
  316. });
  317. return temp;
  318. };
  319. }
  320. });
  321. </script>
  322. {/block}