【Django】网上蛋糕商城后台-商品管理

1.商品管理功能

当管理员点击商品管理时,发送服务器请求

path('admin/goods_list/', viewsAdmin.goods_list),
# 处理商品列表请求
def goods_list(request):try:type = request.GET["type"]except:type = 0try:ym = request.GET["ym"]except:ym = 1if int(type) == 0:goodsList = Goods.objects.all().order_by("-id").values()for goods in goodsList:typeName = Type.objects.get(id=goods["type_id"]).nametypes = Recommend.objects.filter(goods_id=goods["id"]).values_list("type")if (1,) in types:goods["isScroll"] = Trueif (2,) in types:goods["isHot"] = Trueif (3,) in types:goods["isNew"] = Truegoods["typeName"] = typeNameelse:recommends = Recommend.objects.filter(type=type).order_by("-goods_id")goodsList = []for r in recommends:# 根据推荐栏类型查询商品信息,将查询的对象转换成字典goods = Goods.objects.get(id=r.goods_id).__dict__# 根据商品id查询该商品添加在哪些推荐栏中types = Recommend.objects.filter(goods_id=r.goods_id).values_list("type")if (1,) in types:goods["isScroll"] = Trueif (2,) in types:goods["isHot"] = Trueif (3,) in types:goods["isNew"] = True# 根据商品中的分类id查询分类名称typeName = Type.objects.get(id=goods["type_id"])goods["typeName"] = typeName.namegoodsList.append(goods)# 将该分类的商品信息进行分页处理,每页显示6条记录pag = paginator.Paginator(goodsList, 6)# 根据当前页码获取当前分页信息pageInfo = pag.get_page(ym)# 获取当前页的商品列表信息goodsList = pageInfo.object_list# 获取总页码数yms = pag.page_rangereturn render(request, "adminTemp/goods_list.html",{"goodsList": goodsList, "page": pageInfo, "yms": yms, "type": type})
<!DOCTYPE html>
<html>
<head><title>商品列表</title>{% load static %}<meta charset="utf-8"/><link rel="stylesheet" href="{% static 'css/bootstrap.css' %}"/><link rel="stylesheet" href="{% static 'css/page.css' %}"/>
</head>
<body>
<div class="container-fluid">{% include "adminTemp/header.html" %}<div class="text-right"><a class="btn btn-warning" href="/admin/goods_add/">添加商品</a></div><br><ul role="tablist" class="nav nav-tabs"><li {% if type == 0 %}class="active"{% endif %} role="presentation"><a href="/admin/goods_list/">全部商品</a></li><li {% if type == 1 %}class="active"{% endif %} role="presentation"><a href="/admin/goods_list/?type=1">条幅推荐</a></li><li {% if type == 2 %}class="active"{% endif %} role="presentation"><a href="/admin/goods_list/?type=2">热销推荐</a></li><li {% if type == 3 %}class="active"{% endif %} role="presentation"><a href="/admin/goods_list/?type=3">新品推荐</a></li></ul><br><table class="table table-bordered table-hover"><tr><th width="5%">ID</th><th width="10%">图片</th><th width="10%">名称</th><th width="20%">介绍</th><th width="10%">价格</th><th width="10%">类目</th><th width="25%">操作</th></tr>{% for g in goodsList %}<tr><td><p>{{ g.id }}</p></td><td><p><a href="/goods_detail/?id={{ g.id }}" target="_blank"><img src="{% static g.cover %}"width="100px"height="100px"></a></p></td><td><p><a href="/goods_detail/?id={{ g.id }}" target="_blank">{{ g.name }}</a></p></td><td><p>{{ g.intro }}</p></td><td><p>{{ g.price }}</p></td><td><p>{{ g.typeName }}</p></td><td><p>{% if g.isScroll %}<a class="btn btn-info"href="/admin/goods_recommend/?id={{ g.id }}&method=remove&typeTarget=1">移出条幅</a>{% endif %}{% if not g.isScroll %}<a class="btn btn-primary"href="/admin/goods_recommend/?id={{ g.id }}&method=add&typeTarget=1">加入条幅</a>{% endif %}{% if g.isHot %}<a class="btn btn-info"href="/admin/goods_recommend/?id={{ g.id }}&method=remove&typeTarget=2">移出热销</a>{% endif %}{% if not g.isHot %}<a class="btn btn-primary"href="/admin/goods_recommend/?id={{ g.id }}&method=add&typeTarget=2">加入热销</a>{% endif %}{% if g.isNew %}<a class="btn btn-info"href="/admin/goods_recommend/?id={{ g.id }}&method=remove&typeTarget=3">移出新品</a>{% endif %}{% if not g.isNew %}<a class="btn btn-primary"href="/admin/goods_recommend/?id={{ g.id }}&method=add&typeTarget=3">加入新品</a>{% endif %}</p><a class="btn btn-success"href="/admin/goods_editshow/?id={{ g.id  }}&ym={{ page.number  }}">修改</a><a class="btn btn-danger"href="/admin/goods_delete/?id={{ g.id  }}&ym={{ page.number  }}">删除</a></td></tr>{% endfor %}</table><br><!-- 显示页码导航栏 --><div id="nav" align="center"><!-- 上一页 --><!-- 判断当前页是否有上一页,如果有上一页则显示上一页的按钮,否则就不显示上一页 -->{% if page.has_previous %}<a href="/admin/goods_list/?ym={{ page.previous_page_number }}&type={{ type }}" class="up_page">上一页</a>{% endif %}<!-- 页码 -->{% for ym in yms %}{% if page.number == ym %}<a href="/admin/goods_list/?ym={{ ym }}&type={{ type }}" class="p_page c_page">{{ ym }}</a>{% else %}<a href="/admin/goods_list/?ym={{ ym }}&type={{ type }}" class="p_page">{{ ym }}</a>{% endif %}{% endfor %}<!-- 下一页 -->{% if page.has_next %}<a href="/admin/goods_list/?ym={{ page.next_page_number }}&type={{ type }}" class="do_page">下一页</a>{% endif %}</div><br>
</div>
</body>
</html>

2.加入或移除推荐栏功能

选择某个商品,点击加入条幅,移除条幅,加入热销,移除热销,加入新品,移除新品按钮时,触发请求事件,传递不同参数信息给服务器

path('admin/goods_recommend/',viewsAdmin.goods_recommend),
# 处理商品的推荐栏请求
def goods_recommend(request):id = request.GET["id"]method = request.GET["method"]typeTarget = request.GET["typeTarget"]if "add" == method:# 添加至推荐栏Recommend.objects.create(goods_id=id, type=typeTarget)elif "remove" == method:# 从推荐栏中移除r = Recommend.objects.get(goods_id=id, type=typeTarget)r.delete()# 刷新商品管理列表页面return redirect(goods_list)

3.添加商品功能

当管理员点击添加商品按钮,触发请求事件

path('admin/goods_add/', viewsAdmin.goods_add),
# 处理添加商品页面的跳转请求
def goods_add(request):types = Type.objects.all()return render(request, "adminTemp/goods_add.html", {"typeList": types})
<!DOCTYPE html>
<html>
<head><title>商品添加</title>{% load static %}<meta charset="utf-8"/><link rel="stylesheet" href="{% static 'css/bootstrap.css' %}"/>
</head>
<body>
<div class="container-fluid">{% include "adminTemp/header.html" %}<br><br><form class="form-horizontal" action="/admin/addGoods/" method="post" enctype="multipart/form-data">{% csrf_token %}<div class="form-group"><label for="input_name" class="col-sm-1 control-label">名称</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="name" required="required"></div></div><div class="form-group"><label for="input_name" class="col-sm-1 control-label">价格</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="price"></div></div><div class="form-group"><label for="input_name" class="col-sm-1 control-label">介绍</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="intro"></div></div><div class="form-group"><label for="input_name" class="col-sm-1 control-label">库存</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="stock"></div></div><div class="form-group"><label for="input_file" class="col-sm-1 control-label">封面图片</label><div class="col-sm-6"><input type="file" name="cover" id="input_file" required="required">推荐尺寸: 500 * 500</div></div><div class="form-group"><label for="input_file" class="col-sm-1 control-label">详情图片1</label><div class="col-sm-6"><input type="file" name="image1" id="input_file" required="required">推荐尺寸: 500 * 500</div></div><div class="form-group"><label for="input_file" class="col-sm-1 control-label">详情图片2</label><div class="col-sm-6"><input type="file" name="image2" id="input_file" required="required">推荐尺寸: 500 * 500</div></div><div class="form-group"><label for="select_topic" class="col-sm-1 control-label">类目</label><div class="col-sm-6"><select class="form-control" id="select_topic" name="typeid">{% for t in typeList %}<option value="{{ t.id }}">{{ t.name }}</option>{% endfor %}</select></div></div><div class="form-group"><div class="col-sm-offset-1 col-sm-10"><button type="submit" class="btn btn-success">提交保存</button></div></div></form>
</div>
</body>
</html>

当管理员填写完商品信息以及选择好上传的图片后,点击提交保存按钮,将表单提交给服务器

path('admin/addGoods/',viewsAdmin.addGoods),
# 获取商品添加请求
def addGoods(request):name = request.POST["name"]price = request.POST["price"]intro = request.POST["intro"]stock = request.POST["stock"]pic = request.FILES.getlist('cover')cover = upload(pic[0])pic = request.FILES.getlist('image1')image1 = upload(pic[0])pic = request.FILES.getlist('image2')image2 = upload(pic[0])typeid = request.POST["typeid"]Goods.objects.create(name=name, price=price, intro=intro, stock=stock, cover=cover, image1=image1, image2=image2,type_id=typeid)# 添加成功刷新商品管理列表页面return redirect(goods_list)

对于处理图片上传的函数如下

def upload(pic, image=None):# 指定文件上传路径path = "CookieShopClient/static/picture/"if image:file_path = path + str(image)[8:]# 检查文件是否存在if os.path.isfile(file_path):# 删除文件os.remove(file_path)imageName = ""# 检查文件夹是否存在if not os.path.exists(path):# 如果文件夹不存在,则创建它os.makedirs(path)imageName = pic.name# 获取当前时间的时间戳(秒)timestamp_seconds = time.time()# 转换为毫秒timestamp_milliseconds = int(timestamp_seconds * 1000)imageName = str(imageName).split(".")[0] + str(timestamp_milliseconds) + "." + str(imageName).split(".")[1]url = path + imageNamewith open(url, 'wb') as f:for data in pic.chunks():f.write(data)return "/picture/" + imageName

4.修改商品功能

当管理员选择某个商品进行修改时,将该商品的商品编号以及所在分页页码发送给修改页面

path('admin/goods_editshow/',viewsAdmin.goods_editshow),
# 处理跳转至修改页面的请求
def goods_editshow(request):id = request.GET["id"]ym = request.GET["ym"]goods = Goods.objects.get(id=id)# 查询该商品所属分类typeName = Type.objects.get(id=goods.type_id).nametypes = Type.objects.all()return render(request, "adminTemp/goods_edit.html", {"g": goods, "typeName": typeName, "ym": ym, "typeList": types})
<!DOCTYPE html>
<html>
<head><title>商品编辑</title>{% load static %}<meta charset="utf-8" /><link rel="stylesheet" href="{% static 'css/bootstrap.css' %}" />
</head>
<body>
<div class="container-fluid">{% include "adminTemp/header.html" %}<br><br><form class="form-horizontal" action="/admin/goods_edit/" method="post" enctype="multipart/form-data">{% csrf_token %}<input type="hidden" name="id" value="{{ g.id  }}"/><input type="hidden" name="cover" value="{{ g.cover  }}"/><input type="hidden" name="image1" value="{{ g.image1  }}"/><input type="hidden" name="image2" value="{{ g.image2  }}"/><input type="hidden" name="ym" value="{{ ym  }}"/><input type="hidden" name="type" value="{{ typeName }}"/><div class="form-group"><label for="input_name" class="col-sm-1 control-label">名称</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="name" value="{{ g.name  }}" required="required"></div></div><div class="form-group"><label for="input_name" class="col-sm-1 control-label">价格</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="price" value="{{ g.price  }}"></div></div><div class="form-group"><label for="input_name" class="col-sm-1 control-label">介绍</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="intro" value="{{ g.intro  }}"></div></div><div class="form-group"><label for="input_name" class="col-sm-1 control-label">库存</label><div class="col-sm-6"><input type="text" class="form-control" id="input_name" name="stock" value="{{ g.stock  }}"></div></div><div class="form-group"><label for="input_file" class="col-sm-1 control-label">封面图片</label><div class="col-sm-6"><img src="{% static g.cover %}" width="100" height="100"/><input type="file" name="cover"  id="input_file">推荐尺寸: 500 * 500</div></div><div class="form-group"><label for="input_file" class="col-sm-1 control-label">详情图片1</label><div class="col-sm-6"><img src="{% static g.image1 %}" width="100" height="100"/><input type="file" name="image1"  id="input_file">推荐尺寸: 500 * 500</div></div><div class="form-group"><label for="input_file" class="col-sm-1 control-label">详情图片2</label><div class="col-sm-6"><img src="{% static g.image2 %}" width="100" height="100"/><input type="file" name="image2"  id="input_file">推荐尺寸: 500 * 500</div></div><div class="form-group"><label for="select_topic" class="col-sm-1 control-label">类目</label><div class="col-sm-6"><select class="form-control" id="select_topic" name="typeid">{% for t in typeList %}<option{% if t.id == g.type_id %}selected="selected"{% endif %}value="{{ t.id }}">{{ t.name  }}</option>{% endfor %}</select></div></div><div class="form-group"><div class="col-sm-offset-1 col-sm-10"><button type="submit" class="btn btn-success">提交修改</button></div></div></form>
</div>
</body>
</html>

当管理员修改信息以及重新选择新图片后,将表单提交给服务器,服务器通过比较原图片以及新图片地址判断当前图片是否需要重新上传

path('admin/goods_edit/',viewsAdmin.goods_edit),
# 处理修改商品信息的请求
def goods_edit(request):id = request.POST["id"]# 根据id查询出原商品信息goods = Goods.objects.filter(id=id)name = request.POST["name"]price = request.POST["price"]intro = request.POST["intro"]stock = request.POST["stock"]try:# 判断修改页面传递的图片名称是否和数据库中原本图片名称一致,如果一致表示该图片没有被修改# 如果图片没有修改,则返回值为空pic = request.FILES.getlist('cover')[0]cover = upload(pic, goods[0].cover)except:cover = goods[0].covertry:# 判断修改页面传递的图片名称是否和数据库中原本图片名称一致,如果一致表示该图片没有被修改# 如果图片没有修改,则返回值为空pic = request.FILES.getlist('image1')[0]image1 = upload(pic, goods[0].image1)except:image1 = goods[0].image1try:# 判断修改页面传递的图片名称是否和数据库中原本图片名称一致,如果一致表示该图片没有被修改# 如果图片没有修改,则返回值为空pic = request.FILES.getlist('image2')[0]image2 = upload(pic, goods[0].image2)except:image2 = goods[0].image2typeid = request.POST["typeid"]ym = request.POST["ym"]# 修改商品信息goods.update(name=name, price=price, intro=intro, stock=stock, cover=cover, image1=image1, image2=image2,type_id=typeid)return HttpResponseRedirect("/admin/goods_list/?ym=" + ym)

5.删除商品功能

当管理员选择某个商品删除的时候,触发请求

path('admin/goods_delete/',viewsAdmin.goods_delete),
# 处理删除商品的请求
def goods_delete(request):id=request.GET["id"]ym=request.GET["ym"]# 先查看当前商品是否有被添加为推荐栏商品,如果有,需要先从推荐栏中删除rs=Recommend.objects.filter(goods_id=id)if rs:rs.delete()# 根据商品编号将商品信息查询出来goods=Goods.objects.get(id=id)remove_image(goods.cover)remove_image(goods.image1)remove_image(goods.image2)goods.delete()return HttpResponseRedirect("/admin/goods_list/?ym=" + ym)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/3248759.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

5大常用的回归测试工具介绍

回归测试工具介绍 以下是一些可用于创建和执行回归测试的工具。但是&#xff0c;在决定使用哪些产品之前&#xff0c;应彻底研究每种产品的要求。 Selenium Selenium 是一个开源 Web 自动化测试工具&#xff0c;用于测试网站和 Web 应用程序。它被认为是用于Web 应用程序测试的…

路径规划 | 基于DQN深度强化学习算法的路径规划(Matlab)

目录 效果一览基本介绍程序设计参考文献 效果一览 基本介绍 DQN路径规划算法 基于深度强化学习算法的路径规划 matlab2023b 栅格环境&#xff0c;走迷宫&#xff0c;可以通过窗口界面方便观察交互过程&#xff0c;代码注释详尽。 程序设计 完整源码和数据私信博主回复基于DQN深…

【Linux信号】信号的保存、信号在内核中的表示、信号集操作函数、sigprocmask、sigpending

目录 信号在内核中的表示信号阻塞的理解 sigset_t 信号集操作函数 sigprocmask sigpending sigprocmask和sigpending都是系统调用 我们先来了解一下关于信号的一些常见概念&#xff1a; 实际执行 信号的处理动作 称为信号递达。 信号从产生到递达的之间的状态称为信号未决…

场外个股期权交割日是每个月几号?怎么参与场外个股期权?

今天带你了解场外个股期权交割日是每个月几号&#xff1f;怎么参与场外个股期权&#xff1f;在进行期权交易之前&#xff0c;投资者需要选择一个可靠的期权交易平台。 个股场外期权交易是指在股票交易所以外的场所进行的期权交易。期权是一种约定&#xff0c;根据该约定&#…

Docker网络模式和Cgroup资源限制

目录 1、Docker网络 &#xff08;1&#xff09;Docker网络实现原理 查看容器的输出和日志信息 2、Docker 的网络模式 查看docker列表 &#xff08;1&#xff09;网络模式详解 1&#xff09;host模式 2&#xff09;container模式 3&#xff09;none模式 4&#xff09;br…

小程序-3(页面导航+页面事件+生命周期+WXS)

目录 1.页面导航 声明式导航 导航到tabBar页面 导航到非tabBar页面 后退导航 编程式导航 后退导航 导航传参 声明式导航传参 编程式导航传参 在onload中接收导航参数 2.页面事件 下拉刷新 停止下拉刷新的效果 ​编辑 上拉触底 配置上拉触底距离 上拉触底的节…

spring security源码追踪理解(一)

一、前言 近期看了spring security相关的介绍&#xff0c;再加上项目所用若依框架的底层安全模块也是spring security&#xff0c;所以想从源码的角度加深下对该安全模块的理解&#xff08;看源码之前&#xff0c;我们要先有个意识&#xff0c;那就是spring security安全模块主…

一文-深入了解Ansible常见模块、安装和部署

1 Ansible 介绍 Ansible是一个配置管理系统configuration management system, python 语言是运维人员必须会的语言, ansible 是一个基于python 开发的&#xff08;集合了众多运维工具 puppet、cfengine、chef、func、fabric的优点&#xff09;自动化运维工具, 其功能实现基于ss…

Linux中nohup(no hang up)不挂起,用于在系统后台不挂断地运行命令,即使退出终端也不会影响程序的运行。

nohup的英文全称是 no hang up&#xff0c;即“不挂起”。这个命令在Linux或Unix系统中非常有用&#xff0c;主要用于在系统后台不挂断地运行命令&#xff0c;即使退出终端也不会影响程序的运行。默认情况下&#xff08;非重定向时&#xff09;&#xff0c;nohup会将输出写入一…

美国银行:高息下的稳健

“四大银行”财报悉数登场&#xff0c;折射美国经济忧虑。 今天我们来聊——美国银行 上周五摩根大通、富国银行和花旗银行发布了二季度财报&#xff0c;结果不及预期&#xff0c;股价都是一个走法&#xff0c;跌。反倒是姗姗来迟的美国银行Q2业绩超预期&#xff0c;大涨超5%。…

统计学13——时间序列分析

目录 知识结构 ​编辑内容精读 1.时间序列及其分开类 2.描述性分析 3.时间序列预测 3.1预测过程 3.2平稳时间序列 3.3趋势型序列 3.4复合型序列 名词解释 知识结构 内容精读 1.时间序列及其分开类 时间序列顾名思义就是按时间顺序观察排列而成的序列&#xff0c;根…

苍穹外卖图片不显示

上传图片到自己的阿里云oss中&#xff0c;然后对应链接放到数据库中 显示成功

Linux——开机重启、用户登录注销、用户管理、运行级别、帮助指令

目录 关机&重启命令 用户登录&注销 用户管理 用户的添加和删除 查询用户信息指令 切换用户 查看当前登录用户 用户组 用户和组相关的文件 运行级别 基本介绍 设置默认运行级别 ​编辑​编辑 找回root密码 帮助指令 关机&重启命令 用户登录&注销…

Linux工具篇:gdb(调试工具)+ makefile(自动化构建工具)

目录 前言&#xff1a; Linux调试器-gdb使用&#xff1a; Linux项目自动化构建工具-make/Makefile&#xff1a; 问题&#xff1a; 为什么makefile对最新的可执行程序&#xff0c;默认不想不想重新形成呢&#xff1f; make是如何知道到我的程序需要被编译的呢&#xff1f; …

监控系统怎样做?

监控类型自底向上分为资源监控、服务监控和业务监控。希望打造公司级的监控系统最好的时机是系统规划时&#xff0c;如果把监控设计往后放&#xff0c;将会面临一个巨大的难题&#xff1a;推行和现有不兼容的规范。 三种监控类型 资源监控 这个相对简单&#xff0c;随着k8s的兴…

【个人笔记】685. 冗余连接 II 的解释(并查集)

一棵树有n个点和n条边&#xff0c;返回一条能删除的边&#xff0c;使得剩下的图是有 n 个节点的有根树。 解释&#xff1a; 注意不冗余的有根树的特性&#xff01;**根节点入度为0&#xff0c;其余结点只有一个入度&#xff01;**所以冗余的两种情况如下&#xff1a; &#xff…

芯片基础 | Verilog仿真平台及数字逻辑仿真(上)

被测试器件DUT是一个二选一多路器,测试程序(testbench)提供测试激励及验证机制 Testbench使用行为级描述,DUT采用门级描述 下面将给出Testbench的描述、DUT的描述及如何进行混合仿真(行为级+门级) DUT (Device Under Test) module mux2_1(//Port declarations 端口声明outp…

Linux常见配置

linux 常见配置 一、配置固定IP, 主机名映射二、配置环境变量三、vim配置四、ssh配置 一、配置固定IP, 主机名映射 1、修改主机名 hostnamectl set-hostname xxx2、Centos配置固定IP 使用vim编辑/etc/sysconfig/network-scripts/ifcfg-ens33文件&#xff0c;填入下图信息 …

AI伦理之舟:航行于隐私、公正与真实的海域

&#x1f308;所属专栏&#xff1a;【其它】✨作者主页&#xff1a; Mr.Zwq✔️个人简介&#xff1a;一个正在努力学技术的Python领域创作者&#xff0c;擅长爬虫&#xff0c;逆向&#xff0c;全栈方向&#xff0c;专注基础和实战分享&#xff0c;欢迎咨询&#xff01; 您的点…

数据编织 VS 数据仓库 VS 数据湖

目录 1. 什么是数据编织?2. 数据编织的工作原理3. 代码示例4. 数据编织的优势5. 应用场景6. 数据编织 vs 数据仓库6.1 数据存储方式6.2 数据更新和实时性6.3 灵活性和可扩展性6.4 查询性能6.5 数据治理和一致性6.6 适用场景6.7 代码示例比较 7. 数据编织 vs 数据湖7.1 数据存储…