【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://www.mzph.cn/bicheng/47746.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Cloud Loadbalancer 的使用

一、默认负载均衡策略 Spring Cloud LoadBalancer 默认的负载均衡策略是轮询。 轮询效果示例 我们需要示例一个请求分发到不同的模块上&#xff0c;所以我们需要创建多模块项目。 新建 Spring Boot &#xff08;3.0.2&#xff09;的 Maven 项目&#xff08;JDK 17&#xff09…

ETL数据集成丨通过ETLCloud工具,将Oracle数据实时同步至Doris中

ETLCloud是一个全面的数据集成平台&#xff0c;专注于解决大数据量和高合规要求环境下的数据集成需求。采用先进的技术架构&#xff0c;如微服务和全Web可视化的集成设计&#xff0c;为用户提供了一站式的数据处理解决方案。 主要特点和功能包括&#xff1a; 实时数据处理&…

【已解决】Django连接MySQL启动报错Did you install mysqlclient?

在终端执行python manage.py makemigrations报错问题汇总 错误1&#xff1a;已安装mysqlclient&#xff0c;提示Did you install mysqlclient? 当你看到这样的错误信息&#xff0c;表明Django尝试加载MySQLdb模块但未找到&#xff0c;因为MySQLdb已被mysqlclient替代。 【解…

Linux——Shell脚本和Nginx反向代理服务器

1. Linux中的shell脚本【了解】 1.1 什么是shell Shell是一个用C语言编写的程序&#xff0c;它是用户使用Linux的桥梁 Shell 既是一种命令语言&#xff0c;有是一种程序设计语言 Shell是指一种应用程序&#xff0c;这个应用程序提供了一个界面&#xff0c;用户通过这个界面访问…

PyTorch 深度学习实践-处理多维特征的输入

视频指路 参考博客笔记 参考笔记二 通过多个线性模型来模拟非线性的空间变换&#xff0c;矩阵计算就是不同维度之间的空间转换 说明&#xff1a;1、乘的权重(w)都一样&#xff0c;加的偏置(b)也一样。b变成矩阵时使用广播机制。神经网络的参数w和b是网络需要学习的&#xff0c…

MybatisPlusException: Error: Method queryTotal execution error of sql 的报错解决

项目场景&#xff1a; 相关背景&#xff1a; 开发环境 开发系统时 系统页面加载正常 &#xff0c;发布运行环境后运行一段时间&#xff0c;前端页面 突然出现 报错信息&#xff0c; 报错信息如下&#xff1a; MybatisPlusException: Error: Method queryTotal execution erro…

如何在AWS上构建Apache DolphinScheduler

引言 随着云计算技术的发展&#xff0c;Amazon Web Services (AWS) 作为一个开放的平台&#xff0c;一直在帮助开发者更好的在云上构建和使用开源软件&#xff0c;同时也与开源社区紧密合作&#xff0c;推动开源项目的发展。 本文主要探讨2024年值得关注的一些开源软件及其在…

PDF转Word怎么快速转换?格式转换技巧分享

PDF文件和Word文档是我们日常工作中不可或缺的文件格式&#xff0c;同时文件之间的格式转换也十分常见。不同的格式有着不同的优点&#xff0c;将PDF文件改为Word文档后&#xff0c;在编辑或修改文件内容时更为方便。 下面小编就来给大家介绍几种常用的PDF转Word的转换方法&am…

Android 使用FFmpeg解析RTSP流,ANativeWindow渲染 使用SurfaceView播放流程详解

文章目录 ANativeWindow 介绍ANativeWindow 的主要功能和特点ANativeWindow 的常用函数工作流程原理图通过ANativeWindow渲染RGB纯色示例 播放RTSP流工作流程图关键步骤解析自定义SurfaceView组件native 层解码渲染 效果展示注意事项 这篇文章涉及到jni层&#xff0c;以及Ffmpe…

Linux-开机自动挂载(文件系统、交换空间)

准备磁盘 添加三块磁盘&#xff08;两块SATA&#xff0c;一块NVMe&#xff09; 查看设备&#xff1a; [rootlocalhost jian]# ll /dev/sd* [rootlocalhost jian]# ll /dev/nvme0n2 扩&#xff1a;查看当前主机上的所有块设备&#xff0c;通过如下指令实现&#xff1a; [root…

基于语音识别的会议记录系统

文章目录 核心功能页面展示使用技术方案功能结构设计数据库表展示 核心功能页面展示 视频展示功能 1.创建会议 在开始会议之前需要管理员先创建一个会议&#xff0c;为了能够快速开始会议&#xff0c;仅需填写会议的名称、会议举办小组、会议背景等简要会议信息即可成功创建。…

c# .net core中间件,生命周期

某些模块和处理程序具有存储在 Web.config 中的配置选项。但是在 ASP.NET Core 中&#xff0c;使用新配置模型取代了 Web.config。 HTTP 模块和处理程序如何工作 官网地址&#xff1a; 将 HTTP 处理程序和模块迁移到 ASP.NET Core 中间件 | Microsoft Learn 处理程序是&#xf…

【iOS】——内存分区

内存管理 程序运行的过程中往往会涉及到创建对象、定义变量、调用函数或方法&#xff0c;而这些行为都会增加程序的内存占用。为了防止内存占用过多影响到程序的正常运行就需要对内存进行管理。 移动端的内存管理机制&#xff0c;主要有三种&#xff1a; 自动垃圾收集(GC)手…

两台电脑之间如何进行数据传输?两台电脑数据传输攻略

在数字化时代&#xff0c;电脑之间的数据传输变得日益重要。无论是个人用户还是企业用户&#xff0c;经常需要在不同的电脑之间共享或迁移数据。那么&#xff0c;两台电脑之间如何进行数据传输呢&#xff1f;本文将详细介绍两台电脑之间进行数据传输的几种常见方法&#xff0c;…

CI/CD的node.js编译报错npm ERR! network request to https://registry.npmjs.org/

1、背景&#xff1a; 在维护paas云平台过程中&#xff0c;有研发反馈paas云平台上的CI/CD的前端流水线执行异常。 2、问题描述&#xff1a; 流水线执行的是前端编译&#xff0c;使用的是node.js环境。报错内容如下&#xff1a; 2024-07-18T01:23:04.203585287Z npm ERR! code E…

【BUG】已解决:note: This is an issue with the package mentioned above,not pip.

已解决&#xff1a;note: This is an issue with the package mentioned above&#xff0c;not pip. 欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 欢迎来到我的主页&#xff0c;我是博主英杰&#xff0c;211科班出身&#xff0c;就职于医疗科技公司&#xff0c;热衷…

Pytorch基础应用

1.数据加载 1.1 读取文本文件 方法一&#xff1a;使用 open() 函数和 read() 方法 # 打开文件并读取全部内容 file_path example.txt # 替换为你的文件路径 with open(file_path, r) as file:content file.read()print(content)方法二&#xff1a;逐行读取文件内容 # 逐…

【 FPGA 线下免费体验馆】高端 AMD- xilinx 16 nm UltraScale +系列

在复杂的FPGA 开发的过程中&#xff0c;需要能够满足高速、高精度、多通道等的复杂应用。而一个简单的 FPGA 开发板是不具备这些的&#xff0c;因此需要用更高端&#xff0c;大容量&#xff0c;高速IO的 FPGA 芯片与其他硬件组成一个完整的系统开发。这里就产生了高端 FPGA 开发…

redis服务器同 redis 集群

搭建redis服务器 修改服务运行参数 常用命令常用命令 创建redis集群 准备做集群的主机&#xff0c;不允许存储数据、不允许设置连接密码 配置服务器&#xff1a; 1、在任意一台redis服务器上都可以执行创建集群的命令。 2、--cluster-replicas 1 给每个master服务器分配1台…

【go】Excelize处理excel表 带合并单元格、自动换行与固定列宽的文件导出

文章目录 1 简介2 相关需求与实现2.1 导出带单元格合并的excel文件2.2 导出增加自动换行和固定列宽的excel文件 1 简介 之前整理过使用Excelize导出原始excel文件与增加数据校验的excel导出。【go】Excelize处理excel表 带数据校验的文件导出 本文整理使用Excelize导出带单元…