目前在做一个ModBus-RTU的通信界面,在向下位机写多个数据时需要用到QTextEdit控件,发现不能像QLineEdit那样用正则表达式规定它的输入格式,故在槽函数里通过简单的逻辑自动编辑了一下,现提供方法步骤及源码,给大家提供参考,如有更好的方法实现欢迎讨论。转载请注明出处。
该例子实现以下功能:
一、序号不可编辑;
实现方法:
1、如序号为“001.”时,光标打到序号范围内时,在QTextEdit的槽函数cursorPositionChanged()里将光标移至序号之后;
2、若从右至左退格将“.”删除,在QTextEdit的槽函数textChanged()中将“.”再添加回去,并将光标打到上一行末尾;
3、在编辑文本时,按下Enter会新开一行,在textChanged()中吧换行符删掉并将光标打到下一行。经过这3步处理达到序号不可编辑的目的。
二、每行输入的数据字节长度不能超过2,且只能输入十六进制字符;
实现方法:
1、每输入一个字符,在在textChanged()中都会计算一次当前行文本字符串的长度,当超过限制长度将删掉光标前一个字符;
2、输入文本每变化一次,回去光标前的一个字符,判断该字符是否符合16进制格式,不符合删掉。
源码:
#include "texteditformat.h"
#include "ui_texteditformat.h"
#include<QTextEdit>
#include <QTextBlock>
#include <QTextCursor>
#define SERIAL_NUM_LEN 4
#define INPUT_MAX_LEN 2
TextEditFormat::TextEditFormat(QWidget *parent) :
QWidget(parent),
ui(new Ui::TextEditFormat)
{
ui->setupUi(this);
}
TextEditFormat::~TextEditFormat()
{
delete ui;
}
void TextEditFormat::on_lineEdit_editingFinished()
{
bool ok;
int Num=ui->lineEdit->text().toInt(&ok);
ui->textEdit->clear();
for(int i=0;i<Num;i++)
{
disconnect(ui->textEdit,SIGNAL(textChanged()),this,SLOT(on_textEdit_textChanged()));
ui->textEdit->append(QString("%1").arg(i+1,3,10,QLatin1Char('0'))+".");
connect(ui->textEdit,SIGNAL(textChanged()),this,SLOT(on_textEdit_textChanged()));
}
}
void TextEditFormat::on_textEdit_cursorPositionChanged()
{
QTextEdit *mTextEdit=ui->textEdit;
QTextCursor mTextCursor = mTextEdit->textCursor();
int CurLineNum=mTextCursor.blockNumber();//光标所在行号,序号从0开始
int CurLineNumLen=4; //序号和“.”所占用的字节数,序号从1开始
int CursorPosInLine=mTextCursor.positionInBlock(); //光标所在行的位置,从所在行左边开始
int TotalLine=mTextEdit->document()->lineCount();//总行数
//! 设置光标不能进入序号的位置
//! 如果光标进入序号的位置,给它重新设置位置移到序号右边
//! 根据选中位置算出需要移动几个格子
if(CursorPosInLine<=CurLineNumLen&&CurLineNum<TotalLine)
{
int CursorPosGlobal=mTextCursor.position(); //光标全局位置,从第1行左边开始
mTextCursor.setPosition(CursorPosGlobal+(CurLineNumLen-CursorPosInLine));
disconnect(mTextEdit,SIGNAL(cursorPositionChanged()),this,SLOT(on_textEdit_textChanged()));
QString CurLineText=mTextEdit->document()->findBlockByLineNumber(CurLineNum).text();
if(CurLineText.contains("."))
mTextEdit->setTextCursor(mTextCursor);
connect(mTextEdit,SIGNAL(cursorPositionChanged()),this,SLOT(on_textEdit_textChanged()));
}
}
void TextEditFormat::on_textEdit_textChanged()
{
QTextEdit *mTextEdit=ui->textEdit;
QTextCursor mTextCursor = mTextEdit->textCursor();
int CurLineNum=mTextEdit->textCursor().blockNumber(); //光标所在行号,序号从0开始
int mPosInBlock=mTextCursor.positionInBlock(); //光标所在行的位置
QString CurLineText=mTextEdit->document()->findBlockByLineNumber(CurLineNum).text(); //当前行文本
//! 按回车会新开一行,如果判断光标在所在行其实位置时判断为新开一行;
//! 删除光标前一个字符,即删除换行符,并把光标向下移动一行
if(mPosInBlock==0&&CurLineNum!=0)
{
mTextCursor.deletePreviousChar();
mTextEdit->setTextCursor(mTextCursor);
mTextCursor.movePosition(QTextCursor::Down);
mTextEdit->setTextCursor(mTextCursor);
return;
}
//! 判断“.”被删除
//! 如“.”在重新添加回去,先删除原来整行,整行删除后光标已经退到上一行,需先添加换行符,再将添加了“.”的文本添加回去
//! 因此达到了不能编辑序号的目的
if(!CurLineText.contains("."))
{
int CurLineNumLen=SERIAL_NUM_LEN-1;//序号所占用的字节数,序号从1开始,显示格式为“000.”
if(CurLineText=="")
return;
CurLineText.insert(CurLineNumLen,".");
mTextCursor.select(QTextCursor::BlockUnderCursor);
mTextCursor.removeSelectedText();//删除整行
disconnect(mTextEdit,SIGNAL(textChanged()),this,SLOT(on_textEdit_textChanged()));
if(CurLineNum!=0)
mTextEdit->insertPlainText("\n");
mTextEdit->insertPlainText(CurLineText);
connect(mTextEdit,SIGNAL(textChanged()),this,SLOT(on_textEdit_textChanged()));
if(mTextCursor.blockNumber()!=0)
{
mTextCursor.movePosition(QTextCursor::Up);
mTextEdit->setTextCursor(mTextCursor);
}
return;
}
//!输入长度不能超过2位
int CurTextLen=CurLineText.length();
if(CurTextLen>SERIAL_NUM_LEN+INPUT_MAX_LEN)//输入长度限制
{
mTextEdit->textCursor().deletePreviousChar();
mTextEdit->setTextCursor(mTextCursor);
return;
}
//!获取光标前面的一个字符,判断是否符合16进制的输入格式要求
char InputChar=0;
InputChar=CurLineText.at(mPosInBlock-1).toLatin1();
if(InputChar=='.'
||(InputChar>='0'&&InputChar<='9')
||(InputChar>='a'&&InputChar<='f')
||(InputChar>='A'&&InputChar<='F'))
{
return;
}
else
{
mTextEdit->textCursor().deletePreviousChar();
mTextEdit->setTextCursor(mTextEdit->textCursor());
}
}