Files
newspark110/ui/forms/qg_commandwidget.cpp
Chenwenxuan edac2715f0 init
2024-03-06 14:54:30 +08:00

300 lines
8.8 KiB
C++

/****************************************************************************
**
** This file is part of the LibreCAD project, a 2D CAD program
**
** Copyright (C) 2010 R. van Twisk (librecad@rvt.dds.nl)
** Copyright (C) 2001-2003 RibbonSoft. All rights reserved.
**
**
** This file may be distributed and/or modified under the terms of the
** GNU General Public License version 2 as published by the Free Software
** Foundation and appearing in the file gpl-2.0.txt included in the
** packaging of this file.
**
** 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, write to the Free Software
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
**
** This copyright notice MUST APPEAR in all copies of the script!
**
**********************************************************************/
#include "qg_commandwidget.h"
#include <algorithm>
#include <QAction>
#include <QKeyEvent>
#include <QFileDialog>
#include <QSettings>
#include "qg_actionhandler.h"
#include "rs_commands.h"
#include "rs_commandevent.h"
#include "rs_system.h"
#include "rs_utility.h"
/*
* Constructs a QG_CommandWidget as a child of 'parent', with the
* name 'name' and widget flags set to 'f'.
*/
QG_CommandWidget::QG_CommandWidget(QWidget* parent, const char* name, Qt::WindowFlags fl)
: QWidget(parent, fl)
, actionHandler(nullptr)
{
setObjectName(name);
setupUi(this);
connect(leCommand, SIGNAL(command(QString)), this, SLOT(handleCommand(QString)));
connect(leCommand, SIGNAL(escape()), this, SLOT(escape()));
connect(leCommand, SIGNAL(focusOut()), this, SLOT(setNormalMode()));
connect(leCommand, SIGNAL(focusIn()), this, SLOT(setCommandMode()));
connect(leCommand, SIGNAL(tabPressed()), this, SLOT(tabPressed()));
connect(leCommand, SIGNAL(clearCommandsHistory()), teHistory, SLOT(clear()));
connect(leCommand, SIGNAL(message(QString)), this, SLOT(appendHistory(QString)));
connect(leCommand, &QG_CommandEdit::keycode, this, &QG_CommandWidget::handleKeycode);
auto a1 = new QAction(QObject::tr("Keycode mode"), this);
a1->setObjectName("keycode_action");
a1->setCheckable(true);
connect(a1, &QAction::toggled, this, &QG_CommandWidget::setKeycodeMode);
options_button->addAction(a1);
QSettings settings;
if (settings.value("Widgets/KeycodeMode", false).toBool())
{
leCommand->keycode_mode = true;
a1->setChecked(true);
}
auto a2 = new QAction(QObject::tr("Load command file"), this);
connect(a2, &QAction::triggered, this, &QG_CommandWidget::chooseCommandFile);
options_button->addAction(a2);
auto a3 = new QAction(QObject::tr("Paste multiple commands"), this);
connect(a3, &QAction::triggered, leCommand, &QG_CommandEdit::modifiedPaste);
options_button->addAction(a3);
options_button->setStyleSheet("QToolButton::menu-indicator { image: none; }");
}
/*
* Destroys the object and frees any allocated resources
*/
QG_CommandWidget::~QG_CommandWidget()
{
QSettings settings;
auto action = findChild<QAction*>("keycode_action");
settings.setValue("Widgets/KeycodeMode", action->isChecked());
}
/*
* Sets the strings of the subwidgets using the current
* language.
*/
void QG_CommandWidget::languageChange()
{
retranslateUi(this);
}
bool QG_CommandWidget::eventFilter(QObject */*obj*/, QEvent *event)
{
if (event->type() == QEvent::KeyPress) {
QKeyEvent* e=static_cast<QKeyEvent*>(event);
int key {e->key()};
switch(key) {
case Qt::Key_Return:
case Qt::Key_Enter:
if(!leCommand->text().size())
return false;
else
break;
case Qt::Key_Escape:
return false;
default:
break;
}
//detect Ctl- Alt- modifier, but not Shift
//This should avoid filtering shortcuts, such as Ctl-C
Qt::KeyboardModifiers modifiers {e->modifiers()};
if ( !(Qt::GroupSwitchModifier == modifiers && Qt::Key_At == key) // let '@' key pass for relative coords
&& modifiers != Qt::KeypadModifier
&& modifiers & (Qt::KeyboardModifierMask ^ Qt::ShiftModifier)) {
return false;
}
event->accept();
this->setFocus();
QKeyEvent * newEvent = new QKeyEvent(*static_cast<QKeyEvent*>(event));
QApplication::postEvent(leCommand, newEvent);
return true;
}
return false;
}
void QG_CommandWidget::setFocus()
{
if (!isActiveWindow())
activateWindow();
auto newEvent = new QFocusEvent(QEvent::FocusIn);
QApplication::postEvent(leCommand, newEvent);
leCommand->setFocus();
}
void QG_CommandWidget::setCommand(const QString& cmd) {
if (cmd!="") {
lCommand->setText(cmd);
} else {
lCommand->setText(tr("Command:"));
}
leCommand->setText("");
}
void QG_CommandWidget::appendHistory(const QString& msg) {
teHistory->append(msg);
}
void QG_CommandWidget::handleCommand(QString cmd)
{
cmd = cmd.simplified();
bool isAction=false;
if (!cmd.isEmpty()) {
appendHistory(cmd);
}
if (actionHandler) {
isAction=actionHandler->command(cmd);
}
if (!isAction && !(cmd.contains(',') || cmd.at(0)=='@')) {
appendHistory(tr("Unknown command: %1").arg(cmd));
}
leCommand->setText("");
}
void QG_CommandWidget::tabPressed() {
if (actionHandler) {
QStringList reducedChoice;
QString typed = leCommand->text();
QStringList choice;
// check current command:
choice = actionHandler->getAvailableCommands();
if (choice.count()==0) {
choice = RS_COMMANDS->complete(typed);
}
for (QStringList::Iterator it = choice.begin(); it != choice.end(); ++it) {
if (typed.isEmpty() || (*it).startsWith(typed)) {
reducedChoice << (*it);
}
}
// command found:
if (reducedChoice.count()==1) {
leCommand->setText(reducedChoice.first());
}
else if (reducedChoice.count()>0) {
QString const& proposal = this->getRootCommand(reducedChoice, typed);
appendHistory(reducedChoice.join(", "));
leCommand -> setText(proposal);
}
}
}
void QG_CommandWidget::escape() {
//leCommand->clearFocus();
if (actionHandler) {
actionHandler->command(QString(tr("escape", "escape, go back from action steps")));
}
}
void QG_CommandWidget::setActionHandler(QG_ActionHandler* ah) {
actionHandler = ah;
}
void QG_CommandWidget::setCommandMode() {
QPalette palette;
palette.setColor(lCommand->foregroundRole(), Qt::blue);
lCommand->setPalette(palette);
}
void QG_CommandWidget::setNormalMode() {
QPalette palette;
palette.setColor(lCommand->foregroundRole(), Qt::black);
lCommand->setPalette(palette);
}
QString QG_CommandWidget::getRootCommand( const QStringList & cmdList, const QString & typed ) {
//do we have to check for empty cmdList?
if(cmdList.empty()) return QString();
//find the shortest string in cmdList
auto const& shortestString = * std::min_element(cmdList.begin(), cmdList.end(),
[](QString const& a, QString const& b) -> bool
{
return a.size() < b.size();
}
);
int const lengthShortestString = shortestString.size();
// Now we parse the cmdList list, character of each item by character.
int low = typed.length();
int high = lengthShortestString + 1;
while(high > low + 1) {
int mid = (high + low)/2;
bool common = true;
QString const& proposal = shortestString.left(mid);
for(auto const& substring: cmdList) {
if(!substring.startsWith(proposal)) {
common = false;
break;
}
}
if(common) {
low = mid;
}
else {
high = mid;
}
}
// As we assign just before mid value to low (if strings are common), we can use it as parameter for left.
// If not common -> low value does not changes, even if escaping from the while. This avoids weird behaviors like continuing completion when pressing tab.
return shortestString.left(low);
}
void QG_CommandWidget::chooseCommandFile()
{
QString path = QFileDialog::getOpenFileName(this);
if (!path.isEmpty())
{
leCommand->readCommandFile(path);
}
}
void QG_CommandWidget::handleKeycode(QString code)
{
if (actionHandler->keycode(code))
{
leCommand->clear();
}
}
void QG_CommandWidget::setKeycodeMode(bool state)
{
leCommand->keycode_mode = state;
}