Java中规模软件开发实训——简单的文本编辑器(代码注释详解)

news2025/1/4 15:26:39

博主:命运之光

🌸专栏:Python星辰秘典

🐳专栏:web开发(html css js)

❤️专栏:Java经典程序设计

☀️博主的其他文章:点击进入博主的主页

前言:在现代社会中,计算器是我们生活中不可或缺的工具之一。它们可以轻松地进行各种数值计算,从简单的加减乘除到复杂的科学运算,为我们提供了快捷准确的计算结果。但你是否曾想过,我们可以亲手打造一个属于自己的计算器应用程序,体验计算世界的奇妙之旅?本文将带领你进入计算器应用程序的开发领域。我们将使用Java编程语言和Swing图形界面库,从零开始构建一个简单但功能强大的计算器应用程序。无论你是计算机科学专业的学生,还是对编程和应用开发感兴趣的爱好者,这个实践项目都将为你提供一个宝贵的机会来深入了解应用程序开发的流程和技术。

目录

实验内容

问题描述

项目功能

项目分析

文件打开

主题切换

文字颜色

文字大小

编译功能

运行功能

项目设计

设计流程图如下

系统实现

项目源码

总结


实验内容

以下是该项目的主要内容:
创建一个窗口界面,包含一个代码编辑区域、行号显示区域和按钮面板。
代码编辑区域使用JTextPane组件实现,可以进行文本编辑、撤销和重做操作。
行号显示区域使用JTextArea组件实现,显示当前代码编辑区域的行号。
按钮面板包含编译按钮和运行按钮,用于执行编译和运行操作。
提供文件操作功能,包括打开和保存文件。
提供主题设置功能,可以切换代码编辑区域和行号显示区域的背景颜色。
提供字体设置功能,包括设置字体颜色和大小。
支持缩放功能,可以通过鼠标滚轮和Ctrl键进行文本的放大和缩小。
在代码编辑区域中输入代码后,可以进行编译操作,将代码保存到文件并执行javac命令进行编译。
编译结果将显示在消息对话框中,显示编译成功或编译失败的消息。
可以运行编译后的代码,调用系统命令行执行java Main命令,并获取代码运行的输出结果。
运行结果将显示在消息对话框中。


问题描述

该项目主要解决以下问题:
代码编辑器:提供一个功能完善的代码编辑器,用于编辑和显示代码文件的内容。该编辑器具有基本的文本编辑功能,包括插入、删除、撤销和重做等操作。用户可以在编辑器中编写代码,并对代码进行格式化和调整。
文件操作:支持文件的打开和保存操作。用户可以通过打开菜单项选择要打开的文件,并将文件内容加载到代码编辑器中。用户还可以使用保存菜单项将代码编辑器中的内容保存到文件中。
编译代码:提供编译代码的功能。用户可以点击编译按钮,将代码保存到名为 "Main.java" 的文件中,并调用系统命令行执行 javac 命令对代码进行编译。编译成功后,会显示编译成功的消息对话框;编译失败则显示编译失败的消息对话框。
运行代码:支持代码的运行操作。用户可以点击运行按钮,调用系统命令行执行 java Main 命令运行编译后的代码,并获取代码运行的输出结果。运行结果会显示在消息对话框中。
行号显示:提供行号显示功能。代码编辑器的左侧会显示行号区域,显示当前编辑代码的行数。随着代码的插入、删除和换行等操作,行号会相应更新。
主题切换:支持编辑器主题的切换。用户可以通过主题菜单选择不同的主题,包括亮色主题和暗色主题,以满足个性化的显示需求。
字体设置:支持编辑器字体的设置。用户可以通过字体颜色菜单项选择编辑器的字体颜色,通过字体大小菜单项调整编辑器的字体大小。
缩放文本:支持文本缩放功能。用户可以通过按住 Ctrl 键并滚动鼠标滚轮来放大或缩小代码编辑器中的文本内容。
通过解决上述问题,该项目提供了一个功能完善的代码编辑器,方便用户编辑、保存、编译和运行代码,并提供了一些个性化的设置选项,提高了代码编写和调试的效率。


项目功能

该项目最后实现以下功能:
代码编辑功能:提供一个代码编辑器,支持基本的文本编辑操作,如插入、删除、撤销和重做等。用户可以在编辑器中编写代码,并对代码进行格式化和调整。
文件操作功能:支持打开和保存代码文件。用户可以通过打开菜单项选择要打开的文件,并将文件内容加载到代码编辑器中。用户还可以使用保存菜单项将代码编辑器中的内容保存到文件中。
代码编译功能:提供编译代码的功能。用户可以点击编译按钮,将代码保存到名为 "Main.java" 的文件中,并调用系统命令行执行 javac 命令对代码进行编译。编译成功后,会显示编译成功的消息对话框;编译失败则显示编译失败的消息对话框。
代码运行功能:支持代码的运行操作。用户可以点击运行按钮,调用系统命令行执行 java Main 命令运行编译后的代码,并获取代码运行的输出结果。运行结果会显示在消息对话框中。
行号显示功能:在代码编辑器中显示行号区域,以便用户可以轻松地跟踪和定位代码的行数。
主题切换功能:支持编辑器主题的切换。用户可以通过主题菜单选择不同的主题,包括亮色主题和暗色主题,以满足个性化的显示需求。
字体设置功能:支持编辑器字体的设置。用户可以通过字体颜色菜单项选择编辑器的字体颜色,通过字体大小菜单项调整编辑器的字体大小。
文本缩放功能:支持放大或缩小代码编辑器中的文本内容。用户可以通过按住 Ctrl 键并滚动鼠标滚轮来调整文本的大小。
通过以上功能,该项目提供了一个功能完善的代码编辑器,使用户能够方便地编辑、保存、编译和运行代码,并提供了一些个性化设置选项,以提高编码和调试的效率。


项目分析

主界面

文件打开

主题切换

文字颜色

文字大小

编译功能

运行功能

创建窗口:使用 JFrame 类创建了一个文本编译器的窗口,并设置了窗口标题和大小。
编辑区域:使用 JTextPane 类创建了一个代码编辑区域,支持文本编辑和代码高亮显示。
行号显示:通过在编辑区域左侧添加一个 JTextArea 来实现行号的显示,并与编辑区域进行联动。
按钮:创建了编译按钮和运行按钮,用于执行代码编译和运行操作。
文件操作:通过文件选择器实现打开和保存文件功能。
主题切换:通过菜单栏实现了切换主题的功能,包括亮色和暗色主题。
文本编辑功能:支持撤销和重做操作,支持通过鼠标滚轮和快捷键进行文本缩放。
代码编译与运行:将编辑区域的代码保存到名为 "Main.java" 的文件中,调用系统命令行执行 javac 命令进行编译。运行代码则调用系统命令行执行 "java Main" 命令运行编译后的代码,并获取代码运行的输出结果。
字体颜色和大小:通过菜单栏实现了设置字体颜色和大小的功能。
其他事件监听:注册了各种事件监听器,包括文本内容变化监听器、撤销操作监听器、鼠标滚轮监听器等。
该代码虽然实现了基本的文本编辑器和简单代码编译运行的功能,但在实际使用中可能需要进一步优化和改进


项目设计

设计流程图如下


系统实现

项目源码

package 文本编辑器;
import javax.swing.*;
import javax.swing.border.Border;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.text.AttributeSet;
import javax.swing.text.SimpleAttributeSet;
import javax.swing.text.StyleConstants;
import javax.swing.text.StyleContext;
import javax.swing.undo.UndoManager;
import java.awt.*;
import java.awt.event.*;
import java.io.*;

public class CodeEditor extends JFrame {
    // 声明私有成员变量
    private JTextPane codePane;              // 代码编辑区域
    private JTextArea lineNumberArea;        // 行号显示区域
    private JButton compileButton;           // 编译按钮
    private JButton runButton;               // 运行按钮
    private JFileChooser fileChooser;        // 文件选择器
    private JMenu themeMenu;                 // 主题菜单
    private JMenuItem lightThemeItem;        // 亮色主题菜单项
    private JMenuItem darkThemeItem;         // 暗色主题菜单项
    private JMenu editMenu;                  // 编辑菜单
    private JMenuItem fontColorItem;         // 字体颜色菜单项
    private JMenuItem fontSizeItem;          // 字体大小菜单项
    private UndoManager undoManager;         // 撤销管理器
    private double zoomScale = 1.0;          // 缩放比例


    /**
     * 构造函数,创建一个代码编辑器窗口。
     */
    public CodeEditor() {
        setTitle("文本编译器");                        // 设置窗口标题
        setSize(800, 600);                             // 设置窗口大小
        setDefaultCloseOperation(EXIT_ON_CLOSE);      // 设置窗口关闭时的默认操作为退出应用程序

        // 创建主面板
        JPanel mainPanel = new JPanel(new BorderLayout());
        getContentPane().add(mainPanel);

        codePane = new JTextPane();                     // 创建代码编辑区域
        lineNumberArea = new JTextArea("1");            // 创建行号显示区域
        lineNumberArea.setBackground(Color.LIGHT_GRAY);
        lineNumberArea.setEditable(false);//lineNumberArea.setEditable(false); 将 lineNumberArea 设置为不可编辑,意味着用户无法通过键盘输入或编辑该文本区域中的内容。
        lineNumberArea.setFocusable(false);//lineNumberArea.setFocusable(false); 将 lineNumberArea 设置为不可获取焦点,意味着用户无法通过点击或键盘操作将焦点聚焦在该文本区域上。
       
        //这段代码的作用是为 lineNumberArea 文本区域设置一个带有黄色边框和内边距的边框样式。
        Border border = BorderFactory.createMatteBorder(0, 0, 0, 5, Color.yellow);//创建了一个 MatteBorder 对象,该边框在右侧有5个像素的黄色边框,而其他三个边界没有边框。这个边框对象被赋值给 border 变量。
        lineNumberArea.setBorder(BorderFactory.createCompoundBorder(border, BorderFactory.createEmptyBorder(0, 5, 0, 5)));//将 lineNumberArea 的边框设置为一个复合边框。这个复合边框由两个部分组成:第一个部分是之前创建的 MatteBorder 对象 border,它定义了右侧的黄色边框;第二个部分是一个空边框 BorderFactory.createEmptyBorder(0, 5, 0, 5),它在内部创建了一个没有任何边框的空白区域,其上、下、左、右各有5个像素的空白。通过将这两个边框组合在一起,实现了一个具有黄色边框和内边距的边框效果。

        
        
        //创建滚动条
        JScrollPane codeScrollPane = new JScrollPane(codePane);//创建了一个名为 codeScrollPane 的滚动窗格,并将 codePane 添加到其中。codePane 是一个文本区域或其他可滚动的组件,通过将其添加到滚动窗格中,可以在需要时启用滚动功能,以便在需要时浏览大量的文本内容。
        JScrollPane lineNumberScrollPane = new JScrollPane(lineNumberArea);//创建了一个名为 lineNumberScrollPane 的滚动窗格,并将 lineNumberArea 添加到其中。lineNumberArea 是另一个文本区域或可滚动的组件,它可能是用于显示行号的区域。同样,将其添加到滚动窗格中可以实现在需要时滚动内容。
        lineNumberScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_NEVER);//设置了 lineNumberScrollPane 的垂直滚动条策略。JScrollPane.VERTICAL_SCROLLBAR_NEVER 指示不显示垂直滚动条。这意味着在 lineNumberScrollPane 中,即使内容超过可见区域,也不会显示垂直滚动条。

        
        //显示行号
        codeScrollPane.setRowHeaderView(lineNumberScrollPane);//将 lineNumberScrollPane 设置为 codeScrollPane 的行头视图,用于显示行号
        mainPanel.add(codeScrollPane, BorderLayout.CENTER);      // 将代码编辑区域和行号显示区域添加到主面板的中心位置

        
        
        
        // 创建按钮面板
        JPanel buttonPanel = new JPanel();
        compileButton = new JButton("编译");            // 创建编译按钮
        runButton = new JButton("运行");                // 创建运行按钮
        buttonPanel.add(compileButton);
        buttonPanel.add(runButton);
        mainPanel.add(buttonPanel, BorderLayout.SOUTH);         // 将按钮面板添加到主面板的南部位置

        fileChooser = new JFileChooser();               // 创建文件选择器

        
        //创建菜单栏
        JMenuBar menuBar = new JMenuBar();               // 创建菜单栏
        JMenu fileMenu = new JMenu("文件");               // 创建文件菜单
        JMenuItem openItem = new JMenuItem("打开");      // 创建打开菜单项
        JMenuItem saveItem = new JMenuItem("保存");      // 创建保存菜单项
        fileMenu.add(openItem);
        fileMenu.add(saveItem);
        menuBar.add(fileMenu);
        //同上
        themeMenu = new JMenu("主题");                    // 创建主题菜单
        lightThemeItem = new JMenuItem("白色");           // 创建亮色主题菜单项
        darkThemeItem = new JMenuItem("粉色");            // 创建暗色主题菜单项
        themeMenu.add(lightThemeItem);
        themeMenu.add(darkThemeItem);
        menuBar.add(themeMenu);
        //同上
        editMenu = new JMenu("编辑");                     // 创建编辑菜单
        fontColorItem = new JMenuItem("文字颜色");         // 创建字体颜色菜单项
        fontSizeItem = new JMenuItem("文字大小");          // 创建字体大小菜单项
        editMenu.add(fontColorItem);
        editMenu.add(fontSizeItem);
        menuBar.add(editMenu);

        //解释:setJMenuBar(menuBar); 是一个 Swing 窗口的方法调用,用于将给定的菜单栏(menuBar)设置为窗口的菜单栏。
        setJMenuBar(menuBar);                            // 设置菜单栏

        
        
        
        // 注册事件监听器
        
        //e -> openFile() 表示一个处理事件的函数或方法。当事件被触发时,该函数会被调用,并执行 openFile() 方法来处理打开文件的操作。
        openItem.addActionListener(e -> openFile());                   // 注册打开菜单项的点击事件监听器
        saveItem.addActionListener(e -> saveFile());                   // 注册保存菜单项的点击事件监听器

        compileButton.addActionListener(e -> compileCode());           // 注册编译按钮的点击事件监听器
        runButton.addActionListener(e -> runCode());                   // 注册运行按钮的点击事件监听器

        lightThemeItem.addActionListener(e -> setLightTheme());        // 注册亮色主题菜单项的点击事件监听器
        darkThemeItem.addActionListener(e -> setDarkTheme());          // 注册暗色主题菜单项的点击事件监听器

        fontColorItem.addActionListener(e -> setFontColor());          // 注册字体颜色菜单项的点击事件监听器
        fontSizeItem.addActionListener(e -> setFontSize());            // 注册字体大小菜单项的点击事件监听器

        
        
        // 注册文本内容变化监听器//显示行号的监听事件
        codePane.getDocument().addDocumentListener(new DocumentListener() {
            public void changedUpdate(DocumentEvent e) {
                updateLineNumbers();                                    // 文本内容发生改变时更新行号显示
            }

            public void insertUpdate(DocumentEvent e) {
                updateLineNumbers();                                    // 文本内容插入时更新行号显示
            }

            public void removeUpdate(DocumentEvent e) {
                updateLineNumbers();                                    // 文本内容删除时更新行号显示
            }
        });

        
        //创建了一个 UndoManager 对象来管理撤销和恢复操作,并将其与 codePane 的文档关联起来,以便在文档发生可撤销的编辑操作时进行记录和管理。这样,用户可以通过调用 undoManager 的方法来撤销和恢复编辑操作的历史记录。
        //撤销操作
        undoManager = new UndoManager();
        codePane.getDocument().addUndoableEditListener(e -> undoManager.addEdit(e.getEdit()));   // 注册撤销操作监听器

        
        
        
        
        //为 codePane 添加了一个键盘监听器,用于处理用户按键事件,并实现了对撤销和重做操作的响应。
        //通过键盘进行撤销
        codePane.addKeyListener(new KeyAdapter() {
            @Override
            public void keyPressed(KeyEvent e) {
                if (e.isControlDown()) {//e.isControlDown() 检查用户是否按下了 Ctrl 键。
                    if (e.getKeyCode() == KeyEvent.VK_Z) {//通过 e.getKeyCode() 获取按下的键的键码。
                        if (undoManager.canUndo()) {
                            undoManager.undo();                           // Ctrl+Z 撤销操作
                        }
                    } else if (e.getKeyCode() == KeyEvent.VK_Y) {
                        if (undoManager.canRedo()) {
                            undoManager.redo();                           // Ctrl+Y 重做操作
                        }
                    }
                }
            }
        });

        
        
        //通过鼠标滑轮控制字体大小
        codePane.addMouseWheelListener(new MouseAdapter() {
            @Override
            public void mouseWheelMoved(MouseWheelEvent e) {
                if (e.isControlDown()) {//e.isControlDown() 检查用户是否按下了 Ctrl 键。
                    int notches = e.getWheelRotation();//e.getWheelRotation() 获取鼠标滚轮的滚动方向,返回的值可以是正数或负数。
                    if (notches < 0) {
                        zoomIn();                                         // Ctrl+鼠标滚轮向上缩放文本
                    } else {
                        zoomOut();                                        // Ctrl+鼠标滚轮向下缩放文本
                    }
                    e.consume();
                }
            }
        });
    }

    /**
     * 放大文本缩放比例。
     */
    private void zoomIn() {
        zoomScale += 0.1;                               // 将缩放比例增加0.1
        applyZoom();                                    // 应用新的缩放比例
    }

    /**
     * 缩小文本缩放比例。
     */
    private void zoomOut() {
        zoomScale -= 0.1;                               // 将缩放比例减少0.1
        applyZoom();                                    // 应用新的缩放比例
    }

    /**
     * 应用当前的缩放比例到代码编辑区域的字体大小。
     */
    private void applyZoom() {
        if (zoomScale < 0.1) {
            zoomScale = 0.1;                            // 缩放比例不低于0.1
        }
        codePane.setFont(codePane.getFont().deriveFont((float) (codePane.getFont().getSize() * zoomScale)));   // 根据缩放比例设置新的字体大小
    }


    
    
    
    
    
    /**
     * 更新行号显示区域的内容,根据当前代码编辑区域的行数进行更新。
     */
    private void updateLineNumbers() {
        String code = codePane.getText();                                // 获取当前代码编辑区域的文本内容
        int lines = code.split("\n", -1).length;                         // 计算行数,使用换行符作为分隔符进行拆分		具体解释:具体来说,code.split("\n", -1) 使用换行符 (\n) 将字符串 code 拆分成多个行,并返回一个包含拆分后的子字符串的数组。第二个参数 -1 表示在拆分过程中保留所有空行。然后,.length 方法用于获取拆分后的字符串数组的长度,即行数。因此,int lines = code.split("\n", -1).length; 将计算出字符串 code 中的行数,并将其存储在 lines 变量中。
        StringBuilder lineNumbers = new StringBuilder();                // 用于存储行号内容的字符串构建器
        for (int i = 1; i <= lines; i++) {
            lineNumbers.append(i).append("\n");                          // 将行号添加到字符串构建器中,并添加换行符		具体解释:lineNumbers.append(i).append("\n"); 表示将行号 i 和换行符 "\n" 连接起来,并将它们追加到字符串构建器 lineNumbers 的末尾。append() 是字符串构建器的方法,用于将指定的内容添加到构建器的末尾。在这里,lineNumbers.append(i) 将行号 i 添加到字符串构建器中,然后 append("\n") 将换行符 "\n" 添加到字符串构建器中,实现行号和换行符的连接。
        }
        lineNumberArea.setText(lineNumbers.toString());                  // 将行号内容设置到行号显示区域
    }

    /**
     * 打开文件操作,选择文件并将文件内容读取到代码编辑区域中。
     */
    private void openFile() {
        int returnValue = fileChooser.showOpenDialog(this);               // 显示文件选择对话框,并获取用户的操作结果
        
        
        if (returnValue == JFileChooser.APPROVE_OPTION) {//如果用户选择了打开文件的操作(returnValue == JFileChooser.APPROVE_OPTION),则继续执行打开文件的操作。
        	
            File selectedFile = fileChooser.getSelectedFile();           // 获取用户选择的文件	详细解释:通过 fileChooser.getSelectedFile() 获取用户选择的文件,并将其存储在 selectedFile 变量中。
            try {
                BufferedReader reader = new BufferedReader(new FileReader(selectedFile));   // 创建文件读取器	详细解释:使用 BufferedReader 类来创建一个文件读取器 (reader),并将选择的文件作为参数传递给它
                StringBuilder content = new StringBuilder();             // 用于存储文件内容的字符串构建器	详细解释:创建一个 StringBuilder 对象 content,用于存储文件内容。通过循环读取文件中的每一行,并将其逐行添加到 content 字符串构建器中。
                String line;
                while ((line = reader.readLine()) != null) {
                    content.append(line).append("\n");                    // 逐行读取文件内容,并添加到字符串构建器中
                }
                reader.close();                                          // 关闭文件读取器	读取文件完成后,关闭文件读取器 (reader.close())。
                codePane.setText(content.toString());                     // 将文件内容设置到代码编辑区域中	将 content.toString() 得到的文件内容字符串设置为代码编辑区域 (codePane) 的文本,即显示文件内容在代码编辑区域中。
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }


    /**
     * 保存文件操作,将代码编辑区域的内容保存到文件中。
     */
    private void saveFile() {
        int returnValue = fileChooser.showSaveDialog(this);                      // 显示文件保存对话框,并获取用户的操作结果
        if (returnValue == JFileChooser.APPROVE_OPTION) {
            File selectedFile = fileChooser.getSelectedFile();                   // 获取用户选择的文件
            try {
                FileWriter writer = new FileWriter(selectedFile);                // 创建文件写入器
                writer.write(codePane.getText());                                 // 将代码编辑区域的内容写入文件
                writer.close();                                                   // 关闭文件写入器
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    
    
    
    
    
    
    
    /**
     * 编译代码操作,将代码编辑区域的内容保存到名为 "Main.java" 的文件中,并调用系统命令行执行 javac 命令进行编译。
     * 根据编译结果显示编译成功或编译失败的消息对话框。
     */
    private void compileCode() {
        String code = codePane.getText();                                         // 获取代码编辑区域的文本内容
        try {
            File file = new File("Main.java");                                    // 创建名为 "Main.java" 的文件
            FileWriter writer = new FileWriter(file);                             // 创建文件写入器
            writer.write(code);                                                   // 将代码写入文件
            writer.close();                                                       // 关闭文件写入器

            Process process = Runtime.getRuntime().exec("javac Main.java");        // 调用系统命令行执行 javac 命令进行编译
            int exitCode = process.waitFor();                                      // 等待编译进程结束,并获取退出码
            if (exitCode == 0) {
                JOptionPane.showMessageDialog(this, "编译成功!");                // 编译成功,显示编译成功的消息对话框
            } else {
                JOptionPane.showMessageDialog(this, "编译失败!");                // 编译失败,显示编译失败的消息对话框
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    /**
     * 运行代码操作,调用系统命令行执行 "java Main" 命令运行编译后的代码,并获取代码运行的输出结果。
     * 将输出结果显示在消息对话框中。
     */
    private void runCode() {
        try {
            Process process = Runtime.getRuntime().exec("java Main");               // 调用系统命令行执行 "java Main" 命令运行编译后的代码
            BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()));  // 创建用于读取代码运行输出的字符缓冲读取器

            StringBuilder output = new StringBuilder();                            // 用于存储代码运行输出结果的字符串构建器
            String line;
            while ((line = reader.readLine()) != null) {
                output.append(line).append("\n");                                  // 逐行读取代码运行输出,并添加到字符串构建器中
            }

            JOptionPane.showMessageDialog(this, "输出结果:\n" + output.toString());   // 显示运行结果的消息对话框,包含代码运行的输出结果
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    /**
     * 设置浅色主题,将代码编辑区域和行号区域的背景颜色调整为相应的浅色。
     */
    private void setLightTheme() {
        codePane.setBackground(Color.white);                     // 将代码编辑区域的背景颜色设置为白色
        lineNumberArea.setBackground(Color.LIGHT_GRAY);          // 将行号区域的背景颜色设置为浅灰色
    }

    /**
     * 设置粉色主题,将代码编辑区域和行号区域的背景颜色调整为相应的暗色。
     */
    private void setDarkTheme() {
        codePane.setBackground(Color.pink);                      // 将代码编辑区域的背景颜色设置为粉色
        lineNumberArea.setBackground(Color.GRAY);                // 将行号区域的背景颜色设置为灰色
    }

    /**
     * 设置文字颜色,通过颜色选择器选择新的文字颜色,并将其应用到代码编辑区域的前景色。
     * 用户选择颜色后,将选中的颜色设置为代码编辑区域的前景色。
     */
    private void setFontColor() {
        Color selectedColor = JColorChooser.showDialog(this, "选择文字颜色", codePane.getForeground());  // 打开颜色选择器对话框,获取用户选择的颜色
        /*详细解释:代码中的 JColorChooser.showDialog(this, "选择文字颜色",
         *  codePane.getForeground()) 表示在当前界面 (this) 上显示一个颜色选择器对话框。对话框的标题是 "选择文字颜色",
         *  并且默认选中的颜色是 codePane 的前景色(文本颜色)。
         *  当用户在颜色选择器对话框中选择了一个颜色后,JColorChooser.showDialog() 方法会返回用户选择的颜色值,
         *  并将其存储在 selectedColor 变量中。*/
        if (selectedColor != null) {
            codePane.setForeground(selectedColor);               // 将选中的颜色设置为代码编辑区域的前景色
        }
    }


    /**
     * 设置字体大小,通过输入对话框获取用户输入的字体大小,并将其应用到代码编辑区域的字体。
     * 用户输入的字体大小应为正整数,如果输入无效或不是正整数,将显示相应的错误消息对话框。
     */
    private void setFontSize() {
        String input = JOptionPane.showInputDialog(this, "请输入字体大小:", codePane.getFont().getSize());  // 打开输入对话框,获取用户输入的字体大小   
        //JOptionPane.showInputDialog() 方法会返回用户输入的字体大小,并将其存储在 input 变量中。
        try {
            int fontSize = Integer.parseInt(input);                                 // 将用户输入的字体大小转换为整数	通过 Integer.parseInt(input) 将用户输入的字体大小转换为整数,并存储在 fontSize 变量中。
            if (fontSize > 0) {//通过条件判断 fontSize > 0 来验证用户输入的字体大小是否有效。如果字体大小大于 0,则执行以下操作:
                Font currentFont = codePane.getFont();//codePane.getFont() 获取当前代码编辑区域的字体。
                codePane.setFont(new Font(currentFont.getFontName(), currentFont.getStyle(), fontSize));  // 将新的字体大小应用到代码编辑区域的字体
                
                //创建一个新的 Font 对象,并将原字体的字体名称、字体样式以及新的字体大小应用到该对象中。使用 codePane.setFont() 方法将新的字体设置为代码编辑区域的字体。
                
                
            } else {
            	
            	//如果字体大小无效(小于等于 0),则使用 JOptionPane.showMessageDialog(this, "无效的字体大小。请输入一个正整数。") 显示一个错误消息对话框,提示用户输入一个有效的字体大小。
                JOptionPane.showMessageDialog(this, "无效的字体大小。请输入一个正整数。");      // 显示无效字体大小的错误消息对话框
                
                
            }
        } catch (NumberFormatException e) {
            JOptionPane.showMessageDialog(this, "无效输入。请输入有效的字体大小。");           // 显示无效输入的错误消息对话框
        }
    }


    /**
     * 主方法,程序的入口点。在Swing事件调度线程中创建并显示代码编辑器。
     * 在主方法中,使用SwingUtilities.invokeLater() 方法调度创建和显示代码编辑器的任务,
     * 确保在正确的线程上进行Swing组件的创建和更新操作。
     */
    public static void main(String[] args) {
    	
    	
    	
    	/*
    	 * 使用 SwingUtilities.invokeLater() 方法
    	 * 来确保在事件分派线程(Event Dispatch Thread)中执行代码。
    	 * 这是因为 Swing 组件需要在事件分派线程中进行操作,以保证界面的响应性和线程安全性。
    	 * */
        SwingUtilities.invokeLater(() -> {
        	
        	
        	/*
        	 * SwingUtilities.invokeLater(() -> {...})
        	 * 使用 Lambda 表达式创建一个 Runnable 对象,其中的代码会在事件分派线程中执行。*/
        	
        	
            CodeEditor editor = new CodeEditor();  // 创建代码编辑器对象
            editor.setVisible(true);              // 设置代码编辑器可见
        });
    }

}

总结

在本实验中,我们实现了一个简单的文本编辑器(CodeEditor),该编辑器具有打开、保存、编译、运行代码等功能,并支持撤销、重做、缩放文本等操作。

#include<iostream>
using namespace std;
int main(){
    cout<<"看完了点个赞再走吧,谢谢啦(●'◡'●)❤";
    return 0;
}

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

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

相关文章

k8s如何伸缩应用程序和执行滚动更新

一、伸缩应用程序 我们创建了一个 Deployment (opens new window)&#xff0c;然后通过 服务 (opens new window)提供访问 Pod 的方式。我们发布的 Deployment 只创建了一个 Pod 来运行我们的应用程序。当流量增加时&#xff0c;我们需要对应用程序进行伸缩操作以满足系统性能…

第五章:L2JMobius学习 – 快速部署L2JMobius汉化版

L2JMobius是一套开源的 LineageII 的服务器端代码&#xff0c;使用Java语言编写。在前面的章节中&#xff0c;我们安装了mariadb10数据库以及jdk17运行环境&#xff0c;这两个是必须的。紧接着&#xff0c;我们又安装了eclipse开发工具&#xff0c;然后创建了“L2J_Mobius”Jav…

按键输入实验(stm32)

目录 按键的相关代码key.ckey.h LED的相关代码BEEP的相关代码beep.cbeep.h main.c代码的相关说明相关硬件说明实验结果 说明&#xff1a;以下内容参考正点原子资料 按键的相关代码 key.c void KEY_Init(void) //IO初始化 { GPIO_InitTypeDef GPIO_InitStructure;RCC_APB2Peri…

ModaHub魔搭社区:可视化的AI原生云向量数据库 Milvus 2.2.9 :JSON、PartitionKey、Dynamic Schema

目录 新特性 功能增强 其他优化 问题修复 亮点颇多、精彩程度堪比大版本的 Milvus 2.2.9 来啦&#xff01; 随着 LLM 的持续火爆&#xff0c;众多应用开发者将目光投向了向量数据库领域&#xff0c;而作为开源向量数据库的领先者&#xff0c;Milvus 也充分吸收了大量来自社…

ROS:话题名称设置

目录 一、 前言二、rosrun设置话题重映射三、launch文件设置话题重映射四、编码设置话题名称4.1C4.1.1全局名称4.1.2相对名称4.1.3私有名称 4.2Python 实现4.2.1全局名称4.2.2相对名称4.2.3私有名称 一、 前言 在ROS中节点名称可能出现重名的情况&#xff0c;同理话题名称也可…

[攻防世界] [RE] [APK] app2

解题思路 导入jadx查看manifest.xml 查看主函数并未发现有价值的东西&#xff0c;于是查看manifest.xml中主函数下一个<activity> 截取FileDataActivity代码 package com.tencent.testvuln;import android.os.Bundle; import android.widget.TextView; import com.tence…

2022年真题 - 15 - 磁盘管理(vdo磁盘)

磁盘管理 - vdo磁盘 题目配置验证配置题目 StorageSrv - 磁盘管理 在 storagesrv 上新加一块 10G 磁盘;创建 vdo 磁盘,并开启 vdo 磁盘的重删和压缩;名字为 vdodisk,大小为150G,文件系统为 ext4;并设置开机自动挂载。挂载到 /vdodata。配置 新加一块 10G 磁盘; 安装…

驱动 作业 day4

编写LED灯的驱动&#xff0c;创建三个设备文件&#xff0c;每个设备文件和一个LED灯绑定&#xff0c;当操作这个设备文件时只能控制设备文件对应的这盏灯。 此时没有安装led2 和led3的驱动所以会打开设备文件失败 装完以后就可以正常控制了 以下是设备现象 head.h ubuntuu…

docker 的整体架构及各模块组件 《深入docker底层原理》

1.Docker 整体架构 Docker 是一个 C/S 模式的架构&#xff0c;后端是一个松耦合架构&#xff0c;模块各司其职。 1、用户是使用 Docker Client 与 Docker Daemon 建立通信&#xff0c;并发送请求给后者。 2、Docker Daemon 作为 Docker 架构中的主体部分&#xff0c;首先提供…

Windows如何设置自动关闭未响应的程序?Windows设置自动关闭未响应的程序方法,带图详解

Windows系统程序经常出现程序未响应现象&#xff0c;如何通过注册表使其自动关闭呢 1、首先快捷键winR唤出【运行】 输入regedit 2、确定后就打开了注册表编辑器&#xff0c;定位到【HKEY_CURREnT_UsER\Control panel\desktop】项下 3、在右侧找【AutoEndTasks】数值数据&#…

yolo.h5文件问题的解决 - 吴恩达深度学习:目标检测之YOLO算法

1.下载下载yad2k: git clone https://github.com/allanzelener/yad2k.git 这里面顺便有yad2k.py文件 2.下载yolov2.cfg https://github.com/pjreddie/darknet/tree/master/cfg 3.下载yolov2.weights http://pjreddie.com/media/files/yolo.weights 需要这三个文件 自己去githup…

易模为真人3D手办制作带来了创新

3d打印技术是一项近年来迅速发展的先进制造技术&#xff0c;逐渐在各个领域展现出无限的潜力。其中&#xff0c;3d打印真人手办成为了一个备受关注的领域。在市面上&#xff0c;我们常常可以看到一些热门动漫角色或明星的真人3d手办&#xff0c;逼真的细节和完美的再现度让人们…

iOS上架报错:无法添加以供审核

无法提交以供审核 要开始审核流程 必须提供以下项目 您必须为要添加的 app 提供版权信息。 您在提交 app 审核时遇到的问题是因为需要提供版权信息&#xff0c;而您的 app 缺少相关的版权信息。以下是解决此问题的步骤&#xff1a; 确认您是否拥有 app 的版权&#xff1a;在提…

mac苹果电脑,怎么把mkv转换mp4格式

mac苹果电脑&#xff0c;怎么把mkv转换mp4格式&#xff1f;如果你是一名mac苹果电脑的用户&#xff0c;在电脑上下载到mkv格式的视频后会发现它使用起来非常的麻烦&#xff0c;甚至不能直接打开播放。mkv其实也是一种时间比较久远的视频文件格式&#xff0c;但是不知道是什么原…

eNSP-OSPF组播拓展

OSPF组播拓展 文章目录 OSPF组播拓展一、拓扑结构二、基础配置三、测试验证 启动 OSPF 协议后 &#xff0c; OSPF 将向本地所有运行 OSPF 协议的接口以组播224.0.0.5的形式发送hello报 文 &#xff1b; hello 报文中将携带本地 RID 值 &#xff0c; 以及本地已知的邻居的RID值&…

nginx反向代理 404 问题

发现我们设置了反向代理没有起作用&#xff0c;最后发现原来是伪静态惹得祸 解决nginx添加反向代理代码不生效-与原rewrite伪静态规则冲突了 以thinkphp官方给的伪静态为例 if (!-e $request_filename){rewrite ^(.*)$ /index.php?s$1 last; break;}仔细研究发现发现问…

Android OpenGL ES 学习(十三) -离屏渲染FBO(截图)RBO, OES转 FBO

Android OpenGL ES 学习(一) – 基本概念 Android OpenGL ES 学习(二) – 图形渲染管线和GLSL Android OpenGL ES 学习(三) – 绘制平面图形 Android OpenGL ES 学习(四) – 正交投屏 Android OpenGL ES 学习(五) – 渐变色 Android OpenGL ES 学习(六) – 使用 VBO、VAO 和 EB…

威胁和漏洞管理增强远程 IT 安全性

威胁和漏洞管理是保护组织设备和数据的主动方法。它可以帮助管理员识别漏洞并检查安全设置是否薄弱。通过使用此方法&#xff0c;可以在任何弱点成为安全漏洞之前对其进行修复。 对远程威胁和漏洞管理工具的需求 随着越来越多的员工远程工作&#xff0c;网络攻击的可能性也在…

基于Unity 3D实现的融合多元素风格游戏

完整资料进入【数字空间】查看——baidu搜索"writebug" 1.综合描述 1.1 产品背景 随着人们对游戏的追求&#xff0c;越来越多的优秀游戏被开发出来。目前&#xff0c;多风格元素游戏深受大众喜爱&#xff0c;例如绝地求生融合了 FPS 射击与生 存元素&#xff0c;守…

【软件分析/静态分析】chapter4 课程05/06 数据流分析—基础(Data Flow Analysis—Foundations)

&#x1f517; 课程链接&#xff1a;李樾老师和谭天老师的&#xff1a; 南京大学《软件分析》课程05&#xff08;Data Flow Analysis - Foundations I&#xff09;_哔哩哔哩_bilibili 目录 第四章 数据流分析——基础 4.1 从另一个视角看迭代算法&#xff08;Iterative Algor…