【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.rhkb.cn/news/384995.html

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

相关文章

基于微信小程序+SpringBoot+Vue的刷题系统(带1w+文档)

基于微信小程序SpringBootVue的刷题系统(带1w文档) 基于微信小程序SpringBootVue的刷题系统(带1w文档) 本系统是将网络技术和现代的管理理念相结合&#xff0c;根据试题信息的特点进行重新分配、整合形成动态的、分类明确的信息资源&#xff0c;实现了刷题的自动化&#xff0c;…

Springboot 多数据源事务

起因 在一个service方法上使用的事务,其中有方法是调用的多数据源orderDB 但是多数据源没有生效,而是使用的primaryDB 原因 spring 事务实现的方式 以 Transactional 注解为例 (也可以看 TransactionTemplate&#xff0c; 这个流程更简单一点)。 入口&#xff1a;ProxyTransa…

电商项目之如何判断线程池是否执行完所有任务

文章目录 1 问题背景2 前言3 4种常用的方法4 代码4.1 isTerminated()4.2 线程池的任务总数是否等于已执行的任务数4.3 CountDownLatch计数器4.4 CyclicBarrier计数器 1 问题背景 真实生产环境的电商项目&#xff0c;常使用线程池应用于执行大批量操作达到高性能的效果。应用场景…

基于Python的房产数据分析系统的设计与实现(源码+lw+部署文档+讲解等)

文章目录&#xff1a; 目录 详细视频演示 设计文档详细参考 技术开发的参考技术栈&#xff01; 2.1 Python语言 2.2 Django框架 2.3 MySQL 2.4 Hadoop介绍 2.5 Scrapy介绍 4.2 系统结构设计 4.3 数据库设计 界面设计与功能实现 5.1系统登录注册实现 5.2管理员模块…

[Meachines] [Easy] Admirer Adminer远程Mysql反向+Python三方库函数劫持权限提升

信息收集 IP AddressOpening Ports10.10.10.187TCP:21,22,80 $ nmap -p- 10.10.10.187 --min-rate 1000 -sC -sV PORT STATE SERVICE VERSION 21/tcp open ftp vsftpd 3.0.3 22/tcp open ssh OpenSSH 7.4p1 Debian 10deb9u7 (protocol 2.0) | ssh-hostkey: | …

Golang零基础入门课_20240726 课程笔记

视频课程 最近发现越来越多的公司在用Golang了&#xff0c;所以精心整理了一套视频教程给大家&#xff0c;这个只是其中的第一部&#xff0c;后续还会有很多。 视频已经录制完成&#xff0c;完整目录截图如下&#xff1a; 课程目录 01 第一个Go程序.mp402 定义变量.mp403 …

微信公众号获取用户openid(PHP版,snsapi_base模式)

微信公众号获取用户openid的接口有2个&#xff1a;snsapi_base、snsapi_userinfo 详情见微信公众号开发文档&#xff1a;https://developers.weixin.qq.com/doc/offiaccount/OA_Web_Apps/Wechat_webpage_authorization.html 本文介绍用PHP方式调用snsapi_base接口获取微信用户…

机器学习课程学习周报五

机器学习课程学习周报五 文章目录 机器学习课程学习周报五摘要Abstract一、机器学习部分1.1 向量序列作为模型输入1.1.1 文字的向量表达1.1.2 语音的向量表达 1.2 自注意力机制原理1.2.1 自注意力机制理论1.2.2 矩阵运算自注意力机制 1.3 多头自注意力1.4 位置编码1.5 截断自注…

什么是Socket、Socket在Java中的应用、Socket和SocketChannel区别

目录 什么是Socket TCP\IP UDP体系结构 Socket和TCP\IP的关系 Socket在Java中的应用 Socket和SocketChannel的区别 SocketChannel和Selector的关系 服务器的设计演化历程---多线程版 服务器的设计演化历程---线程池版 服务器的设计演化历程---Selector版 参考链接 什么…

Docker 搭建Elasticsearch详细步骤

本章教程使用Docker搭建Elasticsearch环境。 一、拉取镜像 docker pull docker.elastic.co/elasticsearch/elasticsearch:8.8.2二、运行容器 docker run -d --name elasticsearch -p 9200:9200 -p 9300:9300 -e "discovery.type=single-n

【前端 14】Vue常见指令

Vue常见指令 Vue.js 是一个构建用户界面的渐进式框架&#xff0c;它通过一系列简洁的指令&#xff08;Directives&#xff09;来增强HTML的功能&#xff0c;使得开发者能够更加方便地构建出响应式的Web应用。本文将详细讲解Vue中的几个核心指令&#xff1a;v-bind、v-model、v…

软件缺陷(Bug)、禅道

目录 软件缺陷的判定标准 软件缺陷的核心内容 构成缺陷的基本要素 缺陷报告 缺陷管理 缺陷的跟踪流程 项目管理工具--禅道 软件在使用过程中存在的任何问题&#xff08;如&#xff1a;错误、异常等&#xff09;&#xff0c;都叫软件的缺陷&#xff0c;简称bug。 软件缺…

独立开发者系列(35)——python环境的理解

新手阶段&#xff0c;为了快速入门&#xff0c;基本都是直接开始写python代码实现自己想要的效果&#xff0c;类似搭建博客&#xff0c;写个web服务器&#xff0c;搭建简易聊天室&#xff0c;偶尔也写些爬虫&#xff0c;或者使用pygame写个简单小游戏&#xff0c;也有tk库做点简…

树和二叉树(不用看课程)

1. 树 1.1 树的概念与结构 树是⼀种非线性的数据结构&#xff0c;它是由 n&#xff08;n>0&#xff09; 个有限结点组成⼀个具有层次关系的集合。把它叫做树是因为它看起来像⼀棵倒挂的树&#xff0c;也就是说它是根朝上&#xff0c;而叶朝下的。 • 有⼀个特殊的结点&am…

[微信小程序] css 解决纯数字或字母不自动换行的问题、控制文字行数

效果 css 代码 word-break: break-all; overflow: hidden; text-overflow: ellipsis; display: -webkit-box; -webkit-line-clamp: 2; -webkit-box-orient: vertical;解释 word-break: break-all; 作用&#xff1a;这个属性允许在单词内部进行换行&#xff0c;即使单词很长也…

FastAPI(七十七)实战开发《在线课程学习系统》接口开发-- 课程编辑和查看评论

源码见&#xff1a;"fastapi_study_road-learning_system_online_courses: fastapi框架实战之--在线课程学习系统" 课程编辑 先来看下课程编辑 1.判断是否登录 2.判断课程是否存在 3.是否有权限&#xff08;只有自己可以修改自己的课程&#xff09; 4.名称是否重复…

7月23日JavaSE学习笔记

异常&#xff1a; 程序中一些程序处理不了的特殊情况 异常类 Exception 继承自 Throwable 类&#xff08;可抛出的&#xff09; Throwable继承树 Error&#xff1a;错误/事故&#xff0c;Java程序无法处理&#xff0c;如 OOM内存溢出错误、内存泄漏...会导出程序崩溃 常见的…

区块链赋能民生大数据,共筑可信共享新生态

一、背景 在信息化浪潮的推动下&#xff0c;政府服务模式正经历着前所未有的变革。民生卡&#xff0c;作为连接政府与民众的桥梁&#xff0c;承载着居民享受多元化公共服务的重任。然而&#xff0c;部门间信息孤岛现象严重制约了服务效率与居民体验的提升。为此&#xff0c;民…

大厂面试官问我:ConcurrentHashMap底层原理?【后端八股文十五:Java集合合集】

本文为【Java集合 合集】初版&#xff0c;后续还会进行优化更新&#xff0c;欢迎大家关注交流~ hello hello~ &#xff0c;这里是绝命Coding——老白~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#…

【Golang 面试基础题】每日 5 题(十)

✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/UWz06 &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 Golang 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏…