文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Qt如何实现多文本编辑器

2023-07-05 13:39

关注

本篇内容主要讲解“Qt如何实现多文本编辑器”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Qt如何实现多文本编辑器”吧!

首先先来看实验成果图,大概就是这么个多文档编辑器。

Qt如何实现多文本编辑器

首先需要在设计模式里进行设计器的设置:

Qt如何实现多文本编辑器

然后就是新建类MdiChild的声明和实现

mdichild.h

#ifndef MDICHILD_H#define MDICHILD_H #include<QTextEdit> class MdiChild:public QTextEdit{    Q_OBJECTpublic:    explicit MdiChild(QWidget *parent = 0);    void newFile(); //新建操作    bool loadFile(const QString &fileName); //加载文件    bool save(); //保存操作    bool saveAs(); //另存为操作    bool saveFile(const QString &fileName); //保存文件    QString userFriendlyCurrentFile(); //提取文件名    QString currentFile(){return curFile;} //返回当前文件路径 protected:    void closeEvent(QCloseEvent *event); //关闭事件    void contextMenuEvent(QContextMenuEvent *e); //右键菜单事件 private slots:    void documentWasModified(); //文档被更改时,窗口显示更改状态标志 private:    bool maybeSave(); //是否需要保存    void setCurrentFile(const QString &fileName); //设置当前文件    QString curFile;  //保存当前文件路径    bool isUntitled;  //作为当前文件是否被保存到硬盘上的标志 }; #endif // MDICHILD_H

mdichild.cpp

#include "mdichild.h"#include<QFile>#include<QMessageBox>#include<QTextStream>#include<QApplication>#include<QFileInfo>#include<QFileDialog>#include<QCloseEvent>#include<QPushButton>#include<QMenu> MdiChild::MdiChild(QWidget *parent):QTextEdit (parent){    //设置在子窗口关闭时销毁这个类的对象    setAttribute(Qt::WA_DeleteOnClose);     //初始isUntitled为true    isUntitled = true;} void MdiChild::newFile(){    //设置窗口编号,因为编号一致被保存,所以需要使用静态变量    static int sequenceNumber =1;     //新建的文档没有被保存过    isUntitled =true;     //将当前文件命名为未命名文档加编号,编号先使用再加1    curFile = tr("未命名文档%1.txt").arg(sequenceNumber++);     //设置窗口,使用[*]可以再文档被更改后再文件名称后显示“*”号    setWindowTitle(curFile +"[*]"+tr(" - 多文档编辑器"));     //文档更改时发射contentsChanged()信号,执行documentWasModified()槽    connect(document(),SIGNAL(contentsChanged()),this,SLOT(documentWasModified()));  } bool MdiChild::loadFile(const QString &fileName){    //新建QFile对象    QFile file(fileName);    //只读方式打开文件,出错则提示,并返回false    if(!file.open(QFile::ReadOnly|QFile::Text))    {        QMessageBox::warning(this,tr("多文档编辑器"),tr("无法读取文件%1:\n%2.").arg(fileName).arg(file.errorString()));        return false;    }     //新建文本流对象    QTextStream in(&file);    //设置鼠标状态为等待状态    QApplication::setOverrideCursor(Qt::WaitCursor);    //读取文件的全部文本内容,并添加到编辑器中    setPlainText(in.readAll());    //恢复鼠标状态    QApplication::restoreOverrideCursor();    //设置当前文件    setCurrentFile(fileName);    connect(document(),SIGNAL(contentsChanged()),this,SLOT(documentWasChanged()));    return true; } bool MdiChild::save(){    //如果文件未被保存过,则执行另存为操作,否则直接保存文件    if(isUntitled)    {        return saveAs();    }    else {        return saveFile(curFile);    }} bool MdiChild::saveAs(){   QString fileName = QFileDialog::getSaveFileName(this,tr("另存为"),curFile);   //获取文件路径,如果为空,则返回false,否则保存文件   if(fileName.isEmpty())       return false;   return saveFile(fileName);} bool MdiChild::saveFile(const QString &fileName){    QFile file(fileName);    if(!file.open(QFile::WriteOnly|QFile::Text))    {        QMessageBox::warning(this,tr("多文档编辑器"),tr("无法写入文件%1:\n%2").arg(fileName).arg(file.errorString()));        return false;    }    QTextStream out(&file);    QApplication::setOverrideCursor(Qt::WaitCursor);    //以纯文本文件写入    out<<toPlainText();    QApplication::restoreOverrideCursor();    setCurrentFile(fileName);} QString MdiChild::userFriendlyCurrentFile(){    //从文件路径中提取文件名    return QFileInfo(curFile).fileName();} void MdiChild::closeEvent(QCloseEvent *event){    //如果maybeSave()函数返回true,则关闭窗口,否则忽略该事件    if(maybeSave())    {        event->accept();    }    else {        event->ignore();    }} void MdiChild::contextMenuEvent(QContextMenuEvent *e){    //创建菜单,并向其中添加动作    QMenu *menu = new QMenu;    QAction *undo =menu->addAction(tr("撤销(&U)"),this,SLOT(undo()),QKeySequence::Undo);    undo->setEnabled(document()->isUndoAvailable());    QAction *redo =menu->addAction(tr("恢复(&R)"),this,SLOT(redo()),QKeySequence::Redo);    redo->setEnabled((document()->isRedoAvailable()));    menu->addSeparator();    QAction *cut =menu->addAction(tr("剪切(&T)"),this,SLOT(cut()),QKeySequence::Cut);    cut->setEnabled(textCursor().hasSelection());    QAction *copy = menu->addAction(tr("复制(&C)"),this,SLOT(copy()),QKeySequence::Copy);    copy->setEnabled(textCursor().hasSelection());    QAction *clear = menu->addAction(tr("清空"),this,SLOT(clear()));    clear->setEnabled(!document()->isEmpty());    menu->addSeparator();    QAction *select = menu->addAction(tr("全选"),this,SLOT(selectAll()),QKeySequence::SelectAll);    select->setEnabled(!document()->isEmpty());     //获取鼠标的位置,然后在这个位置显示菜单    menu->exec(e->globalPos());     //最后销毁这个菜单    delete menu;} void MdiChild::documentWasModified(){    //根据文档的isModified()函数的返回值,判断编辑器内容是否被更改了    //如果被更改了,就要在设置[*]号的地方显示“*”号,这里会在窗口中显示    setWindowModified(document()->isModified());} bool MdiChild::maybeSave(){    //如果文档被更改过    if(document()->isModified())    {        QMessageBox box;        box.setWindowTitle(tr("多文档编辑器"));        box.setText(tr("是否保存为“%1”的更改?").arg(userFriendlyCurrentFile()));        box.setIcon(QMessageBox::Warning);        //添加按钮,QMessageBox::YesRole可以表明这个按钮的行为        QPushButton *yesBtn = box.addButton(tr("是(&Y)"),QMessageBox::YesRole);        box.addButton(tr("否(&N)"),QMessageBox::NoRole);        QPushButton *cancelBtn = box.addButton(tr("取消"),QMessageBox::RejectRole);         //弹出对话框,让用户选择是否保存修改,或者取消关闭操作        box.exec();         //如果用户选择是,则返回保存操作的结果;如果选择取消,则返回false        if(box.clickedButton() ==yesBtn)            return save();        else if(box.clickedButton() ==cancelBtn){            return false;        }      }    //如果文档没有更改过,则直接返回true;    return true;} void MdiChild::setCurrentFile(const QString &fileName){    //canonicalFilePath()可以除去路径中的符号链接,"."和".."等符号    curFile = QFileInfo(fileName).canonicalFilePath();    //文件已经被保存过了    isUntitled = false;     //文档没有被更改过    document()->setModified(false);     //窗口不显示被更改标志    setWindowModified(false);     //设置窗口,userFriendlyCurrentFile()返回文件名    setWindowTitle(userFriendlyCurrentFile()+"[*]"); }

然后是mainwindow类的声明与实现

mainwindow.h

#ifndef MAINWINDOW_H#define MAINWINDOW_H #include <QMainWindow>class QMdiSubWindow;class MdiChild;class QSignalMapper; namespace Ui {class MainWindow;} class MainWindow : public QMainWindow{    Q_OBJECT public:    explicit MainWindow(QWidget *parent = nullptr);    ~MainWindow(); private slots:     void updateMenus(); //更新菜单    MdiChild* createMdiChild(); //创建子窗口    void setActiveSubWindow(QWidget *window); //设置活动子窗口    void updateWindowMenu(); //更新窗口菜单    void showTextRowAndCol(); //显示文本的行号和列号     void on_actionNew_triggered();     void on_actionOpen_triggered();     void on_actionSave_triggered();     void on_actionSaveAs_triggered();     void on_actionUndo_triggered();     void on_actionRedo_triggered();     void on_actionCut_triggered();     void on_actionCopy_triggered();     void on_actionPaste_triggered();     void on_actionClose_triggered();     void on_actionCloseAll_triggered();     void on_actionAbout_triggered();     void on_actionAboutQt_triggered();     void on_actionExit_triggered(); private:    QMdiSubWindow *findMdiChild(const QString &fileName);//查找子窗口    void readSettings(); //读取窗口设置    void writeSettings(); //写入窗口设置    void initWindow(); //初始化窗口  protected:    void closeEvent(QCloseEvent* event); //关闭事件  private:    Ui::MainWindow *ui;    QAction *actionSeparator; //间隔器    MdiChild *activeMdiChild(); //活动窗口    QSignalMapper *windowMapper; //信号映射器}; #endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"#include "ui_mainwindow.h"#include"mdichild.h"#include<QFileDialog>#include<QMdiSubWindow>#include<QSignalMapper>#include<QMessageBox>#include<QSettings>#include<QCloseEvent>#include<QLabel> MainWindow::MainWindow(QWidget *parent) :    QMainWindow(parent),    ui(new Ui::MainWindow){    ui->setupUi(this);         //创建间隔期动作并在其中设置间隔期    actionSeparator = new QAction(this);    actionSeparator->setSeparator(true);    //更新菜单    updateMenus();     //当有活动窗口时更新菜单    connect(ui->mdiArea,SIGNAL(subWindowActivated(QMdiSubWindow*)),this,SLOT(updateMenus()));     //创建信号映射器    windowMapper = new QSignalMapper(this);    //映射器重新发射信号,更具信号设置活动窗口    connect(windowMapper,SIGNAL(mapped(QWidget *)),this,SLOT(setActiveSubWindow(QWidget *)));     //更新窗口菜单,并且设置当窗口菜单将要显示的时候更新窗口菜单    updateWindowMenu();    connect(ui->menu_W,SIGNAL(aboutToShow()),this,SLOT(updateWindowMenu()));     readSettings(); //初始窗口时读取窗口设置信息     //初始化窗口    initWindow();} MainWindow::~MainWindow(){    delete ui;} void MainWindow::on_actionNew_triggered(){    //创建MdiChild    MdiChild *child = createMdiChild();    //新建文件    child->newFile();    //显示子窗口    child->show();} void MainWindow::updateMenus(){    //根据是否有活动窗口来设置各个动作是否可用    bool hasMdiChild = (activeMdiChild()!= 0);    ui->actionSave->setEnabled(hasMdiChild);    ui->actionSaveAs->setEnabled(hasMdiChild);    ui->actionPaste->setEnabled(hasMdiChild);    ui->actionClose->setEnabled(hasMdiChild);    ui->actionCloseAll->setEnabled(hasMdiChild);    ui->actionTile->setEnabled(hasMdiChild);    ui->actionCascade->setEnabled(hasMdiChild);    ui->actionNext->setEnabled(hasMdiChild);    ui->actionPrevious->setEnabled(hasMdiChild);     //有活动窗口且有被选择的文本,剪切复制才可用    bool hasSelection =(activeMdiChild() &&activeMdiChild()->textCursor().hasSelection());    ui->actionCut->setEnabled(hasSelection);    ui->actionPaste->setEnabled(hasSelection);     //有活动窗口且有撤销操作时撤销动作可用    ui->actionUndo->setEnabled(activeMdiChild()&&activeMdiChild()->document()->isUndoAvailable());     //有活动窗口且文档有恢复操作时恢复动作可用    ui->actionRedo->setEnabled(activeMdiChild()&&activeMdiChild()->document()->isRedoAvailable());} MdiChild *MainWindow::createMdiChild(){    //创建MdiChild部件    MdiChild *child = new MdiChild;    //向多文档区域添加子窗口,child为中心部件    ui->mdiArea->addSubWindow(child);     //根据QTextEdit类的是否可以复制信号设置剪切复制动作是否可用    connect(child,SIGNAL(copyAvailable(bool)),ui->actionCut,SLOT(setEnabled(bool)));    connect(child,SIGNAL(copyAvailable(bool)),ui->actionCopy,SLOT(setEnabled(bool)));    //根据QTextDocument类的是否可以撤销恢复信号设置撤销恢复动作是否可用    connect(child->document(),SIGNAL(undoAvailable(bool)),ui->actionUndo,SLOT(setEnabled(bool)));    connect(child->document(),SIGNAL(redoAvailable(bool)),ui->actionRedo,SLOT(setEnabled(bool)));     connect(child,SIGNAL(cursorPositionChanged()),this,SLOT(showTextRowAndCol()));     return child;} void MainWindow::setActiveSubWindow(QWidget *window){    //如果传递了窗口部件,则将其设置为活动窗口    if(!window)        return;    ui->mdiArea->setActiveSubWindow(qobject_cast<QMdiSubWindow*>(window));} void MainWindow::on_actionOpen_triggered(){    //获取文件路劲    QString fileName = QFileDialog::getOpenFileName(this);    //如果路径不为空,则查看该文件是否已经打开    if(!fileName.isEmpty())    {        QMdiSubWindow *existing = findMdiChild(fileName);        //如果已经存在,则将对应的子窗口设置为活动窗口        if(existing)        {            ui->mdiArea->setActiveSubWindow(existing);            return;        }        //如果没有打开,则新建子窗口        MdiChild *child = createMdiChild();        if(child->loadFile(fileName))        {            ui->statusBar->showMessage(tr("打开文件成功"),2000);            child->show();        }        else {            child->close();        }    }} void MainWindow::updateWindowMenu(){    //先清空菜单,然后在添加各个菜单动作    ui->menu_W->clear();    ui->menu_W->addAction(ui->actionClose);    ui->menu_W->addAction(ui->actionCloseAll);    ui->menu_W->addSeparator();    ui->menu_W->addAction(ui->actionTile);    ui->menu_W->addAction(ui->actionCascade);    ui->menu_W->addSeparator();    ui->menu_W->addAction(ui->actionNext);    ui->menu_W->addAction(ui->actionPrevious);    ui->menu_W->addAction(actionSeparator);     //如果有活动窗口,则显示间隔器    QList<QMdiSubWindow*> windows =ui->mdiArea->subWindowList();    actionSeparator->setVisible(!windows.isEmpty());     //遍历各个子窗口    for(int i=0;i<windows.size();++i)    {        MdiChild *child = qobject_cast<MdiChild*>(windows.at(i)->widget());        QString text;        //如果窗口数小于9,则设置编号为快捷键        if(i<9)        {            text=tr("&%1 %2").arg(i+1).arg(child->userFriendlyCurrentFile());        }        else {            text=tr("%1 %2").arg(i+1).arg(child->userFriendlyCurrentFile());        }         //添加动作到菜单,设置动作可以选择        QAction *action = ui->menu_W->addAction(text);        action->setCheckable(true);        //设置当前活动窗口动作为选中状态        action->setChecked(child ==activeMdiChild());        //关联动作的触发信号到信号映射器的map()槽,这个槽会发射mapped()信号        connect(action,SIGNAL(triggered()),windowMapper,SLOT(map()));         //将动作与相应的窗口部件进行映射        //在发射mappedd()信号时就会以这个窗口部件为参数        windowMapper->setMapping(action,windows.at(i));    }} void MainWindow::showTextRowAndCol(){    //如果有活动窗口,则显示其中光标所在的位置    if(activeMdiChild())    {        //因为获取的行号和列号都是从0开始的,所以我们这里进行了加1        int rowNum = activeMdiChild()->textCursor().blockNumber()+1;        int colNum = activeMdiChild()->textCursor().blockNumber()+1;         ui->statusBar->showMessage(tr("%1行 %2列").arg(rowNum).arg(colNum),2000);    }}  QMdiSubWindow *MainWindow::findMdiChild(const QString &fileName){    QString canonicalFilePath = QFileInfo(fileName).canonicalFilePath();    //利用foreach语句遍历子窗口列表,如果其文件路径和要查找的路径相同,则返回该窗口    foreach(QMdiSubWindow* window,ui->mdiArea->subWindowList())    {        MdiChild* mdiChild = qobject_cast<MdiChild*>(window->widget());        if(mdiChild->currentFile() ==canonicalFilePath)            return window;    }    return 0;} void MainWindow::readSettings(){    QSettings settings("yafeilinux","myMdi");    QPoint pos =settings.value("pos",QPoint(200,200)).toPoint();    QSize size =settings.value("size",QSize(400,200)).toSize();    move(pos);    resize(size);} void MainWindow::writeSettings(){    QSettings settings("yafeilinux","myMdi");    //写入位置信息和大小    settings.setValue("pos",pos());    settings.setValue("size",size()); } void MainWindow::initWindow(){    setWindowTitle(tr("多文档编辑器"));     //在工具栏右击时,可以关闭该工具栏    ui->mainToolBar->setWindowTitle(tr("工具栏"));    //当多文档区域的内容超出可是区域后,出现滚动条    ui->mdiArea->setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded);    ui->mdiArea->setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded);     ui->statusBar->showMessage(tr("欢迎使用多文档编辑器"));    QLabel *label =new QLabel(this);    label->setFrameStyle(QFrame::Box|QFrame::Sunken);    label->setText(tr("<a href=\"http://www.yafeilinux.com\">yafeilinux.com</a>"));     //标签文字为富文本    label->setTextFormat(Qt::RichText);     //可以打开外部链接    label->setOpenExternalLinks(true);    ui->statusBar->addPermanentWidget(label);    ui->actionNew->setStatusTip(tr("创建一个文件"));    ui->actionOpen->setStatusTip(tr("打开一个文件")); } void MainWindow::closeEvent(QCloseEvent *event){    //先执行多文档区域的关闭操作    ui->mdiArea->closeAllSubWindows();    //如果还有窗口没有关闭,则忽略该事件    if(ui->mdiArea->currentSubWindow())        event->ignore();    else {        //在关闭前写入窗口设置        writeSettings();        event->accept();    }}MdiChild *MainWindow::activeMdiChild(){    //如果有活动窗口,则将其内的中心部件转换为MdiChild类型,没有则直接返回0    if(QMdiSubWindow *activeSubWindow = ui->mdiArea->activeSubWindow())        return qobject_cast<MdiChild*>(activeSubWindow->widget());    return 0;} void MainWindow::on_actionSave_triggered(){    if(activeMdiChild()&&activeMdiChild()->save())        ui->statusBar->showMessage(tr("文件保存成功"),2000);} void MainWindow::on_actionSaveAs_triggered(){    if(activeMdiChild()&&activeMdiChild()->saveAs())        ui->statusBar->showMessage(tr("文件保存成功"),2000);} void MainWindow::on_actionUndo_triggered(){    if(activeMdiChild())        activeMdiChild()->undo();} void MainWindow::on_actionRedo_triggered(){    if(activeMdiChild())        activeMdiChild()->redo();} void MainWindow::on_actionCut_triggered(){    if(activeMdiChild())        activeMdiChild()->cut();} void MainWindow::on_actionCopy_triggered(){    if(activeMdiChild())        activeMdiChild()->copy();} void MainWindow::on_actionPaste_triggered(){    if(activeMdiChild())        activeMdiChild()->paste();} void MainWindow::on_actionClose_triggered(){    ui->mdiArea->closeActiveSubWindow();} void MainWindow::on_actionCloseAll_triggered(){    ui->mdiArea->closeAllSubWindows();} void MainWindow::on_actionAbout_triggered(){    QMessageBox::about(this,"关于",tr("致力于多文档编辑器普及工作"));} void MainWindow::on_actionAboutQt_triggered(){    QMessageBox::about(this,"关于Qt",tr("Qt是一个1991年由Qt Company开发的跨平台C++图形用户界面应用程序开发框架。它既可以开发GUI程序,也可用于开发非GUI程序,比如控制台工具和服务器。Qt是面向对象的框架,使用特殊的代码生成扩展(称为元对象编译器(Meta Object Compiler, moc))以及一些宏,Qt很容易扩展,并且允许真正地组件编程。"));} void MainWindow::on_actionExit_triggered(){    //等价于QApplication::closeAllWindows();    qApp->closeAllWindows();}

最后是main函数

#include "mainwindow.h"#include <QApplication>  int main(int argc, char *argv[]){    QApplication a(argc, argv);    MainWindow w;    w.show();     return a.exec();}

到此,相信大家对“Qt如何实现多文本编辑器”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯