公共用例库计划--个人版(六)典型Bug页面设计与开发

1、任务概述

  本次计划的核心任务是开发一个,个人版的公共用例库,旨在将各系统和各类测试场景下的通用、基础以及关键功能的测试用例进行系统性地归纳整理,并以提高用例的复用率为目标,力求最大限度地减少重复劳动,提升测试效率。
  计划内容:完成公共用例库的开发实施工作,包括需求分析、系统设计、开发、测试、打包、运行维护等工作。

1.1、 已完成:

  需求分析、数据库表的设计:公共用例库计划–个人版(一)
  主体界面与逻辑设计:公共用例库计划–个人版(二)
  导出Excel功能:公共用例库计划–个人版(三)
  模块选择功能改造与性能优化公共用例库计划–个人版(四)
  QtCharts制作首页饼图与柱状图公共用例库计划–个人版(五)

1.2、 本次待完成:

典型Bug:新增遇见的较典型的bug,进行复盘与经验总结。
  1. bug、图片的库表设计
  2. 典型Bug页面,界面设计
  3. 逻辑代码开发

2、bug、图片的库表设计

  计划创建两张表,一张表存放bug信息,一张表保存bug的图片。
在这里插入图片描述
建表语句:

			# 创建bug表self.cursor.execute("""  CREATE TABLE IF NOT EXISTS bug (  bugid INTEGER PRIMARY KEY AUTOINCREMENT,  --编号title TEXT NOT NULL,  --标题types INTEGER ,  --类型--对应码值表code40-49difficulty INTEGER ,  --难度--对应码值表code30-36moduleid INTEGER ,  --所属模块describe TEXT , --描述think TEXT , --思考severity INTEGER  --严重程度--对应码值表code50-53)""")# 创建图片表self.cursor.execute("""  CREATE TABLE IF NOT EXISTS image (  imageid INTEGER PRIMARY KEY AUTOINCREMENT,  --编号name TEXT NOT NULL,  --名称image_bytes BLOB ,  --图片bugid INTEGER  --所属bug)""")

3、典型Bug页面

主要功能设计介绍:
  左侧使用listWidget显示bug列表,有翻页功能。
在这里插入图片描述
  主体上面,把用例页面的查询条件copy过来改改,选择条件对左边列表的bug进行查询。
在这里插入图片描述
  主体中部,点击左侧列表中bug名称,显示对应bug的详细信息。有图片上传按钮,显示图片、翻页等功能按钮。

  最后建了一个窗口,放一个graphicsView控件,显示大图。(上面页面图片是label显示的,担心显示太小了)
在这里插入图片描述

4、bug新增代码开发

4.1 保存bug

  在设计页面时,对保存按钮增加信号槽,点击保存触发save_bug。
对标题、描述进行 必填验证。如果有获取到bugid,就走编辑流程,没有就新增bug。

    def save_bug(self):"""bug页面,保存"""if not self.lineEdit_4.text() or self.textEdit.toPlainText() == '[步骤]\n\n[结果]\n\n[期望]\n':self.ts.xinxi("标题或描述不能为空!")returnself.save_bug_b.setEnabled(False)  # 保存按钮置灰parts=self.mk_2.currentText().split("—ID:")  # 模块编号if len(parts) == 2:self.mk_2.mkid=parts[1]else:self.mk_2.mkid=''if self.bugid != 0:  # 编辑vlue=(f'{self.lineEdit_4.text()}',tc_sql.codes_dict['mk_types'][f'{self.lileixing_bug_2.currentText()}'],tc_sql.codes_dict['difficulty'][f'{self.nandu_2.currentText()}'],f'{self.mk_2.mkid}', f'{self.textEdit.toPlainText()}',f'{self.textEdit_2.toPlainText()}',tc_sql.codes_dict['severity'][f'{self.nandu_3.currentText()}'], f'{self.bugid}')self.savesql_bug("update bug set title=? ,types=? ,difficulty=?, moduleid=? ,describe = ?,think = ?,severity=? where bugid=?",vlue)else:  # 新增vlue=(f'{self.lineEdit_4.text()}',tc_sql.codes_dict['mk_types'][f'{self.lileixing_bug_2.currentText()}'],tc_sql.codes_dict['difficulty'][f'{self.nandu_2.currentText()}'],f'{self.mk_2.mkid}', f'{self.textEdit.toPlainText()}',f'{self.textEdit_2.toPlainText()}',tc_sql.codes_dict['severity'][f'{self.nandu_3.currentText()}'])self.savesql_bug("INSERT INTO bug VALUES (NULL,?,?,?,?,?,?,?)", vlue)def savesql_bug(self, sql, vlue):"""bug保存sql"""self.case_db.connect()if self.case_db.operate_one(sql, vlue):self.case_db.over()self.chaxun_bug()  # 查询列表self.ts.xinxi("保存成功")logging.info('bug页面,保存')else:self.ts.xinxi("保存失败,请检查")self.save_bug_b.setEnabled(True)  # 保存按钮恢复

4.2 bug查询

  在设计页面时,对每一个查询条件设计信号槽,触发查询按钮,查询按钮触发chaxun_bug。
先获取查询条件的信息,然后通过拼接条件sql,查询出对应的bug,写入bug列表显示。

    def chaxun_bug(self, ye=1):"""bug页面,查询"""parts=self.mk_1.currentText().split("—ID:")if len(parts) == 2:self.mk_1.mkid=parts[1]else:self.mk_1.mkid=''mk_types=tc_sql.codes_dict['mk_types'][f'{self.lileixing_bug.currentText()}'] if self.lileixing_bug.currentText() else ''difficulty=tc_sql.codes_dict['difficulty'][self.nandu.currentText()] if self.nandu.currentText() else ''dic={"bug.types": mk_types,"bug.difficulty": difficulty,"bug.moduleid": self.mk_1.mkid}dic_like={"bug.title": self.biaoti_bug.text()}sql1="select bugid || ':' || title from bug where"  # 拼接sql查询条件sql2="select count(*) from bug where"for k, v in dic.items():if v:sql1+=' ' + k + "= " + str(v) + " and"sql2+=' ' + k + "= " + str(v) + " and"for k, v in dic_like.items():if v:sql1+=' ' + k + " like '%" + v + "%' and"sql2+=' ' + k + " like '%" + v + "%' and"offset=20 * (ye - 1)  # 分页查询sql1=sql1.rstrip('where').rstrip('and') + " order by bug.bugid DESC " + f"LIMIT 20 OFFSET {offset}"sql2=sql2.rstrip('where').rstrip('and')self.case_db.connect()items=self.case_db.query_many(sql1)  # bug查询count=self.case_db.query_one(sql2)self.bugye_count=math.ceil(count[0] / 20)  # 页数计算self.case_db.over()self.listWidget.clear()  # 清空列表self.clear_bug()  # 清空页面if items:for i in range(len(items)):  # 数据写入列表item=QListWidgetItem(items[i][0])item.setToolTip(items[i][0])  # 写入提示self.listWidget.addItem(item)self.label_24.setText(f"总数: {count[0]}条 共: {self.bugye_count}页    ")self.lineEdit_3.setText(f"{ye}")else:self.listWidget.addItem("未查询到数据")  # 查询无数据,加空行self.label_24.setText(f"总数: 0条 共: 0页    ")self.lineEdit_3.setText("0")logging.info('bug页面,查询')

4.3 bug详情

  查询出bug列表后,点击其中的bug,页面显示详情。对列表增加槽函数,点击触发show_bug。
获取点击的bugID,根据bugid查询出信息,回显到页面。

    def show_bug(self):"""点击bug,显示bug详情"""a=self.listWidget.selectedItems()if a:parts=a[0].text().split(":")if len(parts) == 2:self.bugid=parts[0]else:self.bugid=0returnsql26='''select bug.title,codes1.value as types,codes2.value as difficulty,module.modulename || '—ID:' || module.moduleid,bug.describe,bug.think,codes3.value as severityfrom bug INNER JOINcodes AS codes1 ON bug.types = codes1.id AND codes1.id BETWEEN 40 AND 49 -- 关联类型码值表INNER JOINcodes AS codes2 ON bug.difficulty = codes2.id AND codes2.id BETWEEN 30 AND 36 -- 关联难度码值表INNER JOINcodes AS codes3 ON bug.severity = codes3.id AND codes3.id BETWEEN 50 AND 53LEFT JOINmodule  ON bug.moduleid = module.moduleid where bug.bugid = ? '''self.case_db.connect()items=self.case_db.query_one(sql26, (self.bugid,))  # bug查询self.case_db.over()# bug详情回显self.lineEdit_4.setText(items[0])index=self.lileixing_bug_2.findText(items[1], Qt.MatchFlag.MatchFixedString)self.lileixing_bug_2.setCurrentIndex(index)  # 设置下拉框内容index2=self.nandu_2.findText(items[2], Qt.MatchFlag.MatchFixedString)self.nandu_2.setCurrentIndex(index2)self.mk_2.clear()  # 清空模块下拉self.mk_2.addItem(items[3])self.mk_2.setCurrentIndex(0)  # 设置模块下拉显示self.textEdit.setText(items[4])self.textEdit_2.setText(items[5])index3=self.nandu_3.findText(items[6], Qt.MatchFlag.MatchFixedString)self.nandu_3.setCurrentIndex(index3)logging.info('显示bug详情')

4.4 实现情况

  现在bug新增,查询修改没问题。
在这里插入图片描述

5、图片管理

bug保存后,上传图片。有删除、图片翻页、查看大图功能

5.1 图片上传

  上传图片按钮,触发函数up_image。
对图片大小、格式限制,然后将图片转成Base64格式存入数据库。

    def up_image(self):"""上传bug图片"""if self.bugid == 0:self.ts.xinxi("未获取到bugID,请先选择或者保存bug")returnMAX_IMAGE_SIZE=1.5 * 1024 * 1024  # 图片大小限制(例如:5MB)try:filename, _=QFileDialog.getOpenFileName(self, "选择图片", "","Image Files (*.jpg *.jpeg *.png *.bmp *.gif)")if not filename:        # 未选择图片,退出returnname=filename.split('/')[-1]file_size=os.path.getsize(filename)  # 检查文件大小if file_size > MAX_IMAGE_SIZE:self.ts.xinxi(f'图片大小超过限制({MAX_IMAGE_SIZE / (1024 * 1024)}MB),请上传更小的图片!')else:img=Image.open(filename)img.save(f"_internal/{name}", quality=90)  # 压缩图片with open(f"_internal/{name}", 'rb') as image_file:content=base64.b64encode(image_file.read())  # 转换成Base64格式os.remove(f"_internal/{name}")self.case_db.connect()sql=f"INSERT INTO main.image (name, image_bytes,bugid) VALUES (?, ?, ?);"if self.case_db.operate_one(sql, (name, content, self.bugid)):self.case_db.over()self.show_image(self.bugid)logging.info('上传bug图片')else:self.ts.xinxi("图片上传出错")except Exception as e:self.ts.xinxi(f"上传出错:{e}")logging.error(e)

  数据库中,存入的图片。
数据库

5.2 图片显示

  图片上传后,就触发显示函数。将查询bug的图片,显示到label
首先对图像显示区域进行清理,如果查询到图片,将显示图片、名称、页数。

    def show_image(self, bugid, number=0):"""根据bugID,查询bug图片"""self.label_image.setText('暂无图片')  # 清空bug图片区域数据self.pushButton_10.setVisible(False)  # 删除图片按钮,隐藏self.label_15.clear()self.label_34.clear()self.imageid=Noneself.case_db.connect()sql=f"SELECT image_bytes,imageid,name FROM image WHERE bugid=? LIMIT 1 OFFSET {number} "value=self.case_db.query_one(sql, (bugid,))count=self.case_db.query_one(f"SELECT count(*) FROM image WHERE bugid=? ", (bugid,))self.case_db.over()if value:                       # 显示图片、数量、名称self.imageid=value[1]str_encode=base64.b64decode(value[0])  # base64编码对应的解码(解码完字符串)pixmap=QPixmap()pixmap.loadFromData(str_encode)self.label_image.clear()scaled_pixmap=pixmap.scaled(self.label_image.size(), Qt.AspectRatioMode.KeepAspectRatio)self.label_image.setPixmap(scaled_pixmap)self.label_15.setText(f"{number + 1}/{count[0]}")  # 数量self.label_34.setText(f"{value[2]}")self.pushButton_10.setVisible(True)  # 删除图片按钮,显示logging.info('bug图片显示')

5.3 图片上传实现

在这里插入图片描述

5.4 大图窗口

  点击页面图片,触发大图窗口。

    def image_clicked(self, event):"""点击图片,打开图片窗口"""if self.imageid:self.case_db.connect()sql=f"SELECT image_bytes FROM image WHERE imageid=?"value=self.case_db.query_one(sql, (self.imageid,))self.case_db.over()str_encode=base64.b64decode(value[0])if not self.child_windows:image_max=Image_bug(str_encode)self.child_windows.append(image_max)  # 将新打开的窗口添加到子窗口列表else:self.child_windows[0].update_image(str_encode)  # 已有窗口,更新图片logging.info('bug图片,大图窗口')

  大图窗口,继承界面设计好的窗口。
使用单例模式,只允许打开一个窗口,再次点击图片就更新窗口图片显示。
对鼠标滚轮进行重写,实现放大缩小图片。

class Image_bug(QWidget, Ui_imagemax):"""图片页面"""_instance=Nonedef __new__(cls, str_encode, parent=None):if cls._instance is None:instance=super().__new__(cls)instance.__init__(str_encode, parent)cls._instance=instance  # 保存到类变量以便后续使用return instanceelse:cls._instance.update_image(str_encode)  # 更新现有实例的图像数据return cls._instancedef __init__(self, str_encode, parent=None):if self._instance:  # 避免多次初始化同一个实例returnsuper().__init__(parent)self.str_encode=str_encodeself.setupUi(self)self.init_image()def init_image(self):"""窗口加载图片显示"""pixmap=QPixmap()pixmap.loadFromData(self.str_encode)scene=QGraphicsScene()# 创建图形视图、场景及图像项目item=QGraphicsPixmapItem(pixmap)scene.addItem(item)self.graphicsView.setScene(scene)# 设置视图与场景self.graphicsView.wheelEvent=self.wheelEvent_handler    # 重写滚轮事件处理函数self.show()def update_image(self, new_str_encode):"""更新窗口图片"""self.str_encode=new_str_encodeself.init_image()   # 重新加载图片数据self.activateWindow()  # 激活窗口self.raise_()       # 提升窗口至最前面def wheelEvent_handler(self, event):"""滚轮放大缩小图片"""zoom_in_factor=1.25zoom_out_factor=1 / zoom_in_factordelta=event.angleDelta().y() / 120if delta > 0:factor=zoom_in_factorelif delta < 0:factor=zoom_out_factorelse:return# 获取当前鼠标位置在视图坐标系中的位置mouse_pos_scene = self.graphicsView.mapToScene(QPoint(int(event.position().x()), int(event.position().y())))# 记录原始视图中心点old_center=self.graphicsView.viewport().rect().center()# 缩放视图self.graphicsView.scale(factor, factor)# 计算新的视图中心点,并确保鼠标下的内容不变new_center = self.graphicsView.mapFromScene(mouse_pos_scene).toPointF()self.graphicsView.centerOn(new_center)

5.5 大图窗口,放大缩小图片实现

在这里插入图片描述

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

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

相关文章

接口安全测试

一、后台接口分类 1、接口类别&#xff1a;restful(json) soap(xml) 2、协议 &#xff1a;http https(ssl) 3、restful接口请求类型 get操作是安全的 post的操作是不安全的 同put delete也是不安全的 4、现状和问题 大部分APP的接口都采用restful架构&#xff0c;restful…

G7-Semi-Supervised GAN解读

本文为&#x1f517;365天深度学习训练营 中的学习记录博客 原作者&#xff1a;K同学啊|接辅导、项目定制 我的环境&#xff1a; 1.语言&#xff1a;python3.7 2.编译器&#xff1a;pycharm 3.深度学习框架Pytorch 1.8.0cu111 论文地址 一、理论理解 半监督生成对抗网络&…

C语言系列-联合

&#x1f308;个人主页: 会编程的果子君 ​&#x1f4ab;个人格言:“成为自己未来的主人~” 目录 联合体 联合体类型的声明 联合体的特点 相同成员的结构体和联合体对比 联合体大小的计算 联合的一个练习 联合体 联合体类型的声明 像结构体一样&#xff0c;联合体也是由…

闲聊电脑(4)硬盘分区

夜深人静&#xff0c;万籁俱寂&#xff0c;老郭趴在电脑桌上打盹&#xff0c;桌子上的小黄鸭和桌子旁的冰箱又开始窃窃私语…… 小黄鸭&#xff1a;冰箱大哥&#xff0c;上次你说的那个“分区”和“格式化”是什么意思&#xff1f; 冰箱&#xff1a;分区么&#xff0c;就是分…

Jasperreport 生成 PDF之省纸模式

省纸模式顾名思义就是节省纸张&#xff0c;使用 Jasper 去生成 PDF 的时候如果进行分组打印的时候&#xff0c;一页 A4 纸只会打印一组数据。这种情况下&#xff0c;如果每组数据特别少&#xff0c;只有几行&#xff0c;一页 A4 纸张根本用不了&#xff0c;就会另起一页继续打印…

elementui中的tree自定义图标

需求&#xff1a;实现如下样式的树形列表 自定义树的图标以及点击时&#xff0c;可以根据子级的关闭&#xff0c;切换图标 <el-tree :data"treeList" :props"defaultProps"><template #default"{ node, data }"><span class&quo…

linux中vim的操作

(码字不易&#xff0c;关注一下吧w~~w) 命令模式&#xff1a; 当我们按下esc键时&#xff0c;我们会进入命令模式&#xff1b;当使用vi打开一个文件时也是进入命令模式。 光标移动&#xff1a; 1 保存退出&#xff1a;ZZ 2 代码格式化&#xff1a;ggG 3 光标移动&#xff…

谁懂啊!性能测试还能这么快学好。。。。。。

前言 互联网普及&#xff0c;用户群体庞大&#xff0c;用户体验非常重要。性能是一种指标&#xff0c;是软件系统对于及时性的符合程度。对于一个产品根据响应时间和吞吐量衡量性能的及时性&#xff0c;响应时间是一个http完整的请求流程时间之和。对用户而言&#xff0c;响应…

【开源】基于Qt5的ROS1/ROS2人机交互软件(支持地图编辑/多点导航)

本项目基于Qt5开发&#xff0c;基于CMake进行构建&#xff0c;可以实现一套代码同时在ROS1/ROS2系统中使用(本项目已接入CI,保证多ROS版本/系统版本可用性) 项目地址&#xff1a; https://github.com/chengyangkj/Ros_Qt5_Gui_App 软件在编译时会自动识别环境变量中的ROS1/ROS…

Redis -- list列表

只有克服了情感的波动&#xff0c;才能专心致志地追求事业的成功 目录 列表 list命令 lpush lpushx rpush rpushx lrange lpop rpop lindex linsert llen lrem ltrim 阻塞命令 小结 列表 列表相当于 数组或者顺序表。 列表类型是用来存储多个有序的字符串&…

C语言探索:水仙花数的奥秘与计算

摘要&#xff1a; 水仙花数&#xff0c;一种特殊的三位数&#xff0c;其各位数字的立方和等于该数本身。本文将详细介绍水仙花数的定义、性质&#xff0c;以及如何使用C语言来寻找100至999范围内的水仙花数。 目录 一、水仙花数的定义与性质 二、用C语言寻找100至999范围内的…

重写Sylar基于协程的服务器(2、配置模块的设计)

重写Sylar基于协程的服务器&#xff08;2、配置模块的设计&#xff09; 重写Sylar基于协程的服务器系列&#xff1a; 重写Sylar基于协程的服务器&#xff08;0、搭建开发环境以及项目框架 || 下载编译简化版Sylar&#xff09; 重写Sylar基于协程的服务器&#xff08;1、日志模…

【PostgresSQL系列】 ltree简介及基于SpringBoot实现 ltree数据增删改查

本文将对PostgresSQL中的ltree进行相关概念介绍&#xff0c;并以示例代码讲解ltree数据增删改查功能的实现。 作者&#xff1a;后端小肥肠 目录 1.前言 2. 基础概念 2.1. ltree 2.2. lquery 2.3. ltxtquery 2.4. ltree函数及操作符 2.4.1. ltree函数 2.4.2. ltree操作符…

软考 系统分析师系列知识点之需求管理(3)

接前一篇文章&#xff1a;软考 系统分析师系列知识点之需求管理&#xff08;2&#xff09; 所属章节&#xff1a; 第11章. 软件需求工程 第8节. 需求管理 11.8.4 需求跟踪 根据IEEE的定义&#xff0c;可跟踪性包含两个层面的含义&#xff1a;一个是开发过程的两个或多个产品之…

统计图表在线配置服务-百度 SugarBI的学习笔记

最近&#xff0c;有个产品要支持统计图表在线可配置&#xff0c;这样&#xff0c;当用户有新增统计指标的需求时&#xff0c;运维人员通过界面化配置&#xff0c;就可以增加统计指标了&#xff0c;不用开发写代码&#xff0c;画页面了。 上网查了下相关的组件&#xff0c;感觉…

OSPF排错

目录 实验拓扑图 实验要求 实验排错 故障一 故障现象 故障分析 故障解决 故障二 故障现象 故障分析 故障解决 故障三 故障现象 故障分析 故障解决 故障四 故障现象 故障分析 故障解决 故障五 故障现象 故障分析 故障解决 故障六 故障现象 故障分析 …

Linux服务详解

如有错误或有补充&#xff0c;以及任何改进的意见&#xff0c;请在评论区留下您的高见&#xff0c;同时文中给出大部分命令的示例&#xff0c;即是您暂时无法在Linux中查看&#xff0c;您也可以知道各种操作的功能以及输出 如果觉得本文写的不错&#xff0c;不妨点个赞&#x…

曲线拟合、多项式拟合、最小二乘法

最近在做自车轨迹预测的工作&#xff0c;遇到 曲线拟合、多项式拟合、最小二乘法这些概念有点不清晰&#xff0c; 做一些概念区别的总结&#xff1a; 曲线拟合用于查找一系列数据点的“最佳拟合”线或曲线。 大多数情况下&#xff0c;曲线拟合将产生一个函数&#xff0c;可用于…

[嵌入式软件][启蒙篇][仿真平台] STM32F103实现SPI控制OLED屏幕

上一篇&#xff1a; [嵌入式软件][启蒙篇][仿真平台] STM32F103实现LED、按键 [嵌入式软件][启蒙篇][仿真平台] STM32F103实现串口输出输入、ADC采集 [嵌入式软件][启蒙篇][仿真平台]STM32F103实现定时器 [嵌入式软件][启蒙篇][仿真平台] STM32F103实现IIC控制OLED屏幕 文章目…

Git 怎么设置用户的权限

在团队协作的软件开发中&#xff0c;对于版本控制系统Git来说&#xff0c;确保代码与数据的安全性至关重要。为了实现这一目标&#xff0c;Git提供了灵活且可定制的用户权限管理机制。下面将简单的探讨一下Git如何设置用户的权限&#xff0c;以及如何保护代码和数据。 用户身份…