code_edit项目实例08:纯代码自定义代码组件(QPlainTextEdit实现)

创建类

 

 

 

#ifndef MYCODEEDIT_H
#define MYCODEEDIT_H

#include <QPlainTextEdit>
// 全局声明,否则在MyCodeEdit使用这个类的时候会提示error: unknown type name 'LineNumberWidget'
class LineNumberWidget;

class MyCodeEdit : public QPlainTextEdit
{
    Q_OBJECT
public:
    explicit MyCodeEdit(QWidget *parent = nullptr);
    //
    void lineNumberWidgetPaintEvent(QPaintEvent *event);
    // 鼠标点击事件处理,在类外部调用
    void lineNumberWidgetMousePessEvent(QMouseEvent *event);
    // 把滚轮任务提交给MyCodeEditor
    void lineNumberWidgetWheelEvent(QWheelEvent *event);

private slots:
    // 行高亮
    void HighLightCurrentLine();
    void UpdateLineNumberWidget(QRect rect, int dy);
    // 更新行号显示边距
    void updateLIneNumberWIdgetWidth();

protected:
    // 重写resizeEvent方法,方法名要与继承的方法名称保持一致
    void resizeEvent(QResizeEvent *event);

private:
    // 初始字体
    void InitFont();
    // 信号槽绑定
    void InitConnection();
    // 高亮
    void InitHighLighter();
    // 获取行号显示区域宽度
    int GetLineNumberWidgetWidth();
    // 新建任务的成员变量
    LineNumberWidget *lineNumberWidget;

signals:

};

// 新建一个类,用于重写QPaintEvent方法,用于行号显示
class LineNumberWidget : public QWidget
{
public:
    // 构造函数,参数为父级部件(即需要显示行号的组件,如 QPlainTextEdit 等)
    explicit LineNumberWidget(MyCodeEdit *editor = nullptr) : QWidget(editor) {
        codeEditor = editor; // 初始化 codeEditor 指针为传入的 editor 参数
    }

protected:
    // 绘制事件,重写 QWidget 的 paintEvent 方法
    void paintEvent(QPaintEvent *event) override {
        // 调用 codeEditor 的 lineNumberWidgetPaintEvent 方法,实现行号的绘制
        // 把绘制任务交给codeEditor,lineNumberWidgetPaintEvent用于重写QPaintEvent方法
        codeEditor->lineNumberWidgetPaintEvent(event);
    }
    // 重写鼠标点击事件
    void mousePressEvent(QMouseEvent *event) override {
        // 把鼠标点击任务交给MyCodeEdit
        codeEditor->lineNumberWidgetMousePessEvent(event);
    }
    // 重写鼠标滚轮事件
    void wheelEvent(QWheelEvent *event) override {
        // 把滚轮任务提交给MyCodeEditor
        codeEditor->lineNumberWidgetWheelEvent(event);

    }
private:
    MyCodeEdit *codeEditor; // 指向需要显示行号的组件,一切绘制任务交给codeEditor
};

#endif // MYCODEEDIT_H
#include "mycodeedit.h"
#include "myhightlighter.h"
#include <QDebug>
#include <QPainter>
#include <QFontMetricsF>
#include <QScrollBar>

MyCodeEdit::MyCodeEdit(QWidget *parent) : QPlainTextEdit(parent)
{
    lineNumberWidget = new LineNumberWidget(this); // this 相当于是把MyCodeEdit作为参数传进LineNumberWidget类中
    // 信号槽绑定
    InitConnection();
    // 初始字体
    InitFont();
    // 高亮
    InitHighLighter();
    // 行高亮
    HighLightCurrentLine();
    // 设置高亮显示边距,与行号显示宽度保持一致
    updateLIneNumberWIdgetWidth();
    // 水平滚动条, 不自动换行
    setLineWrapMode(QPlainTextEdit::NoWrap);
}

// 信号槽绑定
void MyCodeEdit::InitConnection()
{
    // cursor
    connect(this, SIGNAL(cursorPositionChanged()), this, SLOT(HighLightCurrentLine()));
    // blockCount更新边距
    connect(this, SIGNAL(blockCountChanged(int)), this, SLOT(updateLIneNumberWIdgetWidth()));
    // updateRequest
    connect(this, SIGNAL(updateRequest(QRect, int)), this, SLOT(UpdateLineNumberWidget(QRect, int)));
}

void MyCodeEdit::InitFont()
{
    // 初始字体
    this->setFont(QFont("Consolas", 14));
}

// 高亮
void MyCodeEdit::InitHighLighter()
{
    new MyHightLighter(document());
}

// 获取行号显示区域宽度
int MyCodeEdit::GetLineNumberWidgetWidth()
{
    // 获取合适的宽度
    // 旧版本的 Qt 中,没有 `horizontalAdvance` 函数,这个函数在较新的版本中才被引入。
    // return 30 + QString::number(blockCount() + 1).length() * fontMetrics().horizontalAdvance(QChar('0'));
    // return 30 + QString::number(blockCount() + 1).length() * QFontMetricsF(font()).horizontalAdvance(QChar('0'));
    return 30 + QString::number(blockCount() + 1).length() * QFontMetricsF(font()).width(QChar('0'));
}

void MyCodeEdit::HighLightCurrentLine()
{
    // 声明一个额外选择区域列表 `QList<QTextEdit::ExtraSelection> extraSelections
    QList<QTextEdit::ExtraSelection> extraSelections; // //声明一个额外选择区域列表,用于多行高亮
    // `QTextEdit::ExtraSelection` 是一个用于在 `QTextEdit` 中添加额外选择区域的类。可以使用它来实现高亮、定位光
    QTextEdit::ExtraSelection selection;  //获取文本光标
    selection.format.setBackground(QColor(0, 100, 100, 20)); // 设置光标所在行的背景颜色和背景透明度
    selection.format.setProperty(QTextFormat::FullWidthSelection, true);
    selection.cursor = textCursor(); // 设置为文本编辑器的当前光标
    // 将 `selection` 添加到 `extraSelections` 中,并将整个 `extraSelections` 设置为额外选择区域。
    extraSelections.append(selection);
    setExtraSelections(extraSelections); // 将选择列表设置为额外选择区域
}

void MyCodeEdit::UpdateLineNumberWidget(QRect rect, int dy)
{
    // dy纵向的偏移值
    if (dy) {
        lineNumberWidget->scroll(0, dy);
    } else {
        lineNumberWidget->update(0, rect.y(), GetLineNumberWidgetWidth(), rect.height());
    }
}

// 更新行号显示边距
void MyCodeEdit::updateLIneNumberWIdgetWidth()
{
    // 设置高亮显示边距,与行号显示宽度保持一致
    setViewportMargins(GetLineNumberWidgetWidth(), 0, 0, 0);
}

// 重写resizeEvent方法
void MyCodeEdit::resizeEvent(QResizeEvent *event)
{
    // 调用父类 QPlainTextEdit 的 resizeEvent 事件
    // `QPlainTextEdit::resizeEvent(event)` 是一个父类 `QPlainTextEdit` 中的函数,它会在编辑器的大小改变时自动触发。
    QPlainTextEdit::resizeEvent(event);
    // 自适应宽度和高度,重新设置行号部件的位置和大小
    // `0, 0`:是行号部件左上角的坐标点,这表示行号部件在编辑器中的左上角。
    // `contentsRect().height()`:这是编辑器内容区域的高度。通过这个高度,将行号部件的高度设置为与编辑器内容区域的高度相同。
    lineNumberWidget->setGeometry(0, 0, GetLineNumberWidgetWidth(), contentsRect().height());
}

void MyCodeEdit::lineNumberWidgetPaintEvent(QPaintEvent *event)
{
    QPainter painter(lineNumberWidget);
    // 绘制行号区域
    painter.fillRect(event->rect(), QColor(100, 100, 100, 100));
    // 行号显示
    // 拿到block
    QTextBlock block = firstVisibleBlock();
    // 拿到行号
    int blockNumber = block.blockNumber();
    // 拿到当前block的top
    int cursorTop = blockBoundingGeometry(textCursor().block()).translated(contentOffset()).top();
    // 拿到block的top
    int top = blockBoundingGeometry(block).translated(contentOffset()).top();
    // 拿到block的bottom
    int bottom = top + blockBoundingRect(block).height();
    while (block.isValid() && top <= event->rect().bottom()) {
        // 设置画笔颜色
        painter.setPen(cursorTop == top ? Qt::black : Qt::gray);
        painter.drawText(0, top, GetLineNumberWidgetWidth() - 5, blockBoundingRect(block).height(), Qt::AlignRight, QString::number(blockNumber + 1)); // 绘制文字
        // 拿到下一个block
        block = block.next();
        // 拿到一个新的top值
        top = bottom;
        bottom = top + blockBoundingRect(block).height();
        blockNumber++;
    }
}

void MyCodeEdit::lineNumberWidgetMousePessEvent(QMouseEvent *event)
{
    // 计算行号
    // 先拿到点击的block
    QTextBlock block = document()->findBlockByNumber(event->y() / fontMetrics().height() + verticalScrollBar()->value());
    // 设置光标的位置
    setTextCursor(QTextCursor(block));
}

// 把滚轮任务提交给MyCodeEditor
void MyCodeEdit::lineNumberWidgetWheelEvent(QWheelEvent *event)
{
    qDebug() << event->delta();
    if (event->orientation() == Qt::Horizontal) {
        horizontalScrollBar()->setValue(horizontalScrollBar()->value() - event->delta());
    } else {
        verticalScrollBar()->setValue(verticalScrollBar()->value() - event->delta());
    }
    event->accept();
}

 

#include "mainwindow.h"
#include "mycodeedit.h"
#include "mytextedit.h"
#include "mytexteditbycode.h"
#include "ui_mainwindow.h"
#include <QToolBar>
#include <QDebug>
#include <QFileDialog>
#include <QMessageBox>
#include <QFontDialog>
#include <QSettings> // 保存配置文件
#include <QTabWidget>

// 保存打开历史记录
void saveHistory(QString path);
// 获取历史记录
QList<QString> getHistory();

// 先判断是否支持打印


// 定义全局变量
QSettings *mSettings;

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
    , ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    // this->setCentralWidget(ui->textEdit);
    this->setCentralWidget(ui->tabWidget); // 添加标签页

    // 初始化保存历史记录的全局变量
    if (mSettings == NULL) {
        mSettings = new QSettings("settings.ini", QSettings::IniFormat);
    }
    initMenu();
}

MainWindow::~MainWindow()
{
    delete ui;
}


void MainWindow::initMenu()
{
    // 获取menu
    // 用于查找MainWindow中名为"recent"的子窗口,并将其存储为指向 `QMenu` 类型对象的指针 `recent`。
    QMenu *recent = this->findChild<QMenu *>("recent");
    // qDebug() << recent->title() << endl;
    qDebug() << "最近打开" << endl;
    // 获取Action
    QSet<QObject *> chList = recent->children().toSet();
    foreach(QObject *ch, chList) {
        QAction *action = (QAction *)ch;
        // 清空子菜单栏action
        recent->removeAction(action);
    }
    QList<QString> lists = getHistory();

    // 打开历史记录按时间从近到远
    for (int i = lists.size() - 1; i >= 0; --i) {
        // 生成子菜单
        recent->addAction(lists[i], this, &MainWindow::on_open_recent_file);
    }
    // 添加"清除历史记录"action
    if (lists.size() > 0) {
        recent->addAction("清楚历史菜单", this, &MainWindow::on_clear_history_triggered, QKeySequence("Ctrl+Alt+Shift+C"));
    }
}

// 获取历史记录
QList<QString> getHistory()
{
    // 打开开始读入
    int size = mSettings->beginReadArray("history");
    // 创建返回对象
    QList<QString> lists;
    for (int i = 0; i < size; i++) {
        mSettings->setArrayIndex(i);
        QString path = mSettings->value("path").toString();
        lists.append(path);
        qDebug() << i << ":" << path;
    }
    // 关闭开始读入
    mSettings->endArray();
    return lists;
}

// 保存打开历史记录
void saveHistory(QString path)
{
    // 获取历史
    QList<QString> lists = getHistory();
    lists.append(path);
    foreach(QString str, lists) {
        if (str == path) {
            lists.removeOne(str);
        }
    }
    lists.append(path);
    // lists.toSet().toList(); // 去重
    // 打开开始写入
    mSettings->beginWriteArray("history");
    for (int i = 0; i < lists.size(); ++i) {
        mSettings->setArrayIndex(i);
        // 保存字符串
        mSettings->setValue("path", lists[i]);
    }
    // 关闭开始写入
    mSettings->endArray();
}

// 新建文件
void MainWindow::on_new_file_triggered()
{
    // 调用自定义组件
    // 由于工程没有导入自定义组件的库,鼠标光标放在"MyTextEdit"上Alt+Enter可以导入
#if 0
    MyTextEdit *myTextEdit = new MyTextEdit(this);
    ui->tabWidget->addTab(myTextEdit, "NewTab.txt"); // 添加一个自定义的标签页(ui实现)
#else
    // QTextEdit类实现
    // ui->tabWidget->addTab(new MyTextEditByCode(this), "newTab.txt"); // 添加一个自定义的标签页 (纯代码实现)
    // QPlainTextEdit类实现
    ui->tabWidget->addTab(new MyCodeEdit(this), "NewTab.txt");

#endif
    return;
//    qDebug() << "start create new file ..." << endl;
//    currentFile.clear(); // 如果之前有文件的话先进性清空
//    ui->textEdit->setText(""); // 清空文件内容
}

void MainWindow::on_open_recent_file()
{
    QAction *action = (QAction*)sender();
    QString filename = action->text();
    QFile file(filename);
    if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {
        QMessageBox::warning(this, "警告", "无法打开此文件"+file.errorString());
    }
    currentFile = filename;
    setWindowTitle(filename);
    QTextStream in(&file);
    QString text = in.readAll();
    ui->textEdit->setText(text);
    file.close();
    saveHistory(currentFile);
    initMenu();
}

// 打开文件
void MainWindow::on_open_file_triggered()
{
    QString filename = QFileDialog::getOpenFileName(this, "打开文件");
    QFile file(filename);
    if (!file.open(QIODevice::ReadOnly | QFile::Text)) {
        QMessageBox::warning(this, "警告", "无法打开此文件 : " + file.errorString());
        return;
    }
    currentFile = filename;
    setWindowTitle(filename);
    QTextStream in(&file);
    QString text = in.readAll();
    ui->textEdit->setText(text);
    file.close();
    saveHistory(currentFile);
    initMenu();
}

// 保存文件
void MainWindow::on_save_file_triggered()
{
    QString filename;
    if (currentFile.isEmpty()) {
        filename = QFileDialog::getSaveFileName(this, "保存文件");
        currentFile = filename;
    } else {
        filename = currentFile;
    }
    QFile file(filename);
    if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
        QMessageBox::warning(this, "警告", "无法保存文件:"+file.errorString());
        return;
    }
    setWindowTitle(filename);
    QTextStream out(&file);
    QString text = ui->textEdit->toPlainText();
    out << text;
    file.close();
    saveHistory(currentFile);
    initMenu();
}

// 另存为
void MainWindow::on_save_as_triggered()
{
    QString filename;
    filename = QFileDialog::getSaveFileName(this, "另存为");
    QFile file(filename);
    if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
        QMessageBox::warning(this, "警告", "无法保存文件:"+file.errorString());
        return;
    }
    setWindowTitle(filename);
    QTextStream out(&file);
    QString text = ui->textEdit->toPlainText();
    out << text;
    file.close();
    saveHistory(currentFile);
    initMenu();
}

// 复制
void MainWindow::on_copy_triggered()
{
    ui->textEdit->copy();
}

// 粘贴
void MainWindow::on_paste_triggered()
{
    ui->textEdit->paste();
}

// 剪切
void MainWindow::on_cut_triggered()
{
    ui->textEdit->cut();
}

// 字体
void MainWindow::on_font_triggered()
{
    bool fontSelected;
    QFont font = QFontDialog::getFont(&fontSelected, this);
    if (fontSelected) {
        ui->textEdit->setFont(font);
    }
}

// 撤销
void MainWindow::on_undo_triggered()
{
    ui->textEdit->undo();
}

// 取消撤销
void MainWindow::on_redo_triggered()
{
    ui->textEdit->redo();
}

// 退出
void MainWindow::on_exit_triggered()
{
    QCoreApplication::exit();
}

// 信息
void MainWindow::on_info_triggered()
{
    QMessageBox::about(this, "这是我的notepad", "欢迎学习和使用");
}

// 打印
void MainWindow::on_print_triggered()
{

}

// 加粗
void MainWindow::on_bolder_triggered(bool checked)
{
    qDebug() << "on_bolder_triggered" << endl;
    ui->textEdit->setFontWeight(checked);
}

// 斜体
void MainWindow::on_italics_triggered(bool checked)
{
    ui->textEdit->setFontItalic(checked);
}

// 下划线
void MainWindow::on_underline_triggered(bool checked)
{
    ui->textEdit->setFontUnderline(checked);
}

// 清除历史记录
void MainWindow::on_clear_history_triggered()
{
    qDebug() << "on_clear_history_triggered..." << endl;
    mSettings->remove("history");
    initMenu();
}

`QTextEdit::ExtraSelection` 是一个用于在 `QTextEdit` 中添加额外选择区域的类。可以使用它来实现高亮、定位光标等特效。

`ExtraSelection` 包含以下属性:

- selectionRange: 选择的区域,使用 `QTextEdit::ExtraSelection::Cursor` 和 `QTextEdit::ExtraSelection::FormatRange` 两种方式指定。

- cursor: 光标对象。如果设置了此属性,光标会自动显示在选择区域的末尾。

- format: 文本格式。

- layoutDirection: 文字流的方向。可以指定为 `Qt::LeftToRight` 或 `Qt::RightToLeft`。

- verticalAlignment: 垂直对齐方式。可以指定为 `QTextEdit::ExtraSelection::AlignTop`、

下面是一个使用 `ExtraSelection` 添加额外选择区域的示例:

QTextEdit::ExtraSelection selection;

QTextCursor cursor = textEdit->textCursor(); //获取文本光标

selection.cursor = cursor;

QColor lineColor = QColor(Qt::yellow).lighter(160); //设置高亮颜色

selection.format.setBackground(lineColor);

selection.format.setProperty(QTextFormat::FullWidthSelection, true);

selection.cursor.clearSelection(); //清除光标的选择区域

extraSelections.append(selection); //将选择区域添加到选择列表

textEdit->setExtraSelections(extraSelections); //设置额外选择区域

在示例代码中,我们首先获取 `QTextEdit` 的文本光标,并将其设置为选择区域的末尾。

然后置高亮颜色,并将其设置为选择区域的背景色。

最后将选择区域添加到额外选择列表中,并将其设置为 `QTextEdit` 的额外选择区域。

`explicit` 是在 C++ 中一个关键字,可以用于类定义中的单参数构造函数前面,指示编译器禁止该构造函数进行隐式类型转换。显式类型转换是允许的。

`explicit` 关键字的作用是防止在使用单参数构造函数时,不小心进行了隐式类型转换。

如果没有这个关键字,编译器可能会在一些情况下自动地进行类型转换,导致问题。例如:

```cpp

class MyClass {

public:

    MyClass(int x) : value(x) {}

    int value;

};

void func(MyClass m) {

    std::cout << m.value;

}

int main() {

    func(42); // 自动地将整数 42 转换成 MyClass 对象,导致难以察觉的错误。

    return 0;

}

```

在上面的代码中,`MyClass` 的单参数构造函数可以接收一个整数作为参数。

在 `func` 函数中,`MyClass` 对象被传递进来,并调用了 `value` 属性的值来进行输出。

但是,在 `main` 函数中的调用语句中传递整数值(而不是 `MyClass` 对象)时,

编译器会自动地将整数转换成 `MyClass` 对象,从而导致难以察觉的错误。

可以使用 `explicit` 关键字来防止这种情况的发生。

如果单参数构造函数使用 `explicit` 关键字进行修饰,则不会进行自动的类型转换。

因此,上面的代码可以修改为:

class MyClass

 {

public:

    explicit MyClass(int x) : value(x) {}

    int value;

};

void func(MyClass m) {

    std::cout << m.value;

}

int main() {

    func(MyClass(42)); // 必须显式创建 MyClass 对象,无法自动转换整数值

    return 0;

}

在这个修改后的代码中,`MyClass` 的单参数构造函数使用了 `explicit` 关键字。

因此,在 `func` 函数中必须使用显示创建的 `MyClass` 对象传递,而不能使用整数值进行隐式转换。

总之,`explicit` 关键字可以防止一些常见的类型转换错误,因此在一些情况下可以提高代码的安全性和可读性。

`QPlainTextEdit::resizeEvent(event)` 是一个父类 `QPlainTextEdit` 中的函数,它会在编辑器的大小改变时自动触发。

这个函数默认会处理编辑器大小改变时的一些操作,如视图中的滚动条的位置和显示文本的区域等。

因此,在自定义的 `resizeEvent` 函数中需要先调用父类的 `resizeEvent` 函数来确保继承自 `QPlainTextEdit` 的默认操作得到正确处理。

这段代码的作用是处理编辑器的 `resizeEvent` 事件,它在编辑器大小改变时调用,可以用于重新计算并调整相关控件的位置和大小。

void MyCodeEdit::resizeEvent(QResizeEvent *event)

{

    QPlainTextEdit::resizeEvent(event);

    lineNumberWidget->setGeometry(0, 0, GetLineNumberWidgetWidth(), contentsRect().height());

}

在此段代码中,首先调用了父类的 `resizeEvent` 函数来确保继承自 `QPlainTextEdit` 的默认操作得到正确处理。

然后再重新计算行号部件 `lineNumberWidget` 的位置和大小,并设置为编辑器中的左上角,使其高度与编辑器内容区域的高度相同,并将其宽度设置为 `GetLineNumberWidgetWidth()` 函数返回的值。

总之,`QPlainTextEdit::resizeEvent` 函数是一个重要的、自动触发的编辑器事件,需要在自定义的 `resizeEvent` 函数中先调用父类函数,

以确保继承自 `QPlainTextEdit` 的默认操作得到正确处理,再结合具体的业务逻辑进行控件位置和大小的调整。

1. 为什么重写 `resizeEvent` 方法需要与 `QPlainTextEdit::resizeEvent(event)` 方法名保持一致?

在 C++ 中,函数的名字不只是标识函数的方式,还关系到函数的调用。比如,当一个对象被触发 `resizeEvent` 事件时,该对象所绑定的 `resizeEvent` 函数会被自动调用。如果重写的函数名与父类的函数名不一致,那么重写的函数将不会被调用并无法处理 `resizeEvent` 事件。因此,为了重写事件处理函数,函数名必须与父类中的函数名一致。

在这段代码中,`MyCodeEdit` 继承自 `QPlainTextEdit`,因此需要调用父类中的 `resizeEvent` 函数,让编辑器默认

的重设窗口大小的操作正常进行。

2. 为什么在自定义组件类中没有调用 `MyCodeEdit::resizeEvent`,也会被触发?

这是因为 `QPlainTextEdit` 类在其库文件中已经实现了 `resizeEvent` 函数的实现。

在 `MyCodeEdit` 类中继承自 `QPlainTextEdit`,因此自动继承了这个实现。

如果在 `MyCodeEdit` 中没有定义 `resizeEvent` 函数,那么默认会调用 `QPlainTextEdit` 类中的这个函数,并执行其中的代码。

当自定义 `resizeEvent` 函数时,需要“重载”父类的默认实现以使得自己的代码能够得到正确处理,

可以通过调用父类的函数来做到这点,这使得自定义 `resizeEvent` 函数不需要从头开始实现事件处理逻辑,而只需要在父类的基础上进行微调或调整即可。

1.报错提示:QObject::connect: No such signal MyCodeEdit::cursorPositionCHanged() in ..\codeEdit\mycodeedit.cpp:18

HighLightCurrentLine 

// 信号槽绑定

void MyCodeEdit::InitConnection()

{

    // cursor

    connect(this, SIGNAL(cursorPositionCHanged()), this, SLOT(HighLightCurrentLine()));

}

报错原因分析与修改:

错误提示是因为 `cursorPositionCHanged()` 信号名称拼写错误,应该改为 `cursorPositionChanged()`。

// 修改如下

void MyCodeEdit::InitConnection()

{

    // cursor

    connect(this, SIGNAL(cursorPositionChanged()), this, SLOT(HighLightCurrentLine()));

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值