From 46b95c03aeecf54adc5294bea5ad0e073b8e769e Mon Sep 17 00:00:00 2001 From: royqh1979 Date: Mon, 1 Nov 2021 00:40:11 +0800 Subject: [PATCH] work save: problem set ui done --- RedPandaIDE/RedPandaIDE_zh_CN.qm | Bin 99927 -> 100973 bytes RedPandaIDE/RedPandaIDE_zh_CN.ts | 1107 +++++++++++---------- RedPandaIDE/mainwindow.cpp | 154 +++ RedPandaIDE/mainwindow.h | 18 + RedPandaIDE/mainwindow.ui | 72 +- RedPandaIDE/widgets/ojproblemsetmodel.cpp | 226 ++++- RedPandaIDE/widgets/ojproblemsetmodel.h | 45 +- 7 files changed, 1097 insertions(+), 525 deletions(-) diff --git a/RedPandaIDE/RedPandaIDE_zh_CN.qm b/RedPandaIDE/RedPandaIDE_zh_CN.qm index 5fd850f705914d22a9819b2d755d1fc60618b463..d3f0fb8ffc3b6039564f1a7c1c36410348714126 100644 GIT binary patch delta 8004 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 TQ=R?S4*Q8Y9X>vf?N|IC-QnNV diff --git a/RedPandaIDE/RedPandaIDE_zh_CN.ts b/RedPandaIDE/RedPandaIDE_zh_CN.ts index 96e86334..2fdc02bf 100644 --- a/RedPandaIDE/RedPandaIDE_zh_CN.ts +++ b/RedPandaIDE/RedPandaIDE_zh_CN.ts @@ -430,62 +430,62 @@ Compiler - + Clean before rebuild failed. 重编译前的清理准备工作失败! - + Compile Result: 编译结果: - + - Errors: %1 - 错误数: %1 - + - Warnings: %1 - 警告数: %1 - + - Output Filename: %1 - 输出文件名: %1 - + - Output Size: %1 - 输出文件大小: %1 - + - Compilation Time: %1 secs - 编译时间: %1 秒 - + [Error] [错误] - + [Warning] [警告] - + [Info] [信息] - + [Note] [说明] - + The compiler process for '%1' failed to start. 无法启动编译器进程'%1'。 @@ -494,27 +494,27 @@ 无法启动编译进程。 - + The compiler process crashed after starting successfully. 编译进程启动后崩溃。 - + The last waitFor...() function timed out. waitFor()函数等待超时。 - + An error occurred when attempting to write to the compiler process. 在向编译进程输入内容时出错。 - + An error occurred when attempting to read from the compiler process. 在从编译进程读取内容时出错。 - + An unknown error occurred. 发生了未知错误。 @@ -549,45 +549,45 @@ CompilerManager + + + + + + + No compiler set + 无编译器设置 + - No compiler set - 无编译器设置 - - - - - - - No compiler set is configured. 没有配置编译器设置。 - - - - - + + + + + Can't start debugging. 无法启动调试器 - + Encoding not support 不支持字符编码 - + Clang only support utf-8 encoding. Clang只支持UTF-8编码 - + Strings in the program might be wrongly processed. 程序中的文字内容可能无法被正确处理和显示。 @@ -1065,35 +1065,35 @@ Are you really want to continue? 要剪切的内容超过了字符数限制! - + Print Document 打印文档 - - - + + + Ctrl+click for more info Ctrl+单击以获取更多信息 - - + + Symbol '%1' not found! 未找到符号'%1'! - + Break point condition 断点条件 - + Enter the condition of the breakpoint: 输入当前断点的生效条件: - + Readonly 只读 @@ -2806,44 +2806,44 @@ Are you really want to continue? 小熊猫C++ - - - - - - + + + + + + Issues 编译器 - + Compile Log 编译日志 - + File 文件 - + Tools 工具 - - + + Run 运行 - + Edit 编辑 - + Project 项目 @@ -2867,82 +2867,82 @@ Are you really want to continue? 资源 - - - - + + + + Debug 调试 - + Evaluate: 求值 - + Debug Console 调试主控台 - + Call Stack 调用栈 - + Breakpoints 断点 - + Locals 本地变量 - - + + Search 查找 - + History: 历史: - + Search Again 重新查找 - + Replace with: 替换为: - + Replace 替换 - + Close 关闭 - + Execute 运行 - - + + Code 代码 - + Window 窗口 @@ -2955,737 +2955,802 @@ Are you really want to continue? 工具栏2 - + New 新建 - + Ctrl+N Ctrl+N - + Open... 打开... - + Ctrl+O Ctrl+O - + Save 保存 - + Ctrl+S Ctrl+S - + Save As... 另存为... - + Save As 另存为 - + Save All 全部保存 - + Ctrl+Shift+S Ctrl+Shift+S - + Options 选项 - - - - - - - + + + + + + + Compile 编译 - + F9 F9 - + F10 F10 - + Undo 恢复 - + Ctrl+Z Ctrl+Z - + Redo 重做 - + Ctrl+Y Ctrl+Y - + Cut 剪切 - + Ctrl+X Ctrl+X - - - + + + Copy 复制 - + Ctrl+C Ctrl+C - - + + Paste 粘贴 - + Ctrl+V Ctrl+V - - + + Select All 选择全部 - + Ctrl+A Ctrl+A - + Indent 缩进 - + UnIndent 取消缩进 - + Toggle Comment 切换注释 - + Ctrl+/ Ctrl+/ - + Collapse All 全部收起 - + Uncollapse All 全部展开 - + Encode in ANSI 使用ANSI编码 - + Encode in UTF-8 使用UTF-8编码 - + Auto Detect 自动检测 - + Convert to ANSI 转换为ANSI编码 - + Convert to UTF-8 转换为UTF-8编码 - - + + Compile & Run 编译运行 - + F11 F11 - - + + Rebuild All 全部重编译 - + F12 F12 - + Stop Execution 停止执行 - + F6 F6 - + F5 F5 - + Step Over 单步跳过 - + F7 F7 - + Step Into 单步进入 - + + + Problem Set + 试题集 + + + + New Problem Set + 新建试题集 + + + + Add Problem + 添加试题 + + + + Remove Problem + 删除试题 + + + + + Save Problem Set + 保存试题集 + + + + + Load Problem Set + 载入试题集 + + + Memory 内存 - + Address Expression: Address: 地址表达式: - + Cancel 取消 - + TODO TODO - + Bookmark 书签 - + + + Problem + 试题 + + + + Add Probem Case + 添加试题案例 + + + + Remove Problem Case + Remove Problem Set + 删除试题集 + + + + Output + 输出 + + + + Input + 输入 + + + + Expected + 期望输出 + + + Help 帮助 - + Refactor 重构 - + Main 主工具栏 - + Compiler Set 编译器配置集 - - + + New Source File 新建源代码文件 - + Tab Tab - + Shift+Tab Shift+Tab - + F8 F8 - + Step Out 单步跳出 - + Ctrl+F8 Ctrl+F8 - + Run To Cursor 执行到光标处 - + Ctrl+F5 Ctrl+F5 - + Continue 继续执行 - + F4 F4 - + Add Watch... 添加监视 - + View CPU Window... 打开CPU信息窗口... - + Exit 退出 - + Find... 查找... - + Ctrl+F Ctrl+F - + Find in Files... 在文件中查找... - + Ctrl+Shift+F Ctrl+Shift+F - + Replace... 替换 - + Ctrl+R Ctrl+R - + Find Next 查找下一个 - + F3 F3 - + Find Previous 查找前一个 - + Shift+F3 Shift+F3 - + Remove Watch 删除监视值 - + Remove All Watches Remove All 删除全部监视值 - + Modify Watch... 修改监视值 - + Reformat Code 对代码重新排版 - + Ctrl+Shift+A Ctrl+Shift+A - + Go back 前一次编辑位置 - + Ctrl+Alt+Left Ctrl+Alt+Left - + Forward 后一次编辑位置 - + Ctrl+Alt+Right Ctrl+Alt+Right - + Ctrl+W Ctrl+W - + Close All 全部关闭 - + Ctrl+Shift+W Ctrl+Shift+W - + Maximize Editor 最大化编辑器 - + Ctrl+F11 Ctrl+F11 - + Next 下一窗口 - + Ctrl+Tab Ctrl+Tab - + Previous 前一窗口 - + Ctrl+Shift+Tab Ctrl+Shift+Tab - + Toggle breakpoint 切换断点 - + Ctrl+F4 Ctrl+F4 - - + + Clear all breakpoints 删除所有断点 - + Breakpoint property... 设置断点条件... - + Goto Declaration 跳转到声明处 - + Goto Definition 跳转到定义处 - + Find references 查找符号的引用 - + Open containing folder 打开所在的文件夹 - + Ctrl+B Ctrl+B - + Open a terminal here 打开命令行窗口 - + File Properties... 文件属性... - + Close Project 关闭项目 - + Project options 项目属性 - + New Project... 新建项目... - + New File 新建项目文件 - + Add to project... 添加到项目... - + Remove from project 从项目删除 - + View Makefile 查看Makefile - + Clean 清理构建文件 - + Open Folder in Explorer 在浏览器中打开 - + Open In Terminal 在终端中打开 - + About 关于 - - + + Rename Symbol 重命名符号 - + Shift+F6 Shift+F6 - + Print... 打印... - + Ctrl+P Ctrl+P - - + + Export As RTF 导出为RTF - - + + Export As HTML 导出为HTML - + Move To Other View 移动到其他视图 - + Ctrl+M Ctrl+M - + C/C++ Reference C/C++参考 - + EGE Manual EGE图形库手册 - + Add Bookmark 添加书签 - + Remove Bookmark 删除书签 - + Modify Bookmark Description 修改书签说明 - + Locate in Files View 在文件视图中定位 - - + + Open Folder 打开文件夹 - + Running Parameters... 运行参数... - + File Encoding 文件编码 - + Recent Files 文件历史 - - - - - - + + + + + + Debugging 正在调试 - - - - - - + + + + + + Running 正在运行 - - - - + + + - + + Compiling 正在编译 @@ -3694,599 +3759,627 @@ Are you really want to continue? 行:%1 列:%2 已选择:%3 总行数:%4 总长度:%5 - + Line:%1 Col:%2 Selected:%3 Lines:%4 Length:%5 Line: %1 Col: %2 Selected: %3 Lines: %4 Length: %5 行: %1 列: %2 已选择 :%3 总行数: %4 总长度: %5 - + Read Only 只读 - + Insert 插入 - + Overwrite 覆写 - + Close project 关闭项目 - + Are you sure you want to close %1? 你确定要关闭'%1'吗? - - + + Confirm 确认 - - - + + + Source file is not compiled. 源文件尚未编译。 - - + + Compile now? 现在编译? - - - + + + Source file is more recent than executable. 源文件比可执行程序新。 - + Recompile now? 重新编译? - + No compiler set 无编译器设置 - + No compiler set is configured. 没有配置编译器设置。 - + Can't start debugging. 无法启动调试器 - - + + Enable debugging 启用调试参数 - - + + You have not enabled debugging info (-g3) and/or stripped it from the executable (-s) in Compiler Options.<BR /><BR />Do you want to correct this now? 当前编译设置中未启用调试选项(-g3),或启用了信息剥除选项(-s)<br /><br/>是否纠正这一问题? - + Project not built 项目尚未构建 - + Project hasn't been built. Build it now? 项目尚未构建。是否构建? - + Host applcation missing 宿主程序不存在 - + DLL project needs a host application to run. 动态链接库(DLL)需要一个宿主程序来运行。 - + But it's missing. 但它不存在。 - + Host application not exists 宿主程序不存在 - + Host application file '%1' doesn't exist. 宿主程序'%1'不存在。 - + Recompile? 重新编译? - - + + Save last open info error 保存上次打开信息失败 - + Can't remove old last open information file '%1' 无法删除旧上次打开信息文件'%1' - + Can't save last open info file '%1' 无法保存上次打开信息文件'%1' - + Load last open info error 载入上次打开信息失败 - + Can't load last open info file '%1' 无法载入上次打开信息文件'%1' - + Copy all 全部复制 - - + + Clear 清除 - + Export 导出 - + Insert Snippet 插入代码段 - + + + Problem Set %1 + 试题集%1 + + + Red Panda Dev-C++ 小熊猫Dev-C++ - + Remove 删除 - + Remove All Bookmarks 删除全部书签 - + Modify Description 修改描述 - - - + + + Bookmark Description 书签描述 - - - + + + Description: 描述: - + Show debug logs in the debug console 在调试主控台中显示调试器输出 - + Remove this search 清除这次搜索 - + Clear all searches 删除所有搜索 - + Breakpoint condition... 断点条件... - + Break point condition 断点条件 - + Enter the condition of the breakpoint: 输入当前断点的生效条件: - + Remove All Breakpoints Remove all breakpoints 删除所有断点 - + Remove Breakpoint - 删除当前断点 + 删除当前断点 - + Rename File 重命名文件 - - + + Add Folder 添加文件夹 - + New folder 新文件夹 - + Folder name: 文件夹: - + Rename Folder 重命名 - + Remove Folder 删除文件夹 - + Sort By Type 按类型排序 - + Sort alphabetically 按名称排序 - + Show inherited members 显示继承的成员 - + Goto declaration 跳转到声明处 - + Goto definition 跳转到定义处 - + Open in Editor 在编辑器中打开 - + Open in External Program 使用外部程序打开 - + Open in Terminal 在终端中打开 - + Open in Windows Explorer 在Windows浏览器中打开 - + Character sets 字符集 - + %1 files autosaved 已自动保存%1个文件 - + Save project 保存项目 - + The project '%1' has modifications. 项目'%1'有改动。 - + Do you want to save it? 需要保存吗? - - - + + + + Save Error 保存失败 - + Change Project Compiler Set 改变项目编译器配置集 - + Change the project's compiler set will lose all custom compiler set options. 改变项目的编译器配置集会导致所有的自定义编译器选项被重置。 - + Do you really want to do that? 你真的想要做那些吗? - + Do you really want to clear all breakpoints in this file? 您真的要清除该文件的所有断点吗? - + New project 新建项目 - + Close %1 and start new project? 关闭'%1'以打开新项目? - + Folder not exist 文件夹不存在 - + Folder '%1' doesn't exist. Create it now? 文件夹'%1'不存在。是否创建? - + Can't create folder 无法创建文件夹 - + Failed to create folder '%1'. 创建文件夹'%1'失败。 - + Save new project as - + Red panda Dev-C++ project file (*.dev) 小熊猫Dev-C++项目文件 (*.dev) - + New project fail 新建项目失败 - + Can't assign project template 无法使用模板创建项目 - + untitled 无标题 - + New Project File Name 新的项目文件名 - + File Name: 文件名: - + File Already Exists! 文件已存在! - + File '%1' already exists! 文件'%1'已经存在! - + Add to project 添加到项目 - + Rename Error 重命名出错 - + Symbol '%1' is defined in system header. 符号'%1'在系统头文件中定义,无法修改。 - + New Name 新名称 - - + + Replace Error 替换出错 - + Can't open file '%1' for replace! 无法打开文件'%1'进行替换! - + Contents has changed since last search! 内容和上次查找时不一致。 - + Rich Text Format Files (*.rtf) RTF格式文件 (*.rtf) - + HTML Files (*.html) HTML文件 (*.html) - - - - - - - - - - - - - - - + + Problem %1 + 试题%1 + + + + + Problem Set Files (*.pbs) + 试题集文件 (*.pbs) + + + + Load Error + 载入失败 + + + + Problem Case %1 + 试题案例%1 + + + + + + + + + + + + + + + + + Error 错误 - + Recent Projects 项目历史 - + File '%1' was changed. 磁盘文件'%1'已被修改。 - + Reload its content from disk? 是否重新读取它的内容? - + File '%1' was removed. 磁盘文件'%1'已被删除。 - + Keep it open? 是否保持它在小熊猫C++中打开的编辑窗口? - + Open 打开 - + Compile Failed 编译失败 - + Run Failed 运行失败 - - + + Confirm Convertion 确认转换 - - + + The editing file will be saved using %1 encoding. <br />This operation can't be reverted. <br />Are you sure to continue? 当前编辑器中的文件将会使用%1编码保存。<br />这项操作无法被撤回。<br />你确定要继续吗? - + New Watch Expression 新监视表达式 - + Enter Watch Expression (it is recommended to use 'this->' for class members): 输入监视表达式 - + Parsing file %1 of %2: "%3" (%1/%2)正在解析文件"%3" - - + + Done parsing %1 files in %2 seconds 完成%1个文件的解析,用时%2秒 - + (%1 files per second) (每秒%1个文件) @@ -4343,92 +4436,92 @@ Are you really want to continue? Project - + File Not Found 未找到文件 - + Project file '%1' can't be found! 找不到项目文件'%1'! - + untitled 无标题 - + Can't save file 无法保存文件 - + Can't save file '%1' 无法保存文件'%1'. - + Error Load File 载入文件错误 - + File Exists 文件已存在 - + File '%1' is already in the project 文件'%1'已在项目中 - + Project Updated 项目已升级 - + Your project was succesfully updated to a newer file format! 已成功将项目升级到新的格式 - + If something has gone wrong, we kept a backup-file: '%1'... 旧项目文件备份在'%1'。 - + Settings need update 设置需要更新 - + The compiler settings format of Dev-C++ has changed. Dev-C++的编译器设置格式已发生改变。 - + Please update your settings at Project >> Project Options >> Compiler and save your project. 请在项目 >> 项目属性 >> 编译器设置中修改您的设置并保存您的项目 - + Compiler not found 未找到编译器 - + The compiler set you have selected for this project, no longer exists. 您为该项目设置的编译器不存在。 - + It will be substituted by the global compiler set. 它将会被全局编译器设置代替。 - + Developed using the Red Panda Dev-C++ IDE 使用小熊猫Dev-C++编辑器开发 @@ -4819,24 +4912,24 @@ Are you really want to continue? ProjectModel - + File exists 文件已存在 - + File '%1' already exists. Delete it now? 文件'%1'已存在。是否删除? - - + + Remove failed 删除失败 - - + + Failed to remove file '%1' 无法删除文件'%1' @@ -5055,7 +5148,7 @@ Are you really want to continue? QApplication - + Error 错误 @@ -5109,196 +5202,196 @@ Are you really want to continue? - - + + Error 错误 - + Can't create configuration folder %1 无法创建配置文件夹"%1" - + Can't write to configuration file %1 无法写入配置文件夹"%1" - + Can't load autolink settings 无法载入自动链接设置 - + + - The following %1 directories don't exist: 下列%1文件夹不存在: - - + + binary 二进制 - + No %1 directories have been specified. 未指定%1文件夹 - + C include C包含 + - C++ include C++包含 - - - - + + + + Cannot find the %1 "%2" 无法找到%1程序"%2" - + C options C语言选项 - + Support all ANSI standard C programs (-ansi) 支持所有ANSI标准C程序(-ansi) - + Do not recognize asm,inline or typeof as a keyword (-fno-asm) 不支持将asm、inline和typeof作为关键字(-fno-asm) - + Imitate traditional C preprocessors (-traditional-cpp) 模仿传统C预处理器行为(-traditional-cpp) - + Code Generation 代码生成 - + Optimize for the following machine (-march) 生成特定机器的专用指令(-march) - + Optimize less, while maintaining full compatibility (-tune) 完整兼容特定机器,较少优化(-tune) - + Enable use of specific instructions (-mx) 启用特定指令集(-mx) - + Optimization level (-Ox) 优化级别(-Ox) - + Compile with the following pointer size (-mx) 使用下列指针大小编译(-mx) - + Language standard (-std) 语言标准(-std) - + Profile 性能分析 - + Generate debugging information (-g3) 生成调试信息(-g3) - + Generate profiling info for analysis (-pg) 生成性能分析信息(-pg) - + Warnings 代码警告 - + Inhibit all warning messages (-w) 忽略所有警告信息(-w) - + Show most warnings (-Wall) 启用常见问题警告(-Wall) - + Show some more warnings (-Wextra) 启用更多问题警告(-Wextra) - + Check ISO C/C++/C++0x conformance (-pedantic) 检查ISO C/C++/C++0x语法一致性(-pedantic) - + Only check the code for syntax errors (-fsyntax-only) 只进行语法检查(不编译)(-fsyntax-only) - + Make all warnings into errors (-Werror) 将警告作为错误处理(-Werror) - + Abort compilation on first error (-Wfatal-errors) 遇到第一个错误后立即中止编译(-Wfatal-errors) - + Linker 链接器 - + Link an Objective C program (-lobjc) 链接Objective-C程序 (-lobjc) - + Do not use standard system libraries (-nostdlib) 不使用标准库和系统启动文件(-nostdlib) - + Do not create a console window (-mwindows) 不产生控制台窗口(-mwindows) - + Strip executable (-s) 剥除附加信息(-s) @@ -5307,43 +5400,43 @@ Are you really want to continue? 链接Ojbective C程序(-lobjc) - + Output 输出 - + Put comments in generated assembly code (-fverbose-asm) 在生成的汇编代码中加入注释(-fverbose-asm) - + Use pipes instead of temporary files during compilation (-pipe) 编译时使用管道而不是临时文件(-pipe) - + Do not assemble, compile and generate the assemble code (-S) 只生成汇编代码(-S) - - + + Confirm 确认 - + The following problems were found during validation of compiler set "%1": 在验证编译器设置"%1"时遇到了下列问题: - + Compiler set not configuared. 未配置编译器设置。 - + Would you like Red Panda C++ to search for compilers in the following locations: <BR />'%1'<BR />'%2'? 您需要小熊猫C++在下列位置搜索编译器吗:<br />%1<br />%2 @@ -5677,18 +5770,20 @@ Are you really want to continue? 无标题 - + constructor 构造函数 - + destructor 析构函数 + + Can't open file '%1' for read. 无法读取文件'%1'. @@ -5698,6 +5793,11 @@ Are you really want to continue? Can't open file '%1' for write. 无法写入文件'%1'. + + + Can't parse problem set file '%1':%2 + 无法解析试题集文件"%1":%2 + RegisterModel @@ -6095,7 +6195,7 @@ Are you really want to continue? 自动链接 - + @@ -6171,7 +6271,7 @@ Are you really want to continue? 杂项 - + Program Runner 程序运行 @@ -6394,12 +6494,12 @@ Are you really want to continue? SynEditStringList - + Can't open file '%1' for read! 无法读取文件'%1'! - + Can't open file '%1' for save! 无法写入文件'%2'! @@ -6608,122 +6708,121 @@ Are you really want to continue? 自动计算缩进 - Add Indent after { and : - 在{和:的下一行添加缩进 + 在{和:的下一行添加缩进 - + Replace tab with spaces 自动使用空格代替制表符(Tab) - + Tab Width 制表符(Tab)宽度 - + Show Indent Lines 显示缩进提示线 - + Indent Line Color 缩进提示线颜色 - + Fill Indents 填充缩进区域 - + Caret 光标 - + Move caret to the first non-space char in the current line when press HOME key 按下HOME键时,光标定位在本行的第一个非空格字符处 - + Move caret to the last non-space char in the current line when press END key 按下End键时,光标定位在本行的最后一个非空格字符处 - + Keep X position of the caret when moving vertically 在上下移动光标时,记住起始时光标所在栏数 - + Caret for inserting mode 插入状态下的光标 - + Use text color as caret color 使用文字颜色作为光标颜色 - + Scroll 滚动条 - + Auto hide scroll bars 自动隐藏滚动条 - + Can scroll the last char to the left edge of the editor 可以将每行末尾字符滚动到编辑器最左侧 - + Can scroll the last line to the top edge of the editor 可以将最后一行滚动到编辑器最上方 - + Page Up/Down scrolls half a page 翻页键只滚动半页 - + Forces page scroll to be one line less 在滚动页时少滚动一行 - + Mouse Wheel Scroll Speed 鼠标滚轮卷轴速度(行) - + Show right edge line 显示右边缘线 - + Right egde width 右边缘宽度 - + Right edge line color 右边缘颜色 - + Caret for overwriting mode 覆写状态下的光标 - + Caret Color 光标颜色 diff --git a/RedPandaIDE/mainwindow.cpp b/RedPandaIDE/mainwindow.cpp index 2ca0458b..8331362a 100644 --- a/RedPandaIDE/mainwindow.cpp +++ b/RedPandaIDE/mainwindow.cpp @@ -205,6 +205,18 @@ MainWindow::MainWindow(QWidget *parent) ui->searchView,&QTreeView::expandAll); ui->replacePanel->setVisible(false); + mOJProblemSetNameCounter++; + mOJProblemSetModel.rename(tr("Problem Set %1").arg(mOJProblemSetNameCounter)); + ui->lstProblemSet->setModel(&mOJProblemSetModel); + ui->lstProblemCases->setModel(&mOJProblemModel); + connect(ui->lstProblemSet->selectionModel(), + &QItemSelectionModel::currentRowChanged, + this, &MainWindow::onProblemSetIndexChanged); + connect(ui->lstProblemCases->selectionModel(), + &QItemSelectionModel::currentRowChanged, + this, &MainWindow::onProblemCaseIndexChanged); + ui->tabProblem->setVisible(false); + //files view ui->treeFiles->setModel(&mFileSystemModel); mFileSystemModel.setReadOnly(true); @@ -2503,6 +2515,63 @@ void MainWindow::onFilesViewContextMenu(const QPoint &pos) menu.exec(ui->treeFiles->mapToGlobal(pos)); } +void MainWindow::onProblemSetIndexChanged(const QModelIndex ¤t, const QModelIndex &previous) +{ + QModelIndex idx = current; +// if (previous.isValid()) { +// QModelIndex caseIdx = ui->lstProblemCases->currentIndex(); +// if (caseIdx.isValid()) { +// POJProblemCase problemCase = mOJProblemModel.getCase(caseIdx.row()); +// problemCase->input = ui->txtProblemCaseInput->toPlainText(); +// problemCase->expected = ui->txtProblemCaseExpected->toPlainText(); +// problemCase->output = ui->txtProblemCaseOutput->toPlainText(); +// } +// } + if (!idx.isValid()) { + ui->btnRemoveProblem->setEnabled(false); + ui->tabProblem->setVisible(false); + } else { + ui->btnRemoveProblem->setEnabled(true); + POJProblem problem = mOJProblemSetModel.problem(idx.row()); + mOJProblemModel.setProblem(problem); + ui->lblProblem->setText(problem->name); + ui->tabProblem->setVisible(true); + openCloseBottomPanel(true); + ui->tabMessages->setCurrentWidget(ui->tabProblem); + } +} + +void MainWindow::onProblemCaseIndexChanged(const QModelIndex ¤t, const QModelIndex &previous) +{ + QModelIndex idx = current; + if (previous.isValid()) { + POJProblemCase problemCase = mOJProblemModel.getCase(previous.row()); + problemCase->input = ui->txtProblemCaseInput->toPlainText(); + problemCase->expected = ui->txtProblemCaseExpected->toPlainText(); + problemCase->output = ui->txtProblemCaseOutput->toPlainText(); + } + if (idx.isValid()) { + POJProblemCase problemCase = mOJProblemModel.getCase(idx.row()); + if (problemCase) { + ui->btnRemoveProblemCase->setEnabled(true); + ui->txtProblemCaseInput->setText(problemCase->input); + ui->txtProblemCaseInput->setReadOnly(false); + ui->txtProblemCaseExpected->setText(problemCase->expected); + ui->txtProblemCaseExpected->setReadOnly(false); + ui->txtProblemCaseOutput->setText(problemCase->output); + ui->txtProblemCaseOutput->setReadOnly(true); + return; + } + } + ui->btnRemoveProblemCase->setEnabled(false); + ui->txtProblemCaseInput->clear(); + ui->txtProblemCaseInput->setReadOnly(true); + ui->txtProblemCaseExpected->clear(); + ui->txtProblemCaseExpected->setReadOnly(true); + ui->txtProblemCaseOutput->clear(); + ui->txtProblemCaseOutput->setReadOnly(true); +} + void MainWindow::onShowInsertCodeSnippetMenu() { mMenuInsertCodeSnippet->clear(); @@ -4819,3 +4888,88 @@ void MainWindow::on_actionRun_Parameters_triggered() ); } + +void MainWindow::on_btnNewProblemSet_clicked() +{ + mOJProblemSetNameCounter++; + mOJProblemSetModel.create(tr("Problem Set %1").arg(mOJProblemSetNameCounter)); +} + + +void MainWindow::on_btnAddProblem_clicked() +{ + int startCount = mOJProblemSetModel.count(); + QString name; + while (true) { + name = tr("Problem %1").arg(startCount); + if (!mOJProblemSetModel.problemNameUsed(name)) + break; + } + POJProblem problem = std::make_shared(); + problem->name = name; + mOJProblemSetModel.addProblem(problem); + ui->lstProblemSet->setCurrentIndex(mOJProblemSetModel.index(mOJProblemSetModel.count()-1)); +} + + +void MainWindow::on_btnRemoveProblem_clicked() +{ + QModelIndex idx = ui->lstProblemSet->currentIndex(); + if (!idx.isValid()) + return; + mOJProblemSetModel.removeProblem(idx.row()); +} + + +void MainWindow::on_btnSaveProblemSet_clicked() +{ + QString fileName = QFileDialog::getSaveFileName( + this, + tr("Save Problem Set"), + QDir().absolutePath(), + tr("Problem Set Files (*.pbs)")); + if (!fileName.isEmpty()) { + try { + mOJProblemSetModel.saveToFile(fileName); + } catch (FileError& error) { + QMessageBox::critical(this,tr("Save Error"), + error.reason()); + } + } +} + + +void MainWindow::on_btnLoadProblemSet_clicked() +{ + QString fileName = QFileDialog::getOpenFileName( + this, + tr("Load Problem Set"), + QDir().absolutePath(), + tr("Problem Set Files (*.pbs)")); + if (!fileName.isEmpty()) { + try { + mOJProblemSetModel.loadFromFile(fileName); + } catch (FileError& error) { + QMessageBox::critical(this,tr("Load Error"), + error.reason()); + } + } +} + + +void MainWindow::on_btnAddProblemCase_clicked() +{ + int startCount = mOJProblemModel.count(); + QString name; + while (true) { + name = tr("Problem Case %1").arg(startCount); + if (!mOJProblemSetModel.problemNameUsed(name)) + break; + } + POJProblemCase problemCase = std::make_shared(); + problemCase->name = name; + problemCase->testState = ProblemCaseTestState::NoTested; + mOJProblemModel.addCase(problemCase); + ui->lstProblemCases->setCurrentIndex(mOJProblemModel.index(mOJProblemModel.count()-1)); +} + diff --git a/RedPandaIDE/mainwindow.h b/RedPandaIDE/mainwindow.h index ac69a54b..da52071b 100644 --- a/RedPandaIDE/mainwindow.h +++ b/RedPandaIDE/mainwindow.h @@ -18,6 +18,7 @@ #include "toolsmanager.h" #include "widgets/labelwithmenu.h" #include "widgets/bookmarkmodel.h" +#include "widgets/ojproblemsetmodel.h" QT_BEGIN_NAMESPACE @@ -211,6 +212,8 @@ private slots: void onDebugConsoleContextMenu(const QPoint& pos); void onFileEncodingContextMenu(const QPoint& pos); void onFilesViewContextMenu(const QPoint& pos); + void onProblemSetIndexChanged(const QModelIndex ¤t, const QModelIndex &previous); + void onProblemCaseIndexChanged(const QModelIndex ¤t, const QModelIndex &previous); void onShowInsertCodeSnippetMenu(); @@ -433,6 +436,18 @@ private slots: void on_actionRun_Parameters_triggered(); + void on_btnNewProblemSet_clicked(); + + void on_btnAddProblem_clicked(); + + void on_btnRemoveProblem_clicked(); + + void on_btnSaveProblemSet_clicked(); + + void on_btnLoadProblemSet_clicked(); + + void on_btnAddProblemCase_clicked(); + private: Ui::MainWindow *ui; EditorList *mEditorList; @@ -473,6 +488,9 @@ private: PTodoParser mTodoParser; PToolsManager mToolsManager; QFileSystemModel mFileSystemModel; + OJProblemSetModel mOJProblemSetModel; + OJProblemModel mOJProblemModel; + int mOJProblemSetNameCounter; bool mCheckSyntaxInBack; bool mOpenClosingBottomPanel; diff --git a/RedPandaIDE/mainwindow.ui b/RedPandaIDE/mainwindow.ui index 84ef4751..86554a78 100644 --- a/RedPandaIDE/mainwindow.ui +++ b/RedPandaIDE/mainwindow.ui @@ -306,8 +306,11 @@ + + New Problem Set + - ... + New Problem Set @@ -317,8 +320,11 @@ + + Add Problem + - ... + Add Problem @@ -328,8 +334,11 @@ + + Remove Problem + - ... + Remove Problem @@ -339,8 +348,11 @@ + + Save Problem Set + - ... + Save Problem Set @@ -350,8 +362,11 @@ + + Load Problem Set + - ... + Load Problem Set @@ -1099,7 +1114,7 @@ 0 - + Problem @@ -1146,8 +1161,11 @@ + + Add Probem Case + - ... + Add Probem Case @@ -1157,8 +1175,11 @@ + + Remove Problem Case + - ... + Remove Problem Case @@ -1166,6 +1187,41 @@ + + + + Qt::Vertical + + + + + + + Run Current Case + + + Run Current Case + + + + :/icons/images/newlook24/069-run.png:/icons/images/newlook24/069-run.png + + + + + + + Run All Cases + + + Run All Cases + + + + :/icons/images/newlook24/021-Debug-Continue.png:/icons/images/newlook24/021-Debug-Continue.png + + + diff --git a/RedPandaIDE/widgets/ojproblemsetmodel.cpp b/RedPandaIDE/widgets/ojproblemsetmodel.cpp index e4b37925..7f6650ea 100644 --- a/RedPandaIDE/widgets/ojproblemsetmodel.cpp +++ b/RedPandaIDE/widgets/ojproblemsetmodel.cpp @@ -1,26 +1,240 @@ #include "ojproblemsetmodel.h" +#include +#include +#include +#include +#include "../utils.h" + OJProblemSetModel::OJProblemSetModel(QObject *parent) : QAbstractListModel(parent) { } -OJProblemCaseModel::OJProblemCaseModel(QObject *parent): QAbstractListModel(parent) +void OJProblemSetModel::clear() +{ + beginRemoveRows(QModelIndex(),0,mProblemSet.problems.count()-1); + mProblemSet.problems.clear(); + endRemoveRows(); +} + +int OJProblemSetModel::count() +{ + return mProblemSet.problems.count(); +} + +void OJProblemSetModel::create(const QString& name) +{ + mProblemSet.name = name; + clear(); +} + +void OJProblemSetModel::rename(const QString &newName) +{ + if (mProblemSet.name!=newName) + mProblemSet.name = newName; +} + +QString OJProblemSetModel::name() +{ + return mProblemSet.name; +} + +void OJProblemSetModel::addProblem(POJProblem problem) +{ + beginInsertRows(QModelIndex(), mProblemSet.problems.count(), mProblemSet.problems.count()); + mProblemSet.problems.append(problem); + endInsertRows(); +} + +POJProblem OJProblemSetModel::problem(int index) +{ + return mProblemSet.problems[index]; +} + +void OJProblemSetModel::removeProblem(int index) +{ + Q_ASSERT(index>=0 && index < mProblemSet.problems.count()); + mProblemSet.problems.removeAt(index); +} + +bool OJProblemSetModel::problemNameUsed(const QString &name) +{ + foreach (const POJProblem& problem, mProblemSet.problems) { + if (name == problem->name) + return true; + } + return false; +} + +void OJProblemSetModel::removeAllProblems() +{ + clear(); +} + +void OJProblemSetModel::saveToFile(const QString &fileName) +{ + QFile file(fileName); + if (file.open(QFile::WriteOnly | QFile::Truncate)) { + QJsonObject obj; + obj["name"]=mProblemSet.name; + QJsonArray problemsArray; + foreach (const POJProblem& problem, mProblemSet.problems) { + QJsonObject problemObj; + problemObj["name"]=problem->name; + QJsonArray cases; + foreach (const POJProblemCase& problemCase, problem->cases) { + QJsonObject caseObj; + caseObj["name"]=problemCase->name; + caseObj["input"]=problemCase->input; + caseObj["expected"]=problemCase->expected; + cases.append(caseObj); + } + problemObj["cases"]=cases; + problemsArray.append(problemObj); + } + obj["problems"]=problemsArray; + QJsonDocument doc; + doc.setObject(obj); + file.write(doc.toJson()); + file.close(); + } else { + throw FileError(QObject::tr("Can't open file '%1' for read.") + .arg(fileName)); + } +} + +void OJProblemSetModel::loadFromFile(const QString &fileName) +{ + QFile file(fileName); + if (file.open(QFile::ReadOnly)) { + QByteArray content = file.readAll(); + QJsonParseError error; + QJsonDocument doc(QJsonDocument::fromJson(content,&error)); + if (error.error!=QJsonParseError::NoError) { + throw FileError(QObject::tr("Can't parse problem set file '%1':%2") + .arg(fileName) + .arg(error.errorString())); + } + beginResetModel(); + QJsonObject obj = doc.object(); + mProblemSet.name = obj["name"].toString(); + mProblemSet.problems.clear(); + QJsonArray problemsArray = obj["problems"].toArray(); + foreach (const QJsonValue& problemVal, problemsArray) { + QJsonObject problemObj = problemVal.toObject(); + POJProblem problem = std::make_shared(); + problem->name = problemObj["name"].toString(); + QJsonArray casesArray = problemObj["cases"].toArray(); + foreach (const QJsonValue& caseVal, casesArray) { + QJsonObject caseObj = caseVal.toObject(); + POJProblemCase problemCase = std::make_shared(); + problemCase->name = caseObj["name"].toString(); + problemCase->input = caseObj["input"].toString(); + problemCase->expected = caseObj["expected"].toString(); + problemCase->testState = ProblemCaseTestState::NoTested; + problem->cases.append(problemCase); + } + mProblemSet.problems.append(problem); + } + endResetModel(); + } else { + throw FileError(QObject::tr("Can't open file '%1' for read.") + .arg(fileName)); + } +} + +int OJProblemSetModel::rowCount(const QModelIndex &parent) const +{ + return mProblemSet.problems.count(); +} + +QVariant OJProblemSetModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + if (role == Qt::DisplayRole) { + return mProblemSet.problems[index.row()]->name; + } + return QVariant(); +} + +OJProblemModel::OJProblemModel(QObject *parent): QAbstractListModel(parent) { } -const POJProbelm &OJProblemCaseModel::problem() const +const POJProblem &OJProblemModel::problem() const { return mProblem; } -void OJProblemCaseModel::setProblem(const POJProbelm &newProblem) +void OJProblemModel::setProblem(const POJProblem &newProblem) { - mProblem = newProblem; + if (newProblem!=mProblem) { + beginResetModel(); + mProblem = newProblem; + endResetModel(); + } } -int OJProblemCaseModel::rowCount(const QModelIndex &parent) const +void OJProblemModel::addCase(POJProblemCase problemCase) { - + if (mProblem==nullptr) + return; + beginInsertRows(QModelIndex(),mProblem->cases.count(),mProblem->cases.count()); + mProblem->cases.append(problemCase); + endInsertRows(); +} + +void OJProblemModel::removeCase(int index) +{ + if (mProblem==nullptr) + return; + Q_ASSERT(index >= 0 && index < mProblem->cases.count()); + beginRemoveRows(QModelIndex(),index,index); + mProblem->cases.removeAt(index); + endRemoveRows(); +} + +POJProblemCase OJProblemModel::getCase(int index) +{ + if (mProblem==nullptr) + return POJProblemCase(); + return mProblem->cases[index]; +} + +void OJProblemModel::clear() +{ + if (mProblem==nullptr) + return; + beginRemoveRows(QModelIndex(),0,mProblem->cases.count()-1); + mProblem->cases.clear(); + endRemoveRows(); +} + +int OJProblemModel::count() +{ + if (mProblem == nullptr) + return 0; + return mProblem->cases.count(); +} + +int OJProblemModel::rowCount(const QModelIndex &) const +{ + if (mProblem==nullptr) + return 0; + return mProblem->cases.count(); +} + +QVariant OJProblemModel::data(const QModelIndex &index, int role) const +{ + if (!index.isValid()) + return QVariant(); + if (mProblem==nullptr) + return QVariant(); + if (role == Qt::DisplayRole) { + return mProblem->cases[index.row()]->name; + } + return QVariant(); } diff --git a/RedPandaIDE/widgets/ojproblemsetmodel.h b/RedPandaIDE/widgets/ojproblemsetmodel.h index 74e1208f..ea4934c5 100644 --- a/RedPandaIDE/widgets/ojproblemsetmodel.h +++ b/RedPandaIDE/widgets/ojproblemsetmodel.h @@ -14,7 +14,9 @@ struct OJProblemCase { QString name; QString input; QString expected; - ProblemCaseTestState testState; + + QString output; // no persistence + ProblemCaseTestState testState; // no persistence }; using POJProblemCase = std::shared_ptr; @@ -24,17 +26,27 @@ struct OJProblem { QVector cases; }; -using POJProbelm = std::shared_ptr; +using POJProblem = std::shared_ptr; -class OJProblemCaseModel: public QAbstractListModel { +struct OJProblemSet { + QString name; + QVector problems; +}; + +class OJProblemModel: public QAbstractListModel { Q_OBJECT public: - explicit OJProblemCaseModel(QObject *parent = nullptr); - const POJProbelm &problem() const; - void setProblem(const POJProbelm &newProblem); + explicit OJProblemModel(QObject *parent = nullptr); + const POJProblem &problem() const; + void setProblem(const POJProblem &newProblem); + void addCase(POJProblemCase problemCase); + void removeCase(int index); + POJProblemCase getCase(int index); + void clear(); + int count(); private: - POJProbelm mProblem; + POJProblem mProblem; // QAbstractItemModel interface public: @@ -47,6 +59,25 @@ class OJProblemSetModel : public QAbstractListModel Q_OBJECT public: explicit OJProblemSetModel(QObject *parent = nullptr); + void clear(); + int count(); + void create(const QString& name); + void rename(const QString& newName); + QString name(); + void addProblem(POJProblem problem); + POJProblem problem(int index); + void removeProblem(int index); + bool problemNameUsed(const QString& name); + void removeAllProblems(); + void saveToFile(const QString& fileName); + void loadFromFile(const QString& fileName); +private: + OJProblemSet mProblemSet; + + // QAbstractItemModel interface +public: + int rowCount(const QModelIndex &parent) const override; + QVariant data(const QModelIndex &index, int role) const override; }; #endif // OJPROBLEMSETMODEL_H