qucs_s/qucs/qucs_actions.cpp

1740 lines
53 KiB
C++
Raw Normal View History

2006-03-28 06:10:52 +00:00
/***************************************************************************
qucs_actions.cpp
2006-03-28 06:10:52 +00:00
-----------------
begin : Sat May 1 2004
copyright : (C) 2004 by Michael Margraf
email : michael.margraf@alumni.tu-berlin.de
***************************************************************************/
/* Copyright (C) 2014 Guilherme Brondani Torri <guitorri@gmail.com> */
2006-03-28 06:10:52 +00:00
/***************************************************************************
* *
* 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 2 of the License, or *
* (at your option) any later version. *
* *
***************************************************************************/
/*! \file qucs_actions.cpp
* \brief Actions implementation for the GUI menu items
*/
2013-03-15 18:43:02 +01:00
#include <QtCore>
2006-03-28 06:10:52 +00:00
#include <stdlib.h>
#include <limits.h>
2006-03-28 06:10:52 +00:00
#include <QProcess>
#include <qt3_compat/q3ptrlist.h>
2023-01-17 13:27:12 +03:00
#include <QRegularExpressionValidator>
2014-11-04 12:48:23 +08:00
#include <QLineEdit>
#include <QAction>
#include <QStatusBar>
#include <QMessageBox>
#include <QFileDialog>
#include <QMenu>
#include <QComboBox>
#include <QDockWidget>
2015-01-07 03:38:38 +08:00
#include <QTreeWidgetItem>
#include <QMutableHashIterator>
#include <QListWidget>
#include <QDesktopServices>
2006-03-28 06:10:52 +00:00
#include "portsymbol.h"
#include "projectView.h"
2006-03-28 06:10:52 +00:00
#include "main.h"
#include "qucs.h"
#include "schematic.h"
#include "textdoc.h"
#include "mouseactions.h"
#include "messagedock.h"
2006-03-28 06:10:52 +00:00
#include "components/ground.h"
#include "components/subcirport.h"
#include "components/equation.h"
#include "spicecomponents/sp_nutmeg.h"
2006-03-28 06:10:52 +00:00
#include "dialogs/matchdialog.h"
2006-04-10 06:12:35 +00:00
#include "dialogs/changedialog.h"
#include "dialogs/searchdialog.h"
2006-06-06 06:14:17 +00:00
#include "dialogs/librarydialog.h"
2014-03-02 22:50:12 +01:00
#include "dialogs/loaddialog.h"
2006-07-03 06:02:08 +00:00
#include "dialogs/importdialog.h"
2015-02-01 22:20:16 +01:00
#include "dialogs/aboutdialog.h"
2014-02-19 22:01:06 +01:00
#include "module.h"
2006-03-28 06:10:52 +00:00
2022-02-19 22:21:20 +01:00
#include "extsimkernels/xyce.h"
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2006-03-28 06:10:52 +00:00
// for editing component name on schematic
2023-01-17 13:27:12 +03:00
QRegularExpression Expr_CompProp;
QRegularExpressionValidator Val_CompProp(Expr_CompProp, 0);
2006-03-28 06:10:52 +00:00
// -----------------------------------------------------------------------
// This function is called from all toggle actions.
bool QucsApp::performToggleAction(bool on, QAction *Action,
pToggleFunc Function, pMouseFunc MouseMove, pMouseFunc2 MousePress)
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
// Perform toggle release clean up.
2006-03-28 06:10:52 +00:00
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
activeAction = 0; // no action active
// Return to select mode.
slotEscape();
2006-03-28 06:10:52 +00:00
return false;
}
do {
if(Function) if((Doc->*Function)()) {
Action->blockSignals(true);
Action->setChecked(false); // release toolbar button
2006-03-28 06:10:52 +00:00
Action->blockSignals(false);
Doc->viewport()->update();
break;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
2006-03-28 06:10:52 +00:00
activeAction->blockSignals(false);
}
activeAction = Action;
MouseMoveAction = MouseMove;
MousePressAction = MousePress;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
} while(false); // to perform "break"
Doc->viewport()->update();
return true;
}
// -----------------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "set on grid" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotOnGrid(bool on)
{
performToggleAction(on, onGrid, &Schematic::elementsOnGrid,
&MouseActions::MMoveOnGrid, &MouseActions::MPressOnGrid);
}
// -----------------------------------------------------------------------
// Is called when the rotate toolbar button is pressed.
void QucsApp::slotEditRotate(bool on)
{
performToggleAction(on, editRotate, &Schematic::rotateElements,
&MouseActions::MMoveRotate, &MouseActions::MPressRotate);
}
// -----------------------------------------------------------------------
// Is called when the mirror toolbar button is pressed.
void QucsApp::slotEditMirrorX(bool on)
{
performToggleAction(on, editMirror, &Schematic::mirrorXComponents,
&MouseActions::MMoveMirrorX, &MouseActions::MPressMirrorX);
}
// -----------------------------------------------------------------------
// Is called when the mirror toolbar button is pressed.
void QucsApp::slotEditMirrorY(bool on)
{
performToggleAction(on, editMirrorY, &Schematic::mirrorYComponents,
&MouseActions::MMoveMirrorY, &MouseActions::MPressMirrorY);
}
// -----------------------------------------------------------------------
// Is called when the activate/deactivate toolbar button is pressed.
// It also comments out the selected text on a text document
// \todo update the status or tooltip message
void QucsApp::slotEditActivate (bool on)
2006-03-28 06:10:52 +00:00
{
TextDoc * Doc = (TextDoc *) DocumentTab->currentWidget();
if (isTextDocument (Doc)) {
2013-04-06 19:06:49 +02:00
//TODO Doc->clearParagraphBackground (Doc->tmpPosX);
Doc->commentSelected ();
editActivate->blockSignals (true);
editActivate->setChecked(false); // release toolbar button
editActivate->blockSignals (false);
2006-03-28 06:10:52 +00:00
}
2006-04-10 06:12:35 +00:00
else
performToggleAction (on, editActivate,
2006-05-05 06:00:05 +00:00
&Schematic::activateSelectedComponents,
&MouseActions::MMoveActivate, &MouseActions::MPressActivate);
2006-03-28 06:10:52 +00:00
}
// ------------------------------------------------------------------------
// Is called if "Delete"-Button is pressed.
void QucsApp::slotEditDelete(bool on)
{
TextDoc *Doc = (TextDoc*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
2006-03-28 06:10:52 +00:00
Doc->viewport()->setFocus();
2013-04-06 19:06:49 +02:00
//Doc->del();
Doc->textCursor().deleteChar();
2006-03-28 06:10:52 +00:00
editDelete->blockSignals(true);
editDelete->setChecked(false); // release toolbar button
2006-03-28 06:10:52 +00:00
editDelete->blockSignals(false);
}
2006-04-10 06:12:35 +00:00
else
performToggleAction(on, editDelete, &Schematic::deleteElements,
&MouseActions::MMoveDelete, &MouseActions::MPressDelete);
2006-03-28 06:10:52 +00:00
}
// -----------------------------------------------------------------------
// Is called if "Wire"-Button is pressed.
void QucsApp::slotSetWire(bool on)
{
performToggleAction(on, insWire, 0,
&MouseActions::MMoveWire1, &MouseActions::MPressWire1);
}
// -----------------------------------------------------------------------
void QucsApp::slotInsertLabel(bool on)
{
performToggleAction(on, insLabel, 0,
&MouseActions::MMoveLabel, &MouseActions::MPressLabel);
}
// -----------------------------------------------------------------------
void QucsApp::slotSetMarker(bool on)
{
performToggleAction(on, setMarker, 0,
&MouseActions::MMoveMarker, &MouseActions::MPressMarker);
}
// -----------------------------------------------------------------------
// Toolbar button to update the diagram limits using the mouse - aka zooming.
void QucsApp::slotSetDiagramLimits(bool on)
{
performToggleAction(on, setDiagramLimits, 0,
&MouseActions::MMoveSetLimits, &MouseActions::MPressSetLimits);
}
// -----------------------------------------------------------------------
// Context menu option to reset the diagram limits to defaults.
void QucsApp::slotResetDiagramLimits()
{
if (view->focusElement && view->focusElement->Type == isDiagram)
{
Diagram* diagram = (Diagram*)(view->focusElement);
diagram->xAxis.autoScale = true;
diagram->yAxis.autoScale = true;
diagram->zAxis.autoScale = true;
// Now read in the data.
auto* Doc = (Schematic*)DocumentTab->currentWidget();
QFileInfo Info(Doc->DocName);
QString defaultDataSet = Info.absolutePath() + QDir::separator() + Doc->DataSet;
diagram->loadGraphData(defaultDataSet);
Doc->setChanged(true, true);
Doc->viewport()->update();
}
// Return to select mode (in case SetDiagramLimits is still selected).
slotEscape();
}
2006-03-28 06:10:52 +00:00
// -----------------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "move component text" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotMoveText(bool on)
{
performToggleAction(on, moveText, 0,
&MouseActions::MMoveMoveTextB, &MouseActions::MPressMoveText);
}
// -----------------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Zoom in" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotZoomIn(bool on)
{
auto *Doc = (TextDoc*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
Doc->zoomBy(1.5f);
2006-04-10 06:12:35 +00:00
magPlus->blockSignals(true);
magPlus->setChecked(false);
2006-04-10 06:12:35 +00:00
magPlus->blockSignals(false);
}
else
performToggleAction(on, magPlus, 0,
2006-03-28 06:10:52 +00:00
&MouseActions::MMoveZoomIn, &MouseActions::MPressZoomIn);
}
void QucsApp::slotEscape()
{
select->setChecked(true);
slotSearchClear();
}
2006-03-28 06:10:52 +00:00
// -----------------------------------------------------------------------
// Is called when the select toolbar button is pressed.
void QucsApp::slotSelect(bool on)
{
QWidget *w = DocumentTab->currentWidget();
if(isTextDocument(w)) {
2006-04-10 06:12:35 +00:00
((TextDoc*)w)->viewport()->setFocus();
select->blockSignals(true);
select->setChecked(true);
select->blockSignals(false);
2006-03-28 06:10:52 +00:00
return;
}
// goto to insertWire mode if ESC pressed during wiring
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(MouseMoveAction == &MouseActions::MMoveWire2) {
MouseMoveAction = &MouseActions::MMoveWire1;
MousePressAction = &MouseActions::MPressWire1;
Doc->viewport()->update();
select->blockSignals(true);
select->setChecked(false);
select->blockSignals(false);
return;
}
2006-03-28 06:10:52 +00:00
if(performToggleAction(on, select, 0, 0, &MouseActions::MPressSelect)) {
MouseReleaseAction = &MouseActions::MReleaseSelect;
MouseDoubleClickAction = &MouseActions::MDoubleClickSelect;
}
}
2014-12-11 13:28:36 +08:00
// --------------------------------------------------------------------
void QucsApp::slotEditCut()
{
statusBar()->showMessage(tr("Cutting selection..."));
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2014-12-11 13:28:36 +08:00
QWidget *Doc = DocumentTab->currentWidget();
2014-12-11 13:28:36 +08:00
if(isTextDocument (Doc)) {
((TextDoc *)Doc)->cut();
} else {
((Schematic *)Doc)->cut();
}
statusBar()->showMessage(tr("Ready."));
2014-12-11 13:28:36 +08:00
}
// --------------------------------------------------------------------
void QucsApp::slotEditCopy()
{
statusBar()->showMessage(tr("Copying selection to clipboard..."));
2014-12-11 13:28:36 +08:00
QWidget *Doc = DocumentTab->currentWidget();
2014-12-11 13:28:36 +08:00
if(isTextDocument (Doc)) {
((TextDoc *)Doc)->copy();
} else {
((Schematic *)Doc)->copy();
}
statusBar()->showMessage(tr("Ready."));
2014-12-11 13:28:36 +08:00
}
2006-03-28 06:10:52 +00:00
// -----------------------------------------------------------------------
void QucsApp::slotEditPaste(bool on)
{
2014-01-31 16:59:40 +00:00
// get the current document
QWidget *Doc = DocumentTab->currentWidget();
2014-01-31 16:59:40 +00:00
// if the current document is a text document paste in
// the contents of the clipboard as text
2014-12-11 13:28:36 +08:00
if(isTextDocument (Doc))
2014-01-31 16:59:40 +00:00
{
2006-03-28 06:10:52 +00:00
((TextDoc*)Doc)->paste();
editPaste->blockSignals(true);
editPaste->setChecked(false); // release toolbar button
2006-03-28 06:10:52 +00:00
editPaste->blockSignals(false);
return;
}
2014-12-11 13:28:36 +08:00
else {
// if it's not a text doc, prevent the user from editing
// while we perform the paste operation
2015-01-28 00:14:17 +08:00
slotHideEdit();
2014-12-11 13:28:36 +08:00
if(!on)
{
MouseMoveAction = 0;
MousePressAction = 0;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
activeAction = 0; // no action active
return;
}
2014-12-11 13:28:36 +08:00
if(!view->pasteElements((Schematic *)Doc))
{
editPaste->blockSignals(true); // do not call toggle slot
editPaste->setChecked(false); // set toolbar button off
editPaste->blockSignals(false);
return; // if clipboard empty
}
if(activeAction)
{
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
activeAction->blockSignals(false);
}
activeAction = editPaste;
MouseMoveAction = &MouseActions::MMovePaste;
view->movingRotated = 0;
2006-03-28 06:10:52 +00:00
MousePressAction = 0;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
}
}
2006-04-18 06:03:52 +00:00
// -----------------------------------------------------------------------
void QucsApp::slotInsertEntity ()
2006-04-18 06:03:52 +00:00
{
TextDoc * Doc = (TextDoc *) DocumentTab->currentWidget ();
Doc->viewport()->setFocus ();
2013-04-06 19:06:49 +02:00
//TODO Doc->clearParagraphBackground (Doc->tmpPosX);
Doc->insertSkeleton ();
2006-04-28 06:04:44 +00:00
2013-04-06 19:06:49 +02:00
//int x, y;
//Doc->getCursorPosition (&x, &y);
//x = Doc->textCursor().blockNumber();
//y = Doc->textCursor().columnNumber();
Doc->slotCursorPosChanged();
2006-04-18 06:03:52 +00:00
}
2006-03-28 06:10:52 +00:00
// -----------------------------------------------------------------------
// Is called when the mouse is clicked upon the equation toolbar button.
void QucsApp::slotInsertEquation(bool on)
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
activeAction = 0; // no action active
return;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
2006-03-28 06:10:52 +00:00
activeAction->blockSignals(false);
}
activeAction = insEquation;
if(view->selElem)
delete view->selElem; // delete previously selected component
if (QucsSettings.DefaultSimulator == spicecompat::simNgspice) {
view->selElem = new NutmegEquation();
} else {
view->selElem = new Equation();
}
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
MouseMoveAction = &MouseActions::MMoveElement;
MousePressAction = &MouseActions::MPressElement;
}
// -----------------------------------------------------------------------
// Is called when the mouse is clicked upon the ground toolbar button.
void QucsApp::slotInsertGround(bool on)
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
activeAction = 0; // no action active
return;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
2006-03-28 06:10:52 +00:00
activeAction->blockSignals(false);
}
activeAction = insGround;
if(view->selElem)
delete view->selElem; // delete previously selected component
view->selElem = new Ground();
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
MouseMoveAction = &MouseActions::MMoveElement;
MousePressAction = &MouseActions::MPressElement;
}
// -----------------------------------------------------------------------
// Is called when the mouse is clicked upon the port toolbar button.
void QucsApp::slotInsertPort(bool on)
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
activeAction = 0; // no action active
return;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
2006-03-28 06:10:52 +00:00
activeAction->blockSignals(false);
}
activeAction = insPort;
if(view->selElem)
delete view->selElem; // delete previously selected component
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if (Doc->symbolMode) {
view->selElem = new PortSymbol();
} else {
view->selElem = new SubCirPort();
}
2006-03-28 06:10:52 +00:00
MouseMoveAction = &MouseActions::MMoveElement;
MousePressAction = &MouseActions::MPressElement;
}
// --------------------------------------------------------------
// Is called, when "Undo"-Button is pressed.
void QucsApp::slotEditUndo()
{
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
2006-03-28 06:10:52 +00:00
((TextDoc*)Doc)->viewport()->setFocus();
((TextDoc*)Doc)->undo();
return;
}
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Doc->undo();
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Undo"-Button is pressed.
void QucsApp::slotEditRedo()
{
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
2006-03-28 06:10:52 +00:00
((TextDoc*)Doc)->viewport()->setFocus();
((TextDoc*)Doc)->redo();
return;
}
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Doc->redo();
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Align top" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotAlignTop()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
if(!Doc->aligning(0))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Align bottom" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotAlignBottom()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
if(!Doc->aligning(1))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Align left" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotAlignLeft()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
if(!Doc->aligning(2))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Align right" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotAlignRight()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-03-28 06:10:52 +00:00
if(!Doc->aligning(3))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Distribute horizontally" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotDistribHoriz()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-05-26 06:03:15 +00:00
Doc->distributeHorizontal();
2006-03-28 06:10:52 +00:00
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Distribute vertically" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotDistribVert()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-05-26 06:03:15 +00:00
Doc->distributeVertical();
2006-03-28 06:10:52 +00:00
Doc->viewport()->update();
}
2006-11-06 06:58:05 +00:00
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Center horizontally" action is triggered.
2006-11-06 06:58:05 +00:00
void QucsApp::slotCenterHorizontal()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-11-06 06:58:05 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-11-06 06:58:05 +00:00
if(!Doc->aligning(4))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called, when "Center vertically" action is triggered.
2006-11-06 06:58:05 +00:00
void QucsApp::slotCenterVertical()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-11-06 06:58:05 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-11-06 06:58:05 +00:00
if(!Doc->aligning(5))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
2006-03-28 06:10:52 +00:00
// ---------------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called when the "select all" action is triggered.
2006-03-28 06:10:52 +00:00
void QucsApp::slotSelectAll()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
QWidget *Doc = DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
2006-03-28 06:10:52 +00:00
((TextDoc*)Doc)->viewport()->setFocus();
2013-04-06 19:06:49 +02:00
//((TextDoc*)Doc)->selectAll(true);
((TextDoc*)Doc)->selectAll();
2006-03-28 06:10:52 +00:00
}
2006-04-10 06:12:35 +00:00
else {
auto selectionRect = ((Schematic*)Doc)->allBoundingRect().marginsAdded(QMargins{1, 1, 1, 1});
((Schematic*)Doc)->selectElements(selectionRect, true, false);
2006-04-10 06:12:35 +00:00
((Schematic*)Doc)->viewport()->update();
}
}
2006-05-22 06:01:55 +00:00
// ---------------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called when the "select markers" action is triggered.
2006-05-22 06:01:55 +00:00
void QucsApp::slotSelectMarker()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-05-22 06:01:55 +00:00
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-05-22 06:01:55 +00:00
Doc->selectMarkers();
Doc->viewport()->update();
}
extern QString lastDirOpenSave; // to remember last directory and file
2006-03-28 06:10:52 +00:00
// ------------------------------------------------------------------------
// Is called by slotShowLastMsg(), by slotShowLastNetlist() and from the
// component edit dialog.
void QucsApp::editFile(const QString& File)
{
2023-06-07 01:45:39 +03:00
if ((QucsSettings.Editor.toLower() == "qucs") || QucsSettings.Editor.isEmpty())
{
// The Editor is 'qucs' or empty, open a net document tab
if (File.isEmpty()) {
QucsApp::slotTextNew();
}
else
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
statusBar()->showMessage(tr("Opening file..."));
QFileInfo finfo(File);
if(!finfo.exists())
statusBar()->showMessage(tr("Opening aborted, file not found."), 2000);
else {
gotoPage(File);
lastDirOpenSave = File; // remember last directory and file
statusBar()->showMessage(tr("Ready."));
}
}
}
else
{
// use an external editor
QString prog;
QStringList args;
if (QucsSettings.Editor.toLower().contains("qucsedit")) {
2024-07-04 10:42:54 +03:00
#if defined(_WIN32) || defined(__MINGW32__)
prog = QUCS_NAME"edit.exe";
#elif __APPLE__
prog = "qucsedit.app/Contents/MacOS/qucsedit";
#else
prog = "qucsedit";
#endif
QFileInfo editor(QucsSettings.QucsatorDir + prog);
prog = QDir::toNativeSeparators(editor.canonicalFilePath());
}
else { // user defined editor
QFileInfo editor(QucsSettings.Editor);
prog = QDir::toNativeSeparators(editor.canonicalFilePath());
}
if (!File.isEmpty()) {
args << File;
}
QProcess *QucsEditor = new QProcess();
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
env.insert("PATH", env.value("PATH") );
QucsEditor->setProcessEnvironment(env);
qDebug() << "Command: " << prog << args.join(" ");
QucsEditor->start(prog, args);
if( !QucsEditor->waitForStarted(1000) ) {
QMessageBox::critical(this, tr("Error"), tr("Cannot start text editor! \n\n%1").arg(prog));
delete QucsEditor;
return;
}
qDebug() << QucsEditor->readAllStandardError();
// to kill it before qucs ends
connect(this, SIGNAL(signalKillEmAll()), QucsEditor, SLOT(kill()));
}
2006-03-28 06:10:52 +00:00
}
// ------------------------------------------------------------------------
// Is called to show the output messages of the last simulation.
void QucsApp::slotShowLastMsg()
{
editFile(QucsSettings.tempFilesDir.filePath("log.txt"));
2006-03-28 06:10:52 +00:00
}
// ------------------------------------------------------------------------
// Is called to show the netlist of the last simulation.
void QucsApp::slotShowLastNetlist()
{
2022-02-19 22:21:20 +01:00
QStringList netlists;
QStringList sim_lst;
QWidget *w = DocumentTab->currentWidget();
2023-06-05 23:52:53 +03:00
if (QucsSettings.DefaultSimulator == spicecompat::simXyce) {
2022-02-20 11:30:02 +01:00
if (isTextDocument(w)) {
QMessageBox::information(this, tr("Show netlist"),
tr("Not a schematic tab!"));
return;
} else {
Schematic *sch = (Schematic *) w;
Xyce *xyce = new Xyce(sch,this);
xyce->determineUsedSimulations(&sim_lst);
delete xyce;
}
2022-02-19 22:21:20 +01:00
}
switch (QucsSettings.DefaultSimulator) {
case spicecompat::simQucsator :
netlists.append(QucsSettings.tempFilesDir.filePath("netlist.txt"));
2022-02-19 22:21:20 +01:00
break;
case spicecompat::simNgspice :
case spicecompat::simSpiceOpus :
netlists.append(QDir::toNativeSeparators(QucsSettings.S4Qworkdir
+ "/spice4qucs.cir"));
break;
2023-06-05 23:52:53 +03:00
case spicecompat::simXyce: // Xyce generates one netlist for every simulation
2022-02-19 22:21:20 +01:00
for(const auto &sim : sim_lst) {
netlists.append(QDir::toNativeSeparators(QucsSettings.S4Qworkdir
+ "/spice4qucs."
+ sim + ".cir"));
}
break;
default: break;
}
if (!isTextDocument(w)) {
Schematic *sch = (Schematic *) w;
if (sch->isDigitalCircuit()) {
netlists.clear();
netlists.append(QucsSettings.tempFilesDir.filePath("netlist.txt"));
}
}
2022-02-19 22:21:20 +01:00
for(const auto &netlist: netlists) {
editFile(netlist);
}
2006-03-28 06:10:52 +00:00
}
// ------------------------------------------------------------------------
// Is called to start the text editor.
void QucsApp::slotCallEditor()
{
editFile(QString(""));
}
// ------------------------------------------------------------------------
// Is called to start the filter synthesis program.
void QucsApp::slotCallFilter()
{
launchTool(QUCS_NAME "filter", "filter synthesis");
2006-03-28 06:10:52 +00:00
}
void QucsApp::slotCallActiveFilter()
{
launchTool(QUCS_NAME "activefilter", "active filter synthesis");
}
2006-03-28 06:10:52 +00:00
// ------------------------------------------------------------------------
// Is called to start the transmission line calculation program.
void QucsApp::slotCallLine()
{
2022-10-30 17:34:39 +03:00
launchTool(QUCS_NAME "trans", "line calculation",QStringList());
2006-03-28 06:10:52 +00:00
}
// --------------------------------------------------------------
// Is called to show a dialog for creating matching circuits.
void QucsApp::slotCallMatch()
{
MatchDialog *d = new MatchDialog(this);
d->exec();
}
2006-07-24 06:12:23 +00:00
// ------------------------------------------------------------------------
// Is called to start the attenuator calculation program.
void QucsApp::slotCallAtt()
{
2022-10-30 17:34:39 +03:00
launchTool(QUCS_NAME "attenuator", "attenuator calculation",QStringList());
2006-07-24 06:12:23 +00:00
}
2022-09-25 15:26:50 +03:00
void QucsApp::slotCallPwrComb()
{
2022-10-30 17:34:39 +03:00
launchTool(QUCS_NAME "powercombining", "power combining calculation",QStringList());
2022-09-25 15:26:50 +03:00
}
S-parameter viewer - Initial development (squashed) This commit contains the first draft of the user interface Read Touchstone files It was implemented a basic function to read Touchstone files. It can only read Touchstone files up to 4 ports and only S-parameter data. Please see "Touchstone Specification, Version 2.1, ratified January 26 2024 by the IBIS Open Forum": https://ibis.org/touchstone_ver2.1/ Add QScrollArea widgets to the files and traces lists Large number of files and traces are expected, so there is a need for scrollable areas in the files and traces lists. Basic plotting structure Add default behaviour when loading one single s2p A default behavior is added. When a single s2p file is selected, the program automatically displays S21, S11 and S22 Replace the "Delete" message by a trash image The delete image was taken from here https://commons.wikimedia.org/wiki/File:Delete-button.svg This file is licensed under the Creative Commons Attribution-Share Alike 4.0 International license. You are free: to share – to copy, distribute and transmit the work to remix – to adapt the work Under the following conditions: attribution – You must give appropriate credit, provide a link to the license, and indicate if changes were made. You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use. share alike – If you remix, transform, or build upon the material, you must distribute your contributions under the same or compatible license as the original. Read Touchstone files with more than two ports Update traces combobox depending on the selected dataset If the user has loaded data with different number of ports, the traces combobox must be refreshed each time the user changes the dataset selection. Otherwise, this may cause that the user selects a non existing trace Fix style in buttons for removing files The QPushButtons were replaced by QToolButtons. With the QPushButtons the widget was too wide Fix style in buttons for removing traces QPushButton was converted into QToolButton Delete dataset and its traces when the user decides to remove a file Remove files and traces Rework on the logic on how to remove datasets and traces After removing a file, remove the associated widgets Remove trace from Chart Update file widget position in the grid after removing Function handler for changing the color of a trace Change linestyle depending on the combo selection Set initial color of the color pickers Added a spinbox control to control the trace width It was added a spinbox that controls the width of the traces displayed. This is very convenient when a bunch of traces are being displayed and the user wants to highlight one of them easily Added function handler for controlling the x-axis A handler function was added to control the x-axis settings as the user changes the minimum, maximum or the tick interval Update traces when changing the axis settings Fix trace plotting refresh Fix frequency limits when loading a GHz range file Dockable widgets Autoadjust y-axis settings Automatically add K, delta, mu_s, m_p, MAG and MSG traces in S2P files When a Touchtone file has two ports, the stability metrics are automatically computed and added to the dataset Add marker table feature It was added a new dock consisting on a marker table and some widgets for its management Add dot marker and vertical lines in the QChart Make case insensitive the frequency scale Files were found were the frequency scale is all in capital letters. This creates a problem when reading the spar data. This commit fixes this by putting the frequency scale in lower case Auto adjust x-axis when changing the units Put x_div values as a ComboBox It makes no much sense in having a decimal spinbox for defining the tick interval. It leads to decimal ticks. It's better to have a closed list of possible values y axis tick in combobox Fix vertical line markers Fix bad "About Qt" connection The "About Qt" message was not properly connected. As a consequence, when the user went to "Help-> About Qt..." nothing showed up. This commit is intended to fix this by connecting the menu with the handler as it's done in the filter design tool Link S-parameter viewer to Qucs-S Add Re{Zin}, Im{Zin} traces to s1p and s2p files Hide y-axis units It makes no much sense for now to have it since it may happen that the y-axis represent dB, Ohm or simply its unitless (e.g. K, mu, ...) Fix segfault when removing one single file In previous commits, it was observed a segfault when removing one single s-par file. This happened because the program was freeing widgets already freed. This situation is avoided by ordering the list of widgets to remove Autoadjust y-axis Remove widgets for marker placement They are actually not needed. The SpinBox and the combo with units just add clutter. The user can set the marker freq once added Update x-axis limits after removing file Increase maximum x-ticks Get suffix using Qt method This is more robust than the previous approach Fix frequency scale in markers Enable drag and drop to open files Fix segfault when removing file Readjust frequency limits when dataset has no traces Fix read touchstone Files were found whose header contains no ! Fix initial marker step Fix autoscale y-axis Prevent docks from closing It makes no sense the user can close the docks Solve infinite loop when fmax=3000 [unit] Implemented button for removing all files on a row Implement button for removing all markers on a row
2024-09-08 08:33:53 +02:00
void QucsApp::slotCallSPAR_Viewer()
{
launchTool(QUCS_NAME "spar-viewer", "s-parameter viewer",QStringList());
}
/*!
* \brief launch an external application passing arguments
*
* \param prog executable program name (will be transformed depending on OS)
* \param progDesc program description string (used for error messages)
* \param args arguments to pass to the executable
*/
2022-02-24 22:30:33 +01:00
void QucsApp::launchTool(const QString& prog, const QString& progDesc, const QStringList &args,
bool qucs_tool)
{
QProcess *tool = new QProcess();
QString tooldir;
if (qucs_tool) tooldir = QucsSettings.QucsatorDir;
else tooldir = QucsSettings.BinDir;
2024-07-04 10:42:54 +03:00
#if defined(_WIN32) || defined(__MINGW32__)
2022-02-24 22:30:33 +01:00
QString cmd = QDir::toNativeSeparators("\""+tooldir + prog + ".exe\"");
#elif __APPLE__
2022-02-24 22:30:33 +01:00
QString cmd = QDir::toNativeSeparators(tooldir + prog + ".app/Contents/MacOS/" + prog);
#else
2022-02-24 22:30:33 +01:00
QString cmd = QDir::toNativeSeparators(tooldir + prog);
#endif
tool->setWorkingDirectory(tooldir);
qDebug() << "Command :" << cmd;
2022-02-24 22:30:33 +01:00
tool->start(cmd,args);
if(!tool->waitForStarted(1000) ) {
QMessageBox::critical(this, tr("Error"),
tr("Cannot start %1 program! \n\n(%2)").arg(progDesc, cmd));
delete tool;
return;
}
2006-07-24 06:12:23 +00:00
// to kill the application first before qucs finishes exiting
connect(this, SIGNAL(signalKillEmAll()), tool, SLOT(kill()));
}
2024-03-20 09:26:11 +03:00
void QucsApp::slotCallRFLayout()
{
QString input_file, netlist_file, odir;
if (!isTextDocument(DocumentTab->currentWidget())) {
Schematic *sch = (Schematic*)DocumentTab->currentWidget();
if(sch->fileSuffix() == "dpl") {
QMessageBox::critical(this,tr("Error"),
tr("Layouting of display pages is not supported!"));
return;
}
input_file = sch->DocName;
QFileInfo inf(sch->DocName);
odir = inf.absolutePath();
netlist_file = inf.absolutePath() + QDir::separator()
+ inf.baseName() + ".net";
QFile f(netlist_file);
if (!f.open(QIODevice::WriteOnly)) {
QMessageBox::critical(this, tr("Error"), tr("Cannot write netlist!"));
return;
}
QTextStream stream(&f);
QStringList Collect;
QPlainTextEdit *ErrText = new QPlainTextEdit(); //dummy
int pNum = sch->prepareNetlist(stream, Collect, ErrText);
if (!sch->isAnalog) {
QMessageBox::critical(this, tr("Error"), tr("Digital schematic not supported!"));
return;
}
stream << '\n';
sch->createNetlist(stream, pNum);
f.close();
} else {
QMessageBox::critical(this,tr("Error"),
tr("Layouting of text documents is not supported!"));
return;
}
2024-03-20 09:26:11 +03:00
QProcess *tool = new QProcess();
QStringList args;
args.append("-G");
args.append("-i");
args.append(input_file);
args.append("-n");
args.append(netlist_file);
args.append("-o");
args.append(odir);
tool->start(QucsSettings.RFLayoutExecutable,args);
if(!tool->waitForStarted(1000) ) {
QMessageBox::critical(this, tr("Error"),
tr("Cannot start Qucs-RFLayout: \n%1")
.arg(QucsSettings.RFLayoutExecutable));
delete tool;
return;
}
connect(this, SIGNAL(signalKillEmAll()), tool, SLOT(kill()));
2024-03-20 09:26:11 +03:00
}
2006-03-28 06:10:52 +00:00
// --------------------------------------------------------------
void QucsApp::slotHelpIndex()
{
2017-10-30 17:13:17 +03:00
QDesktopServices::openUrl(QUrl("https://qucs-s-help.readthedocs.io/"));
}
void QucsApp::slotHelpQucsIndex()
{
QDesktopServices::openUrl(QUrl("https://qucs-help.readthedocs.io/"));
2006-03-28 06:10:52 +00:00
}
// --------------------------------------------------------------
void QucsApp::slotGettingStarted()
{
QDesktopServices::openUrl(QUrl("https://ra3xdh.github.io/pdf/qucs_s_tutorial.pdf"));
2006-03-28 06:10:52 +00:00
}
2006-05-05 06:00:05 +00:00
// ---------------------------------------------------------------------
2015-01-10 23:15:41 +08:00
// Is called when the find action is triggered.
2006-05-05 06:00:05 +00:00
void QucsApp::slotEditFind()
{
SearchDia->initSearch(DocumentTab->currentWidget(),
((TextDoc *)DocumentTab->currentWidget())->textCursor().selectedText(), false);
2006-05-05 06:00:05 +00:00
}
2006-03-28 06:10:52 +00:00
// --------------------------------------------------------------
void QucsApp::slotChangeProps()
{
QWidget *Doc = DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
2006-05-05 06:00:05 +00:00
((TextDoc*)Doc)->viewport()->setFocus();
2014-11-25 03:59:14 +08:00
SearchDia->initSearch(Doc,
((TextDoc *)Doc)->textCursor().selectedText(), true);
2006-05-05 06:00:05 +00:00
}
else {
ChangeDialog *d = new ChangeDialog((Schematic*)Doc);
if(d->exec() == QDialog::Accepted) {
((Schematic*)Doc)->setChanged(true, true);
((Schematic*)Doc)->viewport()->update();
}
2006-03-28 06:10:52 +00:00
}
}
// --------------------------------------------------------------
void QucsApp::slotAddToProject()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-03-28 06:10:52 +00:00
if(ProjName.isEmpty()) {
QMessageBox::critical(this, tr("Error"), tr("No project open!"));
return;
}
2013-03-15 18:43:02 +01:00
QStringList List = QFileDialog::getOpenFileNames(this, tr("Select files to copy"),
lastDir.isEmpty() ? QString(".") : lastDir, QucsFileFilter);
2006-03-28 06:10:52 +00:00
if(List.isEmpty()) {
statusBar()->showMessage(tr("No files copied."), 2000);
2006-03-28 06:10:52 +00:00
return;
}
char *Buffer = (char*)malloc(0x10000);
if(!Buffer) return; // should never happen
QStringList FileList = List; // make a copy as recommended by Qt
QStringList::Iterator it = FileList.begin();
2006-07-03 06:02:08 +00:00
QFileInfo Info(*it);
lastDir = Info.absolutePath(); // remember last directory
2006-03-28 06:10:52 +00:00
// copy all files to project directory
int Num;
QFile origFile, destFile;
while(it != FileList.end()) {
Info.setFile(*it);
origFile.setFileName(*it);
destFile.setFileName(QucsSettings.QucsWorkDir.absolutePath() +
2006-03-28 06:10:52 +00:00
QDir::separator() + Info.fileName());
if(!origFile.open(QIODevice::ReadOnly)) {
2006-03-28 06:10:52 +00:00
QMessageBox::critical(this, tr("Error"), tr("Cannot open \"%1\" !").arg(*it));
it++;
continue;
}
if(destFile.exists())
if(QMessageBox::information(this, tr("Overwrite"),
2023-06-13 17:48:27 +03:00
tr("File \"%1\" already exists.\nOverwrite ?").arg(*it),
QMessageBox::Yes|QMessageBox::No)
2006-03-28 06:10:52 +00:00
!= QMessageBox::Yes) {
origFile.close();
it++;
continue;
}
if(!destFile.open(QIODevice::WriteOnly)) {
2006-03-28 06:10:52 +00:00
QMessageBox::critical(this, tr("Error"), tr("Cannot create \"%1\" !").arg(*it));
origFile.close();
it++;
continue;
}
// copy data
do {
Num = origFile.read(Buffer, 0x10000);
2006-03-28 06:10:52 +00:00
if(Num < 0) {
QMessageBox::critical(this, tr("Error"), tr("Cannot read \"%1\" !").arg(*it));
break;
}
Num = destFile.write(Buffer, Num);
2006-03-28 06:10:52 +00:00
if(Num < 0) {
QMessageBox::critical(this, tr("Error"), tr("Cannot write \"%1\" !").arg(*it));
2006-04-18 06:03:52 +00:00
break;
2006-03-28 06:10:52 +00:00
}
} while(Num == 0x10000);
origFile.close();
destFile.close();
it++;
}
2006-03-28 06:10:52 +00:00
free(Buffer);
slotUpdateTreeview();
statusBar()->showMessage(tr("Ready."));
2006-03-28 06:10:52 +00:00
}
// -----------------------------------------------------------
void QucsApp::slotCursorLeft(bool left)
2006-03-28 06:10:52 +00:00
{
int sign = 1;
if(left){
sign = -1;
}
2006-03-28 06:10:52 +00:00
if(!editText->isHidden()) return; // for edit of component property ?
Q3PtrList<Element> movingElements;
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-05-22 06:01:55 +00:00
int markerCount = Doc->copySelectedElements(&movingElements);
if((movingElements.count() - markerCount) < 1) {
if(markerCount > 0) { // only move marker if nothing else selected
Doc->markerLeftRight(left, &movingElements);
} else if(left) {
2024-01-06 18:11:27 +03:00
Doc->scrollLeft(Doc->horizontalScrollBar()->singleStep());
}else{ // right
Doc->scrollRight(Doc->horizontalScrollBar()->singleStep());
2006-05-22 06:01:55 +00:00
}
2006-03-28 06:10:52 +00:00
Doc->viewport()->update();
return;
} else { // random selection. move all of them
view->moveElements(&movingElements, sign*Doc->GridX, 0);
view->MAx3 = 1; // sign for moved elements
2024-01-30 10:56:23 +03:00
view->endElementMoving(Doc, &movingElements);
2006-03-28 06:10:52 +00:00
}
}
// -----------------------------------------------------------
void QucsApp::slotCursorUp(bool up)
2006-03-28 06:10:52 +00:00
{
if(editText->isHidden()) { // for edit of component property ?
}else if(up){
2006-03-28 06:10:52 +00:00
if(view->MAx3 == 0) return; // edit component namen ?
Component *pc = (Component*)view->focusElement;
Property *pp = pc->Props.at(view->MAx3-1); // current property
int Begin = pp->Description.indexOf('[');
2006-03-28 06:10:52 +00:00
if(Begin < 0) return; // no selection list ?
int End = pp->Description.indexOf(editText->text(), Begin); // current
2006-03-28 06:10:52 +00:00
if(End < 0) return; // should never happen
End = pp->Description.lastIndexOf(',', End);
2006-03-28 06:10:52 +00:00
if(End < Begin) return; // was first item ?
End--;
int Pos = pp->Description.lastIndexOf(',', End);
2006-03-28 06:10:52 +00:00
if(Pos < Begin) Pos = Begin; // is first item ?
Pos++;
if(pp->Description.at(Pos) == ' ') Pos++; // remove leading space
editText->setText(pp->Description.mid(Pos, End-Pos+1));
editText->selectAll();
return;
}else{ // down
2006-03-28 06:10:52 +00:00
if(view->MAx3 == 0) return; // edit component namen ?
Component *pc = (Component*)view->focusElement;
Property *pp = pc->Props.at(view->MAx3-1); // current property
int Pos = pp->Description.indexOf('[');
2006-03-28 06:10:52 +00:00
if(Pos < 0) return; // no selection list ?
Pos = pp->Description.indexOf(editText->text(), Pos); // current list item
2006-03-28 06:10:52 +00:00
if(Pos < 0) return; // should never happen
Pos = pp->Description.indexOf(',', Pos);
2006-03-28 06:10:52 +00:00
if(Pos < 0) return; // was last item ?
Pos++;
if(pp->Description.at(Pos) == ' ') Pos++; // remove leading space
int End = pp->Description.indexOf(',', Pos);
2006-03-28 06:10:52 +00:00
if(End < 0) { // is last item ?
End = pp->Description.indexOf(']', Pos);
2006-03-28 06:10:52 +00:00
if(End < 0) return; // should never happen
}
editText->setText(pp->Description.mid(Pos, End-Pos));
editText->selectAll();
return;
}
Q3PtrList<Element> movingElements;
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
2006-05-22 06:01:55 +00:00
int markerCount = Doc->copySelectedElements(&movingElements);
if((movingElements.count() - markerCount) < 1) { // all selections are markers
2006-05-22 06:01:55 +00:00
if(markerCount > 0) { // only move marker if nothing else selected
Doc->markerUpDown(up, &movingElements);
} else if(up) { // nothing selected at all
2024-01-06 19:05:24 +03:00
Doc->scrollUp(Doc->verticalScrollBar()->singleStep());
} else { // down
2024-01-06 18:59:07 +03:00
Doc->scrollDown(Doc->verticalScrollBar()->singleStep());
2006-05-22 06:01:55 +00:00
}
2006-03-28 06:10:52 +00:00
Doc->viewport()->update();
return;
}else{ // some random selection, put it back
view->moveElements(&movingElements, 0, ((up)?-1:1) * Doc->GridY);
view->MAx3 = 1; // sign for moved elements
2024-01-30 10:56:23 +03:00
view->endElementMoving(Doc, &movingElements);
2006-03-28 06:10:52 +00:00
}
}
// -----------------------------------------------------------
// Is called if user clicked on component text of if return is
// pressed in the component text QLineEdit.
2006-04-18 06:03:52 +00:00
// In "view->MAx3" is the number of the current property.
2006-03-28 06:10:52 +00:00
void QucsApp::slotApplyCompText()
{
QFont f = QucsSettings.font;
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
f.setPointSizeF( Doc->Scale * float(f.pointSize()) );
2006-03-28 06:10:52 +00:00
editText->setFont(f);
Component *const component = dynamic_cast<Component*>(view->focusElement);
if(!component) return; // should never happen
view->MAx1 = component->cx + component->tx;
view->MAy1 = component->cy + component->ty;
// Here is a bit of *magic* and implicit coupling: the value of view->MAx3
// comes from Component::getTextSelected, and it's equal 0 when component
// name is clicked, N + 1 when Nth component property is clicked.
const int component_text_index = view->MAx3;
const bool is_name = component_text_index == 0;
Property *const component_property = !is_name
? component->Props.at(component_text_index - 1)
: nullptr;
if (editText->isVisible()) { // is called the first time ?
if (is_name) {
const auto new_name{editText->text()};
if (!new_name.isEmpty() && component->Name != new_name) {
// TODO: rewrite with std::none_of after replacing Q3PtrList
// with modern container
bool is_unique = true;
for (auto* other : *Doc->Components) {
if (other->Name == new_name) {
is_unique = false;
break;
2006-03-28 06:10:52 +00:00
}
}
if (is_unique) {
component->Name = new_name;
Doc->setChanged(true, true); // only one undo state
}
}
2006-03-28 06:10:52 +00:00
}
else if (component_property) { // property was applied
if (component_property->Value != editText->text()) {
component_property->Value = editText->text();
Doc->recreateComponent(component); // because of "Num" and schematic symbol
Doc->setChanged(true, true); // only one undo state
2006-03-28 06:10:52 +00:00
}
}
}
2006-05-24 05:59:39 +00:00
const QString s = is_name
? component->Name
: component_property->Value;
2006-03-28 06:10:52 +00:00
editText->setReadOnly(false);
QPoint editTextTopLeft;
if (component_property) { // is it a property ?
editTextTopLeft = Doc->modelToViewport(QPoint{component->cx, component->cy} + component_property->boundingRect().topLeft());
editTextTopLeft.rx() += editText->fontMetrics().boundingRect(component_property->Name + "=" + '\u0020').width();
if(component_property->Description.indexOf('[') >= 0) // is selection list ?
2006-03-28 06:10:52 +00:00
editText->setReadOnly(true);
Expr_CompProp.setPattern("[^\"]*");
2006-03-28 06:10:52 +00:00
}
else { // it is the component name
2006-03-28 06:10:52 +00:00
Expr_CompProp.setPattern("[\\w_]+");
editTextTopLeft = Doc->modelToViewport(QPoint{component->cx + component->tx, component->cy + component->ty});
}
{
auto size = editText->fontMetrics().boundingRect(s ).size();
size.rwidth() += editText->fontMetrics().averageCharWidth();
editText->setFixedSize(size);
}
view->MAx2 = editTextTopLeft.x();
view->MAy2 = editTextTopLeft.y();
2023-01-17 13:27:12 +03:00
Val_CompProp.setRegularExpression(Expr_CompProp);
2006-03-28 06:10:52 +00:00
editText->setValidator(&Val_CompProp);
editText->setText(s);
editText->setStyleSheet("color: black; background-color: " + QucsSettings.BGColor.name());
2006-03-28 06:10:52 +00:00
editText->setFocus();
editText->selectAll();
editText->setParent(Doc->viewport());
editText->move(view->MAx2, view->MAy2);
editText->show();
//editText->reparent(Doc->viewport(), 0, QPoint(view->MAx2, view->MAy2), true);
2006-03-28 06:10:52 +00:00
}
2006-05-08 06:13:04 +00:00
// -----------------------------------------------------------
// Is called if the text of the property edit changed, to match
// the width of the edit field.
void QucsApp::slotResizePropEdit(const QString& t)
{
2022-02-24 22:30:33 +01:00
editText->resize(editText->fontMetrics().boundingRect(t).width()+4,
2006-05-18 06:08:50 +00:00
editText->fontMetrics().lineSpacing());
2006-05-08 06:13:04 +00:00
}
2006-06-06 06:14:17 +00:00
// -----------------------------------------------------------
void QucsApp::slotCreateLib()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-07-03 06:02:08 +00:00
2006-06-06 06:14:17 +00:00
if(ProjName.isEmpty()) {
QMessageBox::critical(this, tr("Error"), tr("Please open project with subcircuits!"));
return;
}
LibraryDialog *d = new LibraryDialog(this);
d->fillSchematicList(Content->exportSchematic());
d->exec();
2006-06-06 06:14:17 +00:00
}
2006-07-03 06:02:08 +00:00
// -----------------------------------------------------------
void QucsApp::slotImportData()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-07-03 06:02:08 +00:00
QString import_dir = ".";
if (!ProjName.isEmpty()) {
import_dir = QucsSettings.QucsWorkDir.absolutePath();
} else {
QString dname;
if (isTextDocument(DocumentTab->currentWidget())) {
TextDoc *doc = (TextDoc *)DocumentTab->currentWidget();
dname = doc->DocName;
} else {
Schematic *doc = (Schematic *)DocumentTab->currentWidget();
dname = doc->DocName;
}
QFileInfo inf(dname);
if (inf.exists()) {
import_dir = inf.absolutePath();
}
}
2006-07-03 06:02:08 +00:00
ImportDialog *d = new ImportDialog(this);
d->setImportDir(import_dir);
2006-07-03 06:02:08 +00:00
if(d->exec() == QDialog::Accepted)
slotUpdateTreeview();
2006-07-03 06:02:08 +00:00
}
// -----------------------------------------------------------
void QucsApp::slotExportGraphAsCsv()
{
2015-01-28 00:14:17 +08:00
slotHideEdit(); // disable text edit of component property
2006-07-03 06:02:08 +00:00
for(;;) {
if(view->focusElement)
if(view->focusElement->Type == isGraph)
break;
QMessageBox::critical(this, tr("Error"), tr("Please select a diagram graph!"));
return;
}
2013-03-15 18:43:02 +01:00
/*QString s = Q3FileDialog::getSaveFileName(
2006-07-03 06:02:08 +00:00
lastDir.isEmpty() ? QString(".") : lastDir,
tr("CSV file")+" (*.csv);;" + tr("Any File")+" (*)",
this, 0, tr("Enter an Output File Name"));
2013-03-15 18:43:02 +01:00
*/
QString s = QFileDialog::getSaveFileName(this, tr("Enter an Output File Name"),
2013-03-15 18:43:02 +01:00
lastDir.isEmpty() ? QString(".") : lastDir, tr("CSV file")+" (*.csv);;" + tr("Any File")+" (*)");
2006-07-03 06:02:08 +00:00
if(s.isEmpty())
return;
QFileInfo Info(s);
lastDir = Info.absolutePath(); // remember last directory
if(Info.suffix().isEmpty())
2006-07-03 06:02:08 +00:00
s += ".csv";
QFile File(s);
if(File.exists())
if(QMessageBox::information(this, tr("Info"),
tr("Output file already exists!")+"\n"+tr("Overwrite it?"),
2023-06-13 17:48:27 +03:00
QMessageBox::Yes|QMessageBox::No) == QMessageBox::No)
2006-07-03 06:02:08 +00:00
return;
if(!File.open(QIODevice::WriteOnly)) {
2006-07-03 06:02:08 +00:00
QMessageBox::critical(this, QObject::tr("Error"),
QObject::tr("Cannot create output file!"));
return;
}
2013-03-15 18:43:02 +01:00
QTextStream Stream(&File);
2006-07-03 06:02:08 +00:00
DataX const *pD;
2006-07-03 06:02:08 +00:00
Graph *g = (Graph*)view->focusElement;
// First output the names of independent and dependent variables.
for(unsigned ii=0; (pD=g->axis(ii)); ++ii)
2006-07-03 06:02:08 +00:00
Stream << '\"' << pD->Var << "\";";
Stream << "\"r " << g->Var << "\";\"i " << g->Var << "\"\n";
2006-07-03 06:02:08 +00:00
int n, m;
double *py = g->cPointsY;
int Count = g->countY * g->axis(0)->count;
2006-07-03 06:02:08 +00:00
for(n = 0; n < Count; n++) {
m = n;
for(unsigned ii=0; (pD=g->axis(ii)); ++ii) {
2006-07-03 06:02:08 +00:00
Stream << *(pD->Points + m%pD->count) << ';';
m /= pD->count;
}
Stream << *(py) << ';' << *(py+1) << '\n';
py += 2;
2006-07-03 06:02:08 +00:00
}
File.close();
}
2015-01-11 22:36:06 +08:00
void QucsApp::slotOpenRecent()
{
2015-01-11 22:36:06 +08:00
QAction *action = qobject_cast<QAction *>(sender());
if (action) {
gotoPage(action->data().toString());
updateRecentFilesList(action->data().toString());
}
}
void QucsApp::slotUpdateRecentFiles()
{
2015-01-11 22:36:06 +08:00
QMutableStringListIterator it(QucsSettings.RecentDocs);
while(it.hasNext()) {
if (!QFile::exists(it.next())) {
it.remove();
}
2015-01-11 22:36:06 +08:00
}
2015-03-06 17:17:41 +03:00
2015-01-11 22:36:06 +08:00
for (int i = 0; i < MaxRecentFiles; ++i) {
if (i < QucsSettings.RecentDocs.size()) {
fileRecentAction[i]->setText(QucsSettings.RecentDocs[i]);
fileRecentAction[i]->setData(QucsSettings.RecentDocs[i]);
fileRecentAction[i]->setVisible(true);
} else {
fileRecentAction[i]->setVisible(false);
}
2015-01-11 22:36:06 +08:00
}
}
void QucsApp::slotClearRecentFiles()
{
2015-01-11 22:36:06 +08:00
QucsSettings.RecentDocs.clear();
slotUpdateRecentFiles();
}
/*!
* \brief QucsApp::slotLoadModule launches the dialog to select dynamic modueles
*/
void QucsApp::slotLoadModule()
{
qDebug() << "slotLoadModule";
2014-02-26 21:29:09 +01:00
2014-03-02 22:50:12 +01:00
LoadDialog *ld = new LoadDialog(this);
ld->setApp(this);
// fech list of _symbol.json
// \todo fetch timestamp of VA, JSON, if VA newer, need to reload.
2014-03-02 22:50:12 +01:00
QDir projDir = QucsSettings.QucsWorkDir.absolutePath();
2014-03-02 22:50:12 +01:00
QStringList files;
QString fileSuffix = "*_symbol.json";
files = projDir.entryList(QStringList(fileSuffix),
2014-03-02 22:50:12 +01:00
QDir::Files | QDir::NoSymLinks);
// no JSON files or no a project?
if (!files.size()){
QMessageBox::critical(this, tr("Error"),
tr("Symbol files not found in: %1\n\n"
"Is the project open?\n"
"Have you saved the Verilog-A symbols?")
.arg(QString(projDir.absolutePath())));
return;
}
// initialize dialog
// pass list of potential symbol files
ld->symbolFiles << files;
ld->projDir = projDir;
2014-03-02 22:50:12 +01:00
ld->initDialog();
// \todo check what is already loaded, offer skip, reload
//pass stuff to ld dialog
2014-03-02 22:50:12 +01:00
// run, let user do the selections
if (ld->exec() == QDialog::Accepted) {
Module::vaComponents = ld->selectedComponents;
2014-03-02 22:50:12 +01:00
// dialog write new bitmap into JSON
// load, unload, reload
// inform if symbol changed
// populate Module::vaComponents
// vaComponents are selected with the dialog
// dialog should populate according to checkboxes
// build vaComponents QMap
// remove all previously registered modules
QMutableHashIterator<QString, Module *> it( Module::Modules );
2014-11-03 04:01:47 +08:00
while(it.hasNext()) {
it.next();
2014-11-11 21:29:06 +01:00
if (it.value()->category == QObject::tr("verilog-a user devices")) {
it.remove();
2014-11-03 04:01:47 +08:00
}
}
if (! Module::vaComponents.isEmpty()) {
// Register whatever is in Module::vaComponents
//Module::registerDynamicComponents();
// update the combobox, set new category in view
// pick up new category 'verilog-a user components' from `Module::category`
//set new category into view
QucsApp::fillComboBox(true);
CompChoose->setCurrentIndex(CompChoose->count()-1);
slotSetCompView(CompChoose->count()-1);
// icons of dynamically registered components ready to be dragged
}
else {
// remove any previously registered icons from the listview
int foundCat = CompChoose->findText(QObject::tr("verilog-a user devices"));
if (foundCat != -1) {
CompChoose->setCurrentIndex(foundCat);
CompComps->clear();
}
}
}
2014-02-26 21:29:09 +01:00
delete ld;
2014-03-02 22:50:12 +01:00
}
2014-02-26 22:26:27 +01:00
/*!
* \brief QucsApp::slotBuildModule runs admsXml, C++ compiler to build library
*
2014-02-26 22:26:27 +01:00
* Run the va2cpp
* Run the cpp2lib
*
* TODO
* - split into two actions, elaborate and compile?
* - collect, parse and display output of make
*
*/
void QucsApp::slotBuildModule()
{
qDebug() << "slotBuildModule";
2014-02-26 22:26:27 +01:00
// reset message dock on entry
messageDock->reset();
if (QucsSettings.DefaultSimulator == spicecompat::simNgspice) {
buildWithOpenVAF();
return;
}
messageDock->builderTabs->setTabIcon(0,QPixmap());
2017-01-19 14:55:09 +03:00
messageDock->builderTabs->setTabText(0,tr("admsXml"));
messageDock->builderTabs->setTabIcon(1,QPixmap());
2017-01-19 14:55:09 +03:00
messageDock->builderTabs->setTabText(1,tr("Compiler"));
messageDock->msgDock->setWindowTitle(tr("admsXml Dock"));
QString make;
2014-03-21 05:00:42 -07:00
2024-07-04 10:42:54 +03:00
#if defined(_WIN32) || defined(__MINGW32__)
make = "mingw32-make.exe"; // must be on the path!
2014-03-21 05:00:42 -07:00
#else
make = "make"; // must be on the path!
2014-03-21 05:00:42 -07:00
#endif
QFileInfo inf(QucsSettings.Qucsator);
QString QucsatorPath = inf.path()+QDir::separator();
2014-02-26 22:26:27 +01:00
QDir prefix = QDir(QucsatorPath+"../");
QDir include = QDir(QucsatorPath+"../include/qucs-core");
2014-02-26 22:26:27 +01:00
QString workDir = QucsSettings.QucsWorkDir.absolutePath();
// need to cd into project to make sure output is dropped there?
// need to cd - into previous location?
2014-02-26 22:26:27 +01:00
QDir::setCurrent(workDir);
QProcess *builder = new QProcess();
builder->setProcessChannelMode(QProcess::MergedChannels);
2014-02-26 22:26:27 +01:00
// get current va document
QucsDoc *Doc = getDoc();
QString vaModule = Doc->fileBase(Doc->DocName);
QString admsXml = QucsSettings.AdmsXmlBinDir.canonicalPath();
2024-07-04 10:42:54 +03:00
#if defined(_WIN32) || defined(__MINGW32__)
admsXml = QDir::toNativeSeparators(admsXml+"/"+"admsXml.exe");
#else
admsXml = QDir::toNativeSeparators(admsXml+"/"+"admsXml");
#endif
2014-02-26 22:26:27 +01:00
// admsXml emits C++
QStringList Arguments;
2014-06-06 21:56:45 +02:00
Arguments << "-f" << QDir::toNativeSeparators(include.absoluteFilePath("va2cpp.makefile"))
<< QString("ADMSXML=%1").arg(admsXml)
2014-06-06 21:56:45 +02:00
<< QString("PREFIX=%1").arg(QDir::toNativeSeparators(prefix.absolutePath()))
<< QString("MODEL=%1").arg(vaModule);
2014-02-26 22:26:27 +01:00
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
env.insert("PATH", env.value("PATH") );
builder->setProcessEnvironment(env);
// prepend command to log
QString cmdString = QString("%1 %2\n").arg(make, Arguments.join(" "));
messageDock->admsOutput->appendPlainText(cmdString);
qDebug() << "Command :" << make << Arguments.join(" ");
builder->start(make, Arguments);
2014-02-26 22:26:27 +01:00
// admsXml seems to communicate all via stdout, or is it because of make?
QString vaStatus;
if (!builder->waitForFinished()) {
vaStatus = builder->errorString();
qDebug() << "Make failed:" << vaStatus;
}
2014-02-26 22:26:27 +01:00
else {
vaStatus = builder->readAll();
qDebug() << "Make stdout" << vaStatus;
2014-02-26 22:26:27 +01:00
}
//build libs
qDebug() << "\nbuild libs\n";
Arguments.clear();
2014-06-06 21:56:45 +02:00
Arguments << "-f" << QDir::toNativeSeparators(include.absoluteFilePath("cpp2lib.makefile"))
2014-06-06 23:58:24 +02:00
<< QString("PREFIX=\"%1\"").arg(QDir::toNativeSeparators(prefix.absolutePath()))
<< QString("PROJDIR=\"%1\"").arg(QDir::toNativeSeparators(workDir))
2014-03-21 05:00:42 -07:00
<< QString("MODEL=%1").arg(vaModule);
// prepend command to log
cmdString = QString("%1 %2\n").arg(make, Arguments.join(" "));
messageDock->cppOutput->appendPlainText(cmdString);
builder->start(make, Arguments);
QString cppStatus;
if (!builder->waitForFinished()) {
cppStatus = builder->errorString();
qDebug() << "Make failed:" << cppStatus;
}
else {
cppStatus = builder->readAll();
qDebug() << "Make output:" << cppStatus;
}
delete builder;
// push make output to message dock
messageDock->admsOutput->appendPlainText(vaStatus);
messageDock->cppOutput->appendPlainText(cppStatus);
// shot the message docks
messageDock->msgDock->show();
}
2015-02-01 22:20:16 +01:00
void QucsApp::buildWithOpenVAF()
{
messageDock->builderTabs->setTabIcon(0,QPixmap());
messageDock->builderTabs->setTabText(0,tr("OpenVAF"));
messageDock->msgDock->setWindowTitle(tr("OpenVAF Dock"));
QString workDir = QucsSettings.QucsWorkDir.absolutePath();
QDir::setCurrent(workDir);
QProcess *builder = new QProcess();
builder->setProcessChannelMode(QProcess::MergedChannels);
// get current va document
QucsDoc *Doc = getDoc();
QString vaModule = Doc->DocName;
QString openVAF = QucsSettings.OpenVAFExecutable;
// admsXml emits C++
QStringList Arguments;
Arguments<<vaModule;
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
env.insert("PATH", env.value("PATH") );
builder->setProcessEnvironment(env);
// prepend command to log
QString cmdString = QString("%1 %2\n").arg(openVAF, Arguments.join(" "));
messageDock->admsOutput->appendPlainText(cmdString);
qDebug() << "Command :" << openVAF << Arguments.join(" ");
builder->start(openVAF, Arguments);
// admsXml seems to communicate all via stdout, or is it because of make?
QString vaStatus;
if (!builder->waitForFinished()) {
vaStatus = builder->errorString();
qDebug() << "OpenVAF failed:" << vaStatus;
}
else {
vaStatus = builder->readAll();
qDebug() << "OpenVAF stdout" << vaStatus;
}
delete builder;
// push make output to message dock
messageDock->admsOutput->appendPlainText(vaStatus);
// shot the message docks
messageDock->msgDock->show();
}
2015-02-01 22:20:16 +01:00
// ----------------------------------------------------------
void QucsApp::slotHelpAbout()
{
AboutDialog *ad = new AboutDialog(this);
ad->exec();
}
// vim:ts=8:sw=2:noet