1286 lines
36 KiB
C++
1286 lines
36 KiB
C++
/*
|
|
* Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "qconsole.h"
|
|
|
|
#include <QEvent>
|
|
#include <QInputMethodEvent>
|
|
#include <QKeyEvent>
|
|
#include <QPaintEvent>
|
|
#include <QPainter>
|
|
#include <QRect>
|
|
#include <QScrollBar>
|
|
#include <cmath>
|
|
#include <QDebug>
|
|
#include <QTimer>
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include "../utils.h"
|
|
|
|
QConsole::QConsole(QWidget *parent):
|
|
QAbstractScrollArea{parent},
|
|
mContents{this},
|
|
mContentImage{}
|
|
{
|
|
mMaxHistory = 500;
|
|
mHistoryIndex = -1;
|
|
mCommand = "";
|
|
mCurrentEditableLine = "";
|
|
mRowHeight = 0;
|
|
mTopRow = 1;
|
|
mRowsInWindow = 0;
|
|
mColumnsPerRow = 0;
|
|
mColumnWidth = 0;
|
|
mReadonly = false;
|
|
mSelectionBegin = {0,0};
|
|
mSelectionEnd = {0,0};
|
|
mCaretChar = 0;
|
|
mBackground = palette().color(QPalette::Base);
|
|
mForeground = palette().color(QPalette::Text);
|
|
mSelectionBackground = palette().color(QPalette::Highlight);
|
|
mSelectionForeground = palette().color(QPalette::HighlightedText);
|
|
mInactiveSelectionBackground = palette().color(QPalette::Inactive,QPalette::Highlight);
|
|
mInactiveSelectionForeground = palette().color(QPalette::Inactive,QPalette::HighlightedText);
|
|
mTabSize = 4;
|
|
mBlinkTimerId = 0;
|
|
mBlinkStatus = 0;
|
|
//enable input method
|
|
setAttribute(Qt::WA_InputMethodEnabled);
|
|
// setMouseTracking(false);
|
|
recalcCharExtent();
|
|
mScrollTimer = new QTimer(this);
|
|
mScrollTimer->setInterval(100);
|
|
connect(mScrollTimer,&QTimer::timeout,this, &QConsole::scrollTimerHandler);
|
|
connect(&mContents,&ConsoleLines::layoutFinished,this, &QConsole::contentsLayouted);
|
|
connect(&mContents,&ConsoleLines::rowsAdded,this, &QConsole::contentsRowsAdded);
|
|
connect(&mContents,&ConsoleLines::lastRowsChanged,this, &QConsole::contentsLastRowsChanged);
|
|
connect(&mContents,&ConsoleLines::lastRowsRemoved,this, &QConsole::contentsLastRowsRemoved);
|
|
connect(verticalScrollBar(),&QScrollBar::valueChanged,
|
|
this, &QConsole::doScrolled);
|
|
|
|
}
|
|
|
|
int QConsole::maxHistory() const
|
|
{
|
|
return mMaxHistory;
|
|
}
|
|
|
|
void QConsole::setMaxHistory(int historySize)
|
|
{
|
|
mMaxHistory = historySize;
|
|
}
|
|
|
|
int QConsole::tabSize() const
|
|
{
|
|
return mTabSize;
|
|
}
|
|
|
|
int QConsole::columnsPerRow() const
|
|
{
|
|
return mColumnsPerRow;
|
|
}
|
|
|
|
int QConsole::rowsInWindow() const
|
|
{
|
|
return mRowsInWindow;
|
|
}
|
|
|
|
int QConsole::charColumns(QChar ch, int columnsBefore) const
|
|
{
|
|
if (ch == '\t') {
|
|
return mTabSize - (columnsBefore % mTabSize);
|
|
}
|
|
if (ch == ' ')
|
|
return 1;
|
|
return std::ceil((int)(fontMetrics().horizontalAdvance(ch)) / (double) mColumnWidth);
|
|
}
|
|
|
|
void QConsole::invalidate()
|
|
{
|
|
viewport()->update();
|
|
}
|
|
|
|
void QConsole::invalidateRows(int startRow, int endRow)
|
|
{
|
|
if (!isVisible())
|
|
return;
|
|
if (startRow == -1 && endRow == -1) {
|
|
invalidate();
|
|
} else {
|
|
startRow = std::max(startRow, 1);
|
|
endRow = std::max(endRow, 1);
|
|
// find the visible lines first
|
|
if (startRow > endRow)
|
|
std::swap(startRow, endRow);
|
|
|
|
if (endRow >= mContents.rows())
|
|
endRow = INT_MAX; // paint empty space beyond last line
|
|
|
|
// mTopLine is in display coordinates, so FirstLine and LastLine must be
|
|
// converted previously.
|
|
startRow = std::max(startRow, mTopRow);
|
|
endRow = std::min(endRow, mTopRow + mRowsInWindow-1);
|
|
|
|
// any line visible?
|
|
if (endRow >= startRow) {
|
|
QRect rcInval = {
|
|
0,
|
|
mRowHeight * (startRow - mTopRow),
|
|
clientWidth(), mRowHeight * (endRow - mTopRow + 1)
|
|
};
|
|
invalidateRect(rcInval);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
void QConsole::invalidateRect(const QRect &rect)
|
|
{
|
|
viewport()->update(rect);
|
|
}
|
|
|
|
void QConsole::addLine(const QString &line)
|
|
{
|
|
mCurrentEditableLine = "";
|
|
mCaretChar=0;
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
mContents.addLine(line);
|
|
}
|
|
|
|
void QConsole::addText(const QString &text)
|
|
{
|
|
QStringList lst = textToLines(text);
|
|
for (const QString& line:lst) {
|
|
addLine(line);
|
|
}
|
|
}
|
|
|
|
void QConsole::removeLastLine()
|
|
{
|
|
mCurrentEditableLine = "";
|
|
mCaretChar=0;
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
mContents.RemoveLastLine();
|
|
}
|
|
|
|
void QConsole::changeLastLine(const QString &line)
|
|
{
|
|
mContents.changeLastLine(line);
|
|
}
|
|
|
|
QString QConsole::getLastLine()
|
|
{
|
|
return mContents.getLastLine();
|
|
}
|
|
|
|
void QConsole::clear()
|
|
{
|
|
mContents.clear();
|
|
mCommand = "";
|
|
mCurrentEditableLine = "";
|
|
mTopRow = 1;
|
|
mSelectionBegin = {0,0};
|
|
mSelectionEnd = {0,0};
|
|
mCaretChar = 0;
|
|
updateScrollbars();
|
|
}
|
|
|
|
void QConsole::copy()
|
|
{
|
|
if (!this->hasSelection())
|
|
return;
|
|
QString s = selText();
|
|
QClipboard* clipboard=QGuiApplication::clipboard();
|
|
clipboard->clear();
|
|
clipboard->setText(s);
|
|
}
|
|
|
|
void QConsole::paste()
|
|
{
|
|
if (mReadonly)
|
|
return;
|
|
QClipboard* clipboard=QGuiApplication::clipboard();
|
|
textInputed(clipboard->text());
|
|
}
|
|
|
|
void QConsole::selectAll()
|
|
{
|
|
if (mContents.lines()>0) {
|
|
mSelectionBegin = {1,1};
|
|
mSelectionEnd = { mContents.getLastLine().length()+1,mContents.lines()};
|
|
}
|
|
}
|
|
|
|
QString QConsole::selText()
|
|
{
|
|
if (!hasSelection())
|
|
return "";
|
|
int ColFrom = selectionBegin().ch;
|
|
int First = selectionBegin().line;
|
|
int ColTo = selectionEnd().ch;
|
|
int Last = selectionEnd().line;
|
|
if (First == Last) {
|
|
QString s = mContents.getLine(First);
|
|
if (First == mContents.lines()) {
|
|
s += this->mCommand;
|
|
}
|
|
return s.mid(ColFrom, ColTo - ColFrom);
|
|
|
|
} else {
|
|
QString result = mContents.getLine(First).mid(ColFrom);
|
|
result+= lineBreak();
|
|
for (int i = First + 1; i<=Last - 1; i++) {
|
|
result += mContents.getLine(i);
|
|
result+= lineBreak();
|
|
}
|
|
QString s = mContents.getLine(Last);
|
|
if (Last == mContents.lines())
|
|
s+= this->mCommand;
|
|
result += s.leftRef(ColTo);
|
|
return result;
|
|
}
|
|
}
|
|
|
|
void QConsole::recalcCharExtent() {
|
|
mRowHeight = fontMetrics().lineSpacing();
|
|
mColumnWidth = fontMetrics().horizontalAdvance("M");
|
|
}
|
|
|
|
void QConsole::sizeOrFontChanged(bool)
|
|
{
|
|
if (mColumnWidth != 0) {
|
|
mColumnsPerRow = std::max(clientWidth()-2,0) / mColumnWidth;
|
|
mRowsInWindow = clientHeight() / mRowHeight;
|
|
mContents.layout();
|
|
}
|
|
|
|
}
|
|
|
|
int QConsole::clientWidth()
|
|
{
|
|
return viewport()->size().width();
|
|
}
|
|
|
|
int QConsole::clientHeight()
|
|
{
|
|
return viewport()->size().height();
|
|
}
|
|
|
|
void QConsole::setTopRow(int value)
|
|
{
|
|
value = std::min(value,maxScrollHeight());
|
|
value = std::max(value, 1);
|
|
if (value != mTopRow) {
|
|
verticalScrollBar()->setValue(value);
|
|
}
|
|
}
|
|
|
|
int QConsole::maxScrollHeight()
|
|
{
|
|
return std::max(mContents.rows()-mRowsInWindow+1,1);
|
|
}
|
|
|
|
void QConsole::updateScrollbars()
|
|
{
|
|
setHorizontalScrollBarPolicy(Qt::ScrollBarPolicy::ScrollBarAlwaysOff);
|
|
setVerticalScrollBarPolicy(Qt::ScrollBarPolicy::ScrollBarAsNeeded);
|
|
int nMaxScroll = maxScrollHeight();
|
|
int nMin = 1;
|
|
int nMax = std::max(1, nMaxScroll);
|
|
int nPage = mRowsInWindow;
|
|
int nPos = std::min(std::max(mTopRow,nMin),nMax);
|
|
verticalScrollBar()->setMinimum(nMin);
|
|
verticalScrollBar()->setMaximum(nMax);
|
|
verticalScrollBar()->setPageStep(nPage);
|
|
verticalScrollBar()->setSingleStep(1);
|
|
if (nPos != verticalScrollBar()->value())
|
|
verticalScrollBar()->setValue(nPos);
|
|
else
|
|
invalidate();
|
|
}
|
|
|
|
void QConsole::paintRows(QPainter &painter, int row1, int row2)
|
|
{
|
|
if (row1>row2)
|
|
return;
|
|
QRect rect(0,(row1-mTopRow)*mRowHeight,clientWidth(),(row2-row1+1)*mRowHeight);
|
|
painter.fillRect(rect,mBackground);
|
|
QStringList lst = mContents.getRows(row1,row2);
|
|
int startRow = row1-mTopRow;
|
|
painter.setPen(mForeground);
|
|
RowColumn selBeginRC = mContents.lineCharToRowColumn(selectionBegin());
|
|
RowColumn selEndRC = mContents.lineCharToRowColumn(selectionEnd());
|
|
LineChar editBegin = {
|
|
mContents.getLastLine().length() - mCurrentEditableLine.length(),
|
|
mContents.lines()-1
|
|
};
|
|
RowColumn editBeginRC = mContents.lineCharToRowColumn(editBegin);
|
|
bool isSelection = false;
|
|
painter.setPen(mForeground);
|
|
for (int i=0; i< lst.size(); i++) {
|
|
int currentRow = i+row1-1;
|
|
int left=2;
|
|
int top = (startRow+i) * mRowHeight;
|
|
int baseLine = (startRow+i+1)*mRowHeight - painter.fontMetrics().descent();
|
|
QString s = lst[i];
|
|
int columnsBefore = 0;
|
|
for (QChar ch:s) {
|
|
int charCol = charColumns(ch,columnsBefore);
|
|
int width = charCol * mColumnWidth;
|
|
if ((currentRow > selBeginRC.row ||
|
|
(currentRow == selBeginRC.row && columnsBefore>=selBeginRC.column))
|
|
&&
|
|
(currentRow < selEndRC.row ||
|
|
(currentRow == selEndRC.row && columnsBefore+charCol<=selEndRC.column))) {
|
|
if (!isSelection) {
|
|
isSelection = true;
|
|
}
|
|
if (!mReadonly &&(currentRow>editBeginRC.row ||
|
|
columnsBefore >= editBeginRC.column)) {
|
|
painter.setPen(mSelectionForeground);
|
|
painter.fillRect(left,top,width,mRowHeight,mSelectionBackground);
|
|
} else {
|
|
painter.setPen(mInactiveSelectionForeground);
|
|
painter.fillRect(left,top,width,mRowHeight,mInactiveSelectionBackground);
|
|
}
|
|
} else {
|
|
if (isSelection) {
|
|
isSelection = false;
|
|
painter.setPen(mForeground);
|
|
}
|
|
}
|
|
painter.drawText(left,baseLine,ch);
|
|
left+= width;
|
|
columnsBefore += charCol;
|
|
}
|
|
}
|
|
}
|
|
|
|
void QConsole::ensureCaretVisible()
|
|
{
|
|
int caretRow = mContents.rows();
|
|
if (caretRow < mTopRow) {
|
|
mTopRow = caretRow;
|
|
return;
|
|
}
|
|
if (caretRow >= mTopRow + mRowsInWindow) {
|
|
mTopRow = caretRow + 1 -mRowsInWindow;
|
|
}
|
|
}
|
|
|
|
void QConsole::showCaret()
|
|
{
|
|
if (mBlinkTimerId==0)
|
|
mBlinkTimerId = startTimer(500);
|
|
}
|
|
|
|
void QConsole::hideCaret()
|
|
{
|
|
if (mBlinkTimerId!=0) {
|
|
killTimer(mBlinkTimerId);
|
|
mBlinkTimerId = 0;
|
|
mBlinkStatus = 0;
|
|
updateCaret();
|
|
}
|
|
}
|
|
|
|
void QConsole::updateCaret()
|
|
{
|
|
QRect rcCaret = getCaretRect();
|
|
invalidateRect(rcCaret);
|
|
}
|
|
|
|
LineChar QConsole::caretPos()
|
|
{
|
|
QString lastLine = mContents.getLastLine();
|
|
int line = std::max(mContents.lines()-1,0);
|
|
int charIndex = 0;
|
|
if (mCaretChar>=mCurrentEditableLine.length()) {
|
|
charIndex = lastLine.length();
|
|
} else {
|
|
charIndex = lastLine.length()-mCurrentEditableLine.length()+mCaretChar;
|
|
}
|
|
return {charIndex,line};
|
|
}
|
|
|
|
RowColumn QConsole::caretRowColumn()
|
|
{
|
|
return mContents.lineCharToRowColumn(caretPos());
|
|
}
|
|
|
|
QPoint QConsole::rowColumnToPixels(const RowColumn &rowColumn)
|
|
{
|
|
/*
|
|
mTopRow is 1-based; rowColumn.row is 0-based
|
|
*/
|
|
int row =rowColumn.row+1 - mTopRow;
|
|
int col =rowColumn.column;
|
|
return QPoint(2+col*mColumnWidth, row*mRowHeight);
|
|
}
|
|
|
|
QRect QConsole::getCaretRect()
|
|
{
|
|
LineChar caret = caretPos();
|
|
QChar caretChar = mContents.getChar(caret);
|
|
RowColumn caretRC = mContents.lineCharToRowColumn(caret);
|
|
QPoint caretPos = rowColumnToPixels(caretRC);
|
|
int caretWidth=mColumnWidth;
|
|
//qDebug()<<"caret"<<mCaretX<<mCaretY;
|
|
int columnsBefore = caretRC.column;
|
|
if (!caretChar.isNull()) {
|
|
caretWidth = charColumns(caretChar, columnsBefore)*mColumnWidth;
|
|
}
|
|
return QRect(caretPos.x(),caretPos.y(),caretWidth,
|
|
mRowHeight);
|
|
}
|
|
|
|
void QConsole::doScrolled()
|
|
{
|
|
mTopRow = verticalScrollBar()->value();
|
|
invalidate();
|
|
}
|
|
|
|
void QConsole::contentsLayouted()
|
|
{
|
|
updateScrollbars();
|
|
}
|
|
|
|
void QConsole::contentsRowsAdded(int )
|
|
{
|
|
ensureCaretVisible();
|
|
updateScrollbars();
|
|
}
|
|
|
|
void QConsole::contentsLastRowsRemoved(int )
|
|
{
|
|
ensureCaretVisible();
|
|
updateScrollbars();
|
|
}
|
|
|
|
void QConsole::contentsLastRowsChanged(int rowCount)
|
|
{
|
|
ensureCaretVisible();
|
|
invalidateRows(mContents.rows()-rowCount+1,mContents.rows());
|
|
}
|
|
|
|
void QConsole::scrollTimerHandler()
|
|
|
|
{
|
|
QPoint iMousePos = QCursor::pos();
|
|
iMousePos = mapFromGlobal(iMousePos);
|
|
RowColumn mousePosRC = pixelsToNearestRowColumn(iMousePos.x(),iMousePos.y());
|
|
|
|
if (mScrollDeltaY != 0) {
|
|
if (QApplication::queryKeyboardModifiers().testFlag(Qt::ShiftModifier))
|
|
setTopRow(mTopRow + mScrollDeltaY * mRowsInWindow);
|
|
else
|
|
setTopRow(mTopRow + mScrollDeltaY);
|
|
int row = mTopRow;
|
|
if (mScrollDeltaY > 0) // scrolling down?
|
|
row+=mRowsInWindow - 1;
|
|
mousePosRC.row = row - 1;
|
|
int oldStartRow = mContents.lineCharToRowColumn(selectionBegin()).row+1;
|
|
int oldEndRow = mContents.lineCharToRowColumn(selectionEnd()).row+1;
|
|
invalidateRows(oldStartRow,oldEndRow);
|
|
mSelectionEnd = mContents.rowColumnToLineChar(mousePosRC);
|
|
invalidateRows(row,row);
|
|
}
|
|
|
|
// computeScrollY(Y);
|
|
}
|
|
|
|
void QConsole::mousePressEvent(QMouseEvent *event)
|
|
{
|
|
Qt::MouseButton button = event->button();
|
|
int X=event->pos().x();
|
|
int Y=event->pos().y();
|
|
|
|
QAbstractScrollArea::mousePressEvent(event);
|
|
|
|
//fKbdHandler.ExecuteMouseDown(Self, Button, Shift, X, Y);
|
|
|
|
if (button == Qt::LeftButton) {
|
|
// setMouseTracking(true);
|
|
RowColumn mousePosRC = pixelsToNearestRowColumn(X,Y);
|
|
LineChar mousePos = mContents.rowColumnToLineChar(mousePosRC);
|
|
//I couldn't track down why, but sometimes (and definitely not all the time)
|
|
//the block positioning is lost. This makes sure that the block is
|
|
//maintained in case they started a drag operation on the block
|
|
int oldStartRow = mContents.lineCharToRowColumn(selectionBegin()).row+1;
|
|
int oldEndRow = mContents.lineCharToRowColumn(selectionEnd()).row+1;
|
|
invalidateRows(oldStartRow,oldEndRow);
|
|
mSelectionBegin = mousePos;
|
|
mSelectionEnd = mousePos;
|
|
}
|
|
}
|
|
|
|
void QConsole::mouseReleaseEvent(QMouseEvent *event)
|
|
{
|
|
QAbstractScrollArea::mouseReleaseEvent(event);
|
|
mScrollTimer->stop();
|
|
// setMouseTracking(false);
|
|
|
|
}
|
|
|
|
void QConsole::mouseMoveEvent(QMouseEvent *event)
|
|
{
|
|
QAbstractScrollArea::mouseMoveEvent(event);
|
|
Qt::MouseButtons buttons = event->buttons();
|
|
int x=event->pos().x();
|
|
int y=event->pos().y();
|
|
|
|
if ((buttons == Qt::LeftButton)) {
|
|
// should we begin scrolling?
|
|
computeScrollY(y);
|
|
RowColumn mousePosRC = pixelsToNearestRowColumn(x, y);
|
|
LineChar mousePos = mContents.rowColumnToLineChar(mousePosRC);
|
|
//qDebug()<<x<<y<<mousePosRC.row<<mousePosRC.column<<mousePos.line<<mousePos.ch;
|
|
if (mScrollDeltaY == 0) {
|
|
int oldStartRow = mContents.lineCharToRowColumn(selectionBegin()).row+1;
|
|
int oldEndRow = mContents.lineCharToRowColumn(selectionEnd()).row+1;
|
|
invalidateRows(oldStartRow,oldEndRow);
|
|
mSelectionEnd = mousePos;
|
|
int row = mContents.lineCharToRowColumn(mSelectionEnd).row+1;
|
|
invalidateRows(row,row);
|
|
}
|
|
}
|
|
}
|
|
|
|
void QConsole::keyPressEvent(QKeyEvent *event)
|
|
{
|
|
switch(event->key()) {
|
|
case Qt::Key_Return:
|
|
case Qt::Key_Enter:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
emit commandInput(mCommand);
|
|
if (mMaxHistory>0 && !mCommand.trimmed().isEmpty()) {
|
|
if (mCommandHistory.isEmpty()
|
|
|| mCommandHistory.last()!=mCommand) {
|
|
if (mCommandHistory.length()==mMaxHistory) {
|
|
mCommandHistory.pop_front();
|
|
}
|
|
mCommandHistory.append(mCommand);
|
|
}
|
|
mHistoryIndex = mCommandHistory.length();
|
|
}
|
|
mCommand="";
|
|
addLine("");
|
|
return;
|
|
case Qt::Key_Up:
|
|
event->accept();
|
|
mHistoryIndex--;
|
|
loadCommandFromHistory();
|
|
return;
|
|
case Qt::Key_Down:
|
|
event->accept();
|
|
mHistoryIndex++;
|
|
loadCommandFromHistory();
|
|
return;
|
|
case Qt::Key_Left:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
if (mCaretChar>0 && mCaretChar<=mCurrentEditableLine.size()) {
|
|
setCaretChar(mCaretChar-1, !(event->modifiers() & Qt::ShiftModifier));
|
|
}
|
|
return;
|
|
case Qt::Key_Right:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
if (mCaretChar<mCurrentEditableLine.size()) {
|
|
setCaretChar(mCaretChar+1, !(event->modifiers() & Qt::ShiftModifier));
|
|
}
|
|
return;
|
|
case Qt::Key_Home:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
if (mCaretChar>0 && mCaretChar<=mCurrentEditableLine.size()) {
|
|
setCaretChar(0, !(event->modifiers() & Qt::ShiftModifier));
|
|
}
|
|
return;
|
|
case Qt::Key_End:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
if (mCaretChar<mCurrentEditableLine.size()) {
|
|
setCaretChar(mCurrentEditableLine.size(), !(event->modifiers() & Qt::ShiftModifier));
|
|
}
|
|
return;
|
|
case Qt::Key_Backspace:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
if (!mCurrentEditableLine.isEmpty() && (mCaretChar-1)<mCurrentEditableLine.size()
|
|
&&(mCaretChar-1>=0)) {
|
|
QString lastLine;
|
|
if (caretInSelection()) {
|
|
lastLine = removeSelection();
|
|
} else {
|
|
lastLine = mContents.getLastLine();
|
|
int len=mCurrentEditableLine.length();
|
|
mCaretChar--;
|
|
mCommand.remove(mCommand.length()-len+mCaretChar,1);
|
|
lastLine.remove(lastLine.length()-len+mCaretChar,1);
|
|
mCurrentEditableLine.remove(mCaretChar,1);
|
|
}
|
|
mContents.changeLastLine(lastLine);
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
} else {
|
|
if (hasSelection()) {
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
invalidate();
|
|
}
|
|
}
|
|
return;
|
|
case Qt::Key_Delete:
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
if (!mCurrentEditableLine.isEmpty() && (mCaretChar)<mCurrentEditableLine.size()
|
|
&&(mCaretChar>=0)) {
|
|
QString lastLine;
|
|
if (caretInSelection()) {
|
|
lastLine = removeSelection();
|
|
} else {
|
|
lastLine = mContents.getLastLine();
|
|
int len=mCurrentEditableLine.length();
|
|
mCommand.remove(mCommand.length()-len+mCaretChar,1);
|
|
lastLine.remove(lastLine.length()-len+mCaretChar,1);
|
|
mCurrentEditableLine.remove(mCaretChar,1);
|
|
}
|
|
mContents.changeLastLine(lastLine);
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
} else {
|
|
if (hasSelection()) {
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
invalidate();
|
|
}
|
|
}
|
|
return;
|
|
default:
|
|
if (!event->text().isEmpty()) {
|
|
event->accept();
|
|
if (mReadonly)
|
|
return;
|
|
textInputed(event->text());
|
|
return;
|
|
}
|
|
}
|
|
QAbstractScrollArea::keyPressEvent(event);
|
|
}
|
|
|
|
void QConsole::focusInEvent(QFocusEvent *)
|
|
{
|
|
showCaret();
|
|
}
|
|
|
|
void QConsole::focusOutEvent(QFocusEvent *)
|
|
{
|
|
hideCaret();
|
|
}
|
|
|
|
void QConsole::paintEvent(QPaintEvent *event)
|
|
{
|
|
if (mRowHeight==0)
|
|
return;
|
|
// Now paint everything while the caret is hidden.
|
|
QPainter painter(viewport());
|
|
//Get the invalidated rect.
|
|
QRect rcClip = event->rect();
|
|
QRect rcCaret= getCaretRect();
|
|
|
|
if (rcCaret == rcClip) {
|
|
// only update caret
|
|
painter.drawImage(rcCaret,*mContentImage,rcCaret);
|
|
} else {
|
|
int nL1, nL2;
|
|
// Compute the invalid area in lines
|
|
// lines
|
|
nL1 = std::min(std::max(mTopRow + rcClip.top() / mRowHeight, mTopRow), maxScrollHeight() + mRowsInWindow - 1 );
|
|
nL2 = std::min(std::max(mTopRow + (rcClip.bottom() + mRowHeight - 1) / mRowHeight, 1), maxScrollHeight() + mRowsInWindow - 1);
|
|
QPainter cachePainter(mContentImage.get());
|
|
cachePainter.setFont(font());
|
|
if (viewport()->rect() == rcClip) {
|
|
cachePainter.fillRect(rcClip, mBackground);
|
|
}
|
|
paintRows(cachePainter,nL1,nL2);
|
|
painter.drawImage(rcClip,*mContentImage,rcClip);
|
|
}
|
|
paintCaret(painter, rcCaret);
|
|
}
|
|
|
|
void QConsole::paintCaret(QPainter &painter, const QRect rcClip)
|
|
{
|
|
if (mBlinkStatus!=1)
|
|
return;
|
|
painter.setClipRect(rcClip);
|
|
ConsoleCaretType ct = ConsoleCaretType::ctHorizontalLine;
|
|
QColor caretColor = mForeground;
|
|
switch(ct) {
|
|
case ConsoleCaretType::ctVerticalLine:
|
|
painter.fillRect(rcClip.left()+1,rcClip.top(),rcClip.left()+2,rcClip.bottom(),caretColor);
|
|
break;
|
|
case ConsoleCaretType::ctHorizontalLine:
|
|
painter.fillRect(rcClip.left(),rcClip.bottom()-2,rcClip.right(),rcClip.bottom()-1,caretColor);
|
|
break;
|
|
case ConsoleCaretType::ctBlock:
|
|
painter.fillRect(rcClip, caretColor);
|
|
break;
|
|
case ConsoleCaretType::ctHalfBlock:
|
|
QRect rc=rcClip;
|
|
rc.setTop(rcClip.top()+rcClip.height() / 2);
|
|
painter.fillRect(rcClip, caretColor);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void QConsole::textInputed(const QString &text)
|
|
{
|
|
if (mContents.rows()<=0) {
|
|
mContents.addLine("");
|
|
}
|
|
QString lastLine;
|
|
if (caretInSelection()) {
|
|
lastLine = removeSelection();
|
|
} else {
|
|
lastLine = mContents.getLastLine();
|
|
}
|
|
if (mCaretChar>=mCurrentEditableLine.size()) {
|
|
mCommand += text;
|
|
mCurrentEditableLine += text;
|
|
mCaretChar=mCurrentEditableLine.size();
|
|
mContents.changeLastLine(lastLine+text);
|
|
} else {
|
|
int len=mCurrentEditableLine.length();
|
|
mCommand.insert(mCommand.length()-len+mCaretChar,text);
|
|
lastLine.insert(lastLine.length()-len+mCaretChar,text);
|
|
mCurrentEditableLine.insert(mCaretChar,text);
|
|
mContents.changeLastLine(lastLine);
|
|
mCaretChar+=text.length();
|
|
}
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
}
|
|
|
|
void QConsole::loadCommandFromHistory()
|
|
{
|
|
if (mMaxHistory<=0)
|
|
return;
|
|
if (mHistoryIndex<0)
|
|
mHistoryIndex=0;
|
|
if (mHistoryIndex<mCommandHistory.length())
|
|
mCommand = mCommandHistory[mHistoryIndex];
|
|
else
|
|
mCommand = "";
|
|
QString lastLine = mContents.getLastLine();
|
|
int len=mCurrentEditableLine.length();
|
|
lastLine.remove(lastLine.length()-len,INT_MAX);
|
|
mCurrentEditableLine=mCommand;
|
|
mCaretChar = mCurrentEditableLine.length();
|
|
mSelectionBegin = caretPos();
|
|
mSelectionEnd = caretPos();
|
|
mContents.changeLastLine(lastLine + mCommand);
|
|
}
|
|
|
|
LineChar QConsole::selectionBegin()
|
|
{
|
|
if (mSelectionBegin.line < mSelectionEnd.line ||
|
|
(mSelectionBegin.line == mSelectionEnd.line &&
|
|
mSelectionBegin.ch < mSelectionEnd.ch))
|
|
return mSelectionBegin;
|
|
return mSelectionEnd;
|
|
}
|
|
|
|
LineChar QConsole::selectionEnd()
|
|
{
|
|
if (mSelectionBegin.line < mSelectionEnd.line ||
|
|
(mSelectionBegin.line == mSelectionEnd.line &&
|
|
mSelectionBegin.ch < mSelectionEnd.ch))
|
|
return mSelectionEnd;
|
|
return mSelectionBegin;
|
|
}
|
|
|
|
void QConsole::setCaretChar(int newCaretChar, bool resetSelection)
|
|
{
|
|
RowColumn oldPosRC = caretRowColumn();
|
|
RowColumn oldSelBegin = mContents.lineCharToRowColumn(selectionBegin());
|
|
RowColumn oldSelEnd = mContents.lineCharToRowColumn(selectionEnd());
|
|
int oldStartRow = std::min(std::min(oldPosRC.row,oldSelBegin.row),oldSelEnd.row);
|
|
int oldEndRow = std::max(std::max(oldPosRC.row,oldSelBegin.row),oldSelEnd.row);
|
|
mCaretChar = newCaretChar;
|
|
LineChar newPos = caretPos();
|
|
RowColumn newPosRC = mContents.lineCharToRowColumn(newPos);
|
|
if (resetSelection)
|
|
mSelectionBegin = newPos;
|
|
mSelectionEnd = newPos;
|
|
|
|
int startRow = std::min(newPosRC.row, oldStartRow)+1;
|
|
int endRow = std::max(newPosRC.row, oldEndRow)+1;
|
|
invalidateRows(startRow, endRow);
|
|
}
|
|
|
|
bool QConsole::caretInSelection()
|
|
{
|
|
if (!hasSelection())
|
|
return false;
|
|
//LineChar selBegin = selectionBegin();
|
|
LineChar selEnd = selectionEnd();
|
|
QString lastline = mContents.getLastLine();
|
|
int editBeginChar = lastline.length() - mCurrentEditableLine.length();
|
|
if (selEnd.line == mContents.lines()-1 && selEnd.ch > editBeginChar ) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QString QConsole::removeSelection()
|
|
{
|
|
|
|
QString lastLine = mContents.getLastLine();
|
|
int len=mCurrentEditableLine.length();
|
|
LineChar selBegin = selectionBegin();
|
|
LineChar selEnd = selectionEnd();
|
|
int selLen = selEnd.ch -selBegin.ch;
|
|
int ch = selBegin.ch -(lastLine.length()-len);
|
|
if (selBegin.line < mContents.lines()-1) {
|
|
mCaretChar = 0;
|
|
selLen = selEnd.ch - (lastLine.length()-len);
|
|
} else if (ch<0) {
|
|
mCaretChar = 0;
|
|
selLen = selLen + ch;
|
|
} else {
|
|
mCaretChar=ch;
|
|
}
|
|
mCommand.remove(mCommand.length()-len+mCaretChar,selLen);
|
|
lastLine.remove(lastLine.length()-len+mCaretChar,selLen);
|
|
mCurrentEditableLine.remove(mCaretChar,selLen);
|
|
return lastLine;
|
|
}
|
|
|
|
bool QConsole::hasSelection()
|
|
{
|
|
return (mSelectionBegin.line != mSelectionEnd.line)
|
|
|| (mSelectionBegin.ch != mSelectionEnd.ch);
|
|
}
|
|
|
|
int QConsole::computeScrollY(int y)
|
|
{
|
|
QRect iScrollBounds = viewport()->rect();
|
|
if (y < iScrollBounds.top())
|
|
mScrollDeltaY = (y - iScrollBounds.top()) / mRowHeight - 1;
|
|
else if (y >= iScrollBounds.bottom())
|
|
mScrollDeltaY = (y - iScrollBounds.bottom()) / mRowHeight + 1;
|
|
else
|
|
mScrollDeltaY = 0;
|
|
|
|
if (mScrollDeltaY)
|
|
mScrollTimer->start();
|
|
return mScrollDeltaY;
|
|
}
|
|
|
|
RowColumn QConsole::pixelsToNearestRowColumn(int x, int y)
|
|
{
|
|
// Result is in display coordinates
|
|
// don't return a partially visible last line
|
|
if (y >= mRowsInWindow * mRowHeight) {
|
|
y = mRowsInWindow * mRowHeight - 1;
|
|
if (y < 0)
|
|
y = 0;
|
|
}
|
|
return {
|
|
std::max(0, (x - 2) / mColumnWidth),
|
|
mTopRow + (y / mRowHeight)-1
|
|
};
|
|
}
|
|
|
|
QString QConsole::lineBreak()
|
|
{
|
|
return "\r\n";
|
|
}
|
|
|
|
|
|
void QConsole::fontChanged()
|
|
{
|
|
recalcCharExtent();
|
|
sizeOrFontChanged(true);
|
|
}
|
|
|
|
bool QConsole::event(QEvent *event)
|
|
{
|
|
switch(event->type()) {
|
|
case QEvent::FontChange:
|
|
fontChanged();
|
|
break;
|
|
case QEvent::PaletteChange:
|
|
mBackground = palette().color(QPalette::Base);
|
|
mForeground = palette().color(QPalette::Text);
|
|
mSelectionBackground = palette().color(QPalette::Highlight);
|
|
mSelectionForeground = palette().color(QPalette::HighlightedText);
|
|
mInactiveSelectionBackground = palette().color(QPalette::Inactive,QPalette::Highlight);
|
|
mInactiveSelectionForeground = palette().color(QPalette::Inactive,QPalette::HighlightedText);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return QAbstractScrollArea::event(event);
|
|
}
|
|
|
|
void QConsole::resizeEvent(QResizeEvent *)
|
|
{
|
|
//resize the cache image
|
|
std::shared_ptr<QImage> image = std::make_shared<QImage>(clientWidth(),clientHeight(),
|
|
QImage::Format_ARGB32);
|
|
if (mContentImage) {
|
|
//QRect newRect = image->rect().intersected(mContentImage->rect());
|
|
QPainter painter(image.get());
|
|
painter.fillRect(viewport()->rect(),mBackground);
|
|
// painter.drawImage(newRect,*mContentImage);
|
|
}
|
|
|
|
mContentImage = image;
|
|
|
|
sizeOrFontChanged(false);
|
|
}
|
|
|
|
void QConsole::timerEvent(QTimerEvent *event)
|
|
{
|
|
if (event->timerId() == mBlinkTimerId) {
|
|
mBlinkStatus = 1- mBlinkStatus;
|
|
updateCaret();
|
|
}
|
|
}
|
|
|
|
void QConsole::inputMethodEvent(QInputMethodEvent *event)
|
|
{
|
|
if (mReadonly)
|
|
return;
|
|
QString s=event->commitString();
|
|
if (!s.isEmpty())
|
|
textInputed(s);
|
|
}
|
|
|
|
void QConsole::wheelEvent(QWheelEvent *event)
|
|
{
|
|
if (event->angleDelta().y()>0) {
|
|
verticalScrollBar()->setValue(verticalScrollBar()->value()-1);
|
|
event->accept();
|
|
return;
|
|
} else if (event->angleDelta().y()<0) {
|
|
verticalScrollBar()->setValue(verticalScrollBar()->value()+1);
|
|
event->accept();
|
|
return;
|
|
}
|
|
}
|
|
|
|
int ConsoleLines::rows() const
|
|
{
|
|
return mRows;
|
|
}
|
|
|
|
int ConsoleLines::lines() const
|
|
{
|
|
return mLines.count();
|
|
}
|
|
|
|
void ConsoleLines::layout()
|
|
{
|
|
if (!mConsole)
|
|
return;
|
|
if (mLayouting) {
|
|
mNeedRelayout = true;
|
|
return;
|
|
}
|
|
mLayouting = true;
|
|
mNeedRelayout = false;
|
|
emit layoutStarted();
|
|
mRows = 0;
|
|
bool forceUpdate = (mOldTabSize!=mConsole->tabSize());
|
|
for (PConsoleLine consoleLine: mLines) {
|
|
if (forceUpdate || consoleLine->maxColumns > mConsole->columnsPerRow()) {
|
|
consoleLine->maxColumns = breakLine(consoleLine->text,consoleLine->fragments);
|
|
}
|
|
mRows+=consoleLine->fragments.count();
|
|
}
|
|
emit layoutFinished();
|
|
mLayouting = false;
|
|
if (mNeedRelayout)
|
|
emit needRelayout();
|
|
}
|
|
|
|
ConsoleLines::ConsoleLines(QConsole *console)
|
|
{
|
|
mConsole = console;
|
|
mRows = 0;
|
|
mLayouting = false;
|
|
mNeedRelayout = false;
|
|
mOldTabSize = -1;
|
|
mMaxLines = 1000;
|
|
connect(this,&ConsoleLines::needRelayout,this,&ConsoleLines::layout);
|
|
}
|
|
|
|
void ConsoleLines::addLine(const QString &line)
|
|
{
|
|
PConsoleLine consoleLine=std::make_shared<ConsoleLine>();
|
|
consoleLine->text = line;
|
|
consoleLine->maxColumns = breakLine(line,consoleLine->fragments);
|
|
if (mLines.count()<mMaxLines || mMaxLines <= 0) {
|
|
mLines.append(consoleLine);
|
|
mRows += consoleLine->fragments.count();
|
|
emit rowsAdded(consoleLine->fragments.count());
|
|
} else {
|
|
PConsoleLine firstLine = mLines[0];
|
|
mLines.pop_front();
|
|
mRows -= firstLine->fragments.count();
|
|
mLines.append(consoleLine);
|
|
mRows += consoleLine->fragments.count();
|
|
emit layoutStarted();
|
|
emit layoutFinished();
|
|
}
|
|
}
|
|
|
|
void ConsoleLines::RemoveLastLine()
|
|
{
|
|
if (mLines.count()<=0)
|
|
return;
|
|
PConsoleLine consoleLine = mLines[mLines.count()-1];
|
|
mLines.pop_back();
|
|
mRows -= consoleLine->fragments.count();
|
|
emit lastRowsRemoved(consoleLine->fragments.count());
|
|
}
|
|
|
|
void ConsoleLines::changeLastLine(const QString &newLine)
|
|
{
|
|
if (mLines.count()<=0) {
|
|
return;
|
|
}
|
|
PConsoleLine consoleLine = mLines[mLines.count()-1];
|
|
int oldRows = consoleLine->fragments.count();
|
|
consoleLine->text = newLine;
|
|
breakLine(newLine,consoleLine->fragments);
|
|
int newRows = consoleLine->fragments.count();
|
|
if (newRows == oldRows) {
|
|
emit lastRowsChanged(oldRows);
|
|
return ;
|
|
} else {
|
|
mRows -= oldRows;
|
|
mRows += newRows;
|
|
emit layoutStarted();
|
|
emit layoutFinished();
|
|
}
|
|
}
|
|
|
|
QString ConsoleLines::getLastLine()
|
|
{
|
|
if (mLines.count()<=0)
|
|
return "";
|
|
return mLines[mLines.count()-1]->text;
|
|
}
|
|
|
|
QString ConsoleLines::getLine(int line)
|
|
{
|
|
if (line>=0 && line < mLines.count()) {
|
|
return mLines[line]->text;
|
|
}
|
|
return "";
|
|
}
|
|
|
|
QChar ConsoleLines::getChar(int line, int ch)
|
|
{
|
|
QString s = getLine(line);
|
|
if (ch>=0 && ch<s.length()) {
|
|
return s[ch];
|
|
} else {
|
|
return QChar();
|
|
}
|
|
}
|
|
|
|
QChar ConsoleLines::getChar(const LineChar &lineChar)
|
|
{
|
|
return getChar(lineChar.line,lineChar.ch);
|
|
}
|
|
|
|
QStringList ConsoleLines::getRows(int startRow, int endRow)
|
|
{
|
|
if (startRow>mRows)
|
|
return QStringList();
|
|
if (startRow > endRow)
|
|
return QStringList();
|
|
QStringList lst;
|
|
int row = 0;
|
|
for (PConsoleLine line:mLines) {
|
|
for (const QString& s:line->fragments) {
|
|
row+=1;
|
|
if (row>endRow) {
|
|
return lst;
|
|
}
|
|
if (row>=startRow) {
|
|
lst.append(s);
|
|
}
|
|
}
|
|
}
|
|
return lst;
|
|
}
|
|
|
|
LineChar ConsoleLines::rowColumnToLineChar(const RowColumn &rowColumn)
|
|
{
|
|
return rowColumnToLineChar(rowColumn.row,rowColumn.column);
|
|
}
|
|
|
|
LineChar ConsoleLines::rowColumnToLineChar(int row, int column)
|
|
{
|
|
LineChar result{column,mLines.size()-1};
|
|
int rows=0;
|
|
for (int i=0;i<mLines.size();i++) {
|
|
PConsoleLine line = mLines[i];
|
|
if (row >= rows && row<rows+line->fragments.size()) {
|
|
int r=row - rows;
|
|
QString fragment = line->fragments[r];
|
|
int columnsBefore = 0;
|
|
int charsBefore = 0;
|
|
for (int j=0;j<r;j++) {
|
|
charsBefore += line->fragments[j].length();
|
|
}
|
|
for (int j=0;j<fragment.size();j++) {
|
|
QChar ch = fragment[j];
|
|
int charColumns= mConsole->charColumns(ch, columnsBefore);
|
|
if (column>=columnsBefore && column<columnsBefore+charColumns) {
|
|
result.ch = charsBefore + j;
|
|
break;
|
|
}
|
|
columnsBefore += charColumns;
|
|
}
|
|
result.line = i;
|
|
break;
|
|
}
|
|
rows += line->fragments.size();
|
|
}
|
|
return result;
|
|
}
|
|
|
|
RowColumn ConsoleLines::lineCharToRowColumn(const LineChar &lineChar)
|
|
{
|
|
return lineCharToRowColumn(lineChar.line,lineChar.ch);
|
|
}
|
|
|
|
RowColumn ConsoleLines::lineCharToRowColumn(int line, int ch)
|
|
{
|
|
RowColumn result{ch,std::max(0,mRows-1)};
|
|
int rowsBefore = 0;
|
|
if (line>=0 && line < mLines.size()) {
|
|
for (int i=0;i<line;i++) {
|
|
int rows = mLines[i]->fragments.size();
|
|
rowsBefore += rows;
|
|
}
|
|
PConsoleLine consoleLine = mLines[line];
|
|
int charsBefore = 0;
|
|
for (int r=0;r<consoleLine->fragments.size();r++) {
|
|
int chars = consoleLine->fragments[r].size();
|
|
if (r==consoleLine->fragments.size()-1 || (ch>=charsBefore && ch<charsBefore+chars)) {
|
|
QString fragment = consoleLine->fragments[r];
|
|
int columnsBefore = 0;
|
|
int len = std::min(ch-charsBefore,fragment.size());
|
|
for (int j=0;j<len;j++) {
|
|
QChar ch = fragment[j];
|
|
int charColumns = mConsole->charColumns(ch,columnsBefore);
|
|
columnsBefore += charColumns;
|
|
}
|
|
result.column=columnsBefore;
|
|
result.row = rowsBefore + r;
|
|
break;
|
|
}
|
|
charsBefore += chars;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
bool ConsoleLines::layouting() const
|
|
{
|
|
return mLayouting;
|
|
}
|
|
|
|
int ConsoleLines::breakLine(const QString &line, QStringList &fragments)
|
|
{
|
|
fragments.clear();
|
|
QString s = "";
|
|
int maxColLen = 0;
|
|
int columnsBefore = 0;
|
|
for (QChar ch:line) {
|
|
int charColumn = mConsole->charColumns(ch,columnsBefore);
|
|
if (charColumn + columnsBefore > mConsole->columnsPerRow()) {
|
|
if (ch == '\t') {
|
|
if (columnsBefore != mConsole->columnsPerRow()) {
|
|
charColumn = 0;
|
|
} else
|
|
charColumn = mConsole->tabSize();
|
|
}
|
|
fragments.append(s);
|
|
if (columnsBefore > maxColLen) {
|
|
maxColLen = columnsBefore;
|
|
}
|
|
s = "";
|
|
columnsBefore = 0;
|
|
}
|
|
if (charColumn > 0) {
|
|
columnsBefore += charColumn;
|
|
s += ch;
|
|
}
|
|
}
|
|
if (fragments.count() == 0 || !s.isEmpty()) {
|
|
fragments.append(s);
|
|
if (columnsBefore > maxColLen) {
|
|
maxColLen = columnsBefore;
|
|
}
|
|
}
|
|
return maxColLen;
|
|
}
|
|
|
|
int ConsoleLines::getMaxLines() const
|
|
{
|
|
return mMaxLines;
|
|
}
|
|
|
|
int ConsoleLines::maxLines() const
|
|
{
|
|
return mMaxLines;
|
|
}
|
|
|
|
void ConsoleLines::setMaxLines(int maxLines)
|
|
{
|
|
mMaxLines = maxLines;
|
|
if (mMaxLines > 0) {
|
|
while (mLines.count()>mMaxLines) {
|
|
mLines.pop_front();
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConsoleLines::clear()
|
|
{
|
|
mLines.clear();
|
|
mRows = 0;
|
|
}
|