Files
ThirdParty-6/ParaView-5.0.1/Qt/Components/pqProxyPanel.cxx

310 lines
9.3 KiB
C++

/*=========================================================================
Program: ParaView
Module: pqProxyPanel.cxx
Copyright (c) 2005-2008 Sandia Corporation, Kitware Inc.
All rights reserved.
ParaView is a free software; you can redistribute it and/or modify it
under the terms of the ParaView license version 1.2.
See License_v1.2.txt for the full ParaView license.
A copy of this license can be obtained by contacting
Kitware Inc.
28 Corporate Drive
Clifton Park, NY 12065
USA
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
=========================================================================*/
// this include
#include "pqProxyPanel.h"
// Qt includes
#include <QApplication>
#include <QStyle>
#include <QStyleOption>
#include <QHelpEvent>
#include <QToolTip>
// VTK includes
#include "vtkCommand.h"
#include "vtkEventQtSlotConnect.h"
// ParaView Server Manager includes
#include "vtkSmartPointer.h"
#include "vtkSMProperty.h"
#include "vtkSMSourceProxy.h"
#include "vtkSMDocumentation.h"
// ParaView includes
#include "pqPropertyManager.h"
#include "pqApplicationCore.h"
#include "pqUndoStack.h"
#include "pqView.h"
//-----------------------------------------------------------------------------
class pqProxyPanel::pqImplementation
{
public:
pqImplementation(vtkSMProxy* pxy) : Proxy(pxy)
{
this->VTKConnect = vtkSmartPointer<vtkEventQtSlotConnect>::New();
this->InformationObsolete = true;
this->Selected = false;
}
~pqImplementation()
{
delete this->PropertyManager;
}
vtkSmartPointer<vtkSMProxy> Proxy;
vtkSmartPointer<vtkEventQtSlotConnect> VTKConnect;
pqPropertyManager* PropertyManager;
QPointer<pqView> View;
// Flag indicating to the best of our knowledge, the information properties
// and domains are not up-to-date since the proxy was modified since the last
// time we updated the properties and their domains. This is just a guess,
// to reduce number of updates information calls.
bool InformationObsolete;
// Indicates if the panel is currently selected.
bool Selected;
};
//-----------------------------------------------------------------------------
pqProxyPanel::pqProxyPanel(vtkSMProxy* pxy, QWidget* p) :
QWidget(p),
Implementation(new pqImplementation(pxy))
{
// Just make sure that the proxy is setup properly.
this->Implementation->Proxy->UpdateVTKObjects();
this->updateInformationAndDomains();
this->Implementation->PropertyManager = new pqPropertyManager(this);
QObject::connect(this->Implementation->PropertyManager,
SIGNAL(modified()),
this, SLOT(setModified()));
this->Implementation->VTKConnect->Connect(
this->Implementation->Proxy, vtkCommand::ModifiedEvent,
this, SLOT(proxyModifiedEvent()));
this->Implementation->VTKConnect->Connect(
this->Implementation->Proxy, vtkCommand::UpdateDataEvent,
this, SLOT(dataUpdated()));
}
//-----------------------------------------------------------------------------
pqProxyPanel::~pqProxyPanel()
{
delete this->Implementation;
}
//-----------------------------------------------------------------------------
vtkSMProxy* pqProxyPanel::proxy() const
{
return this->Implementation->Proxy;
}
//-----------------------------------------------------------------------------
pqPropertyManager* pqProxyPanel::propertyManager()
{
return this->Implementation->PropertyManager;
}
//-----------------------------------------------------------------------------
pqView* pqProxyPanel::view() const
{
return this->Implementation->View;
}
//-----------------------------------------------------------------------------
void pqProxyPanel::dataUpdated()
{
this->Implementation->InformationObsolete = true;
if (this->Implementation->Selected)
{
this->updateInformationAndDomains();
}
}
//-----------------------------------------------------------------------------
QSize pqProxyPanel::sizeHint() const
{
// return a size hint that would reasonably fit several properties
ensurePolished();
QFontMetrics fm(font());
int h = qMax(fm.lineSpacing(), 14);
int w = fm.width('x') * 25;
QStyleOptionFrame opt;
opt.rect = rect();
opt.palette = palette();
opt.state = QStyle::State_None;
return (style()->sizeFromContents(QStyle::CT_LineEdit, &opt, QSize(w, h).
expandedTo(QApplication::globalStrut()),
this));
}
//-----------------------------------------------------------------------------
bool pqProxyPanel::selected() const
{
return this->Implementation->Selected;
}
//-----------------------------------------------------------------------------
/// accept the changes made to the properties
/// changes will be propogated down to the server manager
void pqProxyPanel::accept()
{
this->Implementation->PropertyManager->accept();
if (this->Implementation->Selected)
{
this->updateInformationAndDomains();
}
emit this->onaccept();
}
//-----------------------------------------------------------------------------
/// reset the changes made
/// editor will query properties from the server manager
void pqProxyPanel::reset()
{
this->updateInformationAndDomains();
this->Implementation->PropertyManager->reject();
emit this->onreset();
}
//-----------------------------------------------------------------------------
void pqProxyPanel::select()
{
this->Implementation->Selected = true;
this->updateInformationAndDomains();
emit this->onselect();
}
//-----------------------------------------------------------------------------
void pqProxyPanel::deselect()
{
this->Implementation->Selected = false;
emit this->ondeselect();
}
//-----------------------------------------------------------------------------
void pqProxyPanel::setView(pqView* rm)
{
if(this->Implementation->View == rm)
{
return;
}
this->Implementation->View = rm;
emit this->viewChanged(this->Implementation->View);
}
//-----------------------------------------------------------------------------
// Called when vtkSMProxy fires vtkCommand::ModifiedEvent which implies that
// the proxy was modified. If that's the case, the information properties
// need to be updated. We mark them as obsolete so that next
// time the panel becomes active (or is accepted when it is active),
// we'll refresh the information properties and domains.
// Note thate vtkCommand::ModifiedEvent is fired by a proxy when it is
// updated or anyone upstream for it is updated.
void pqProxyPanel::proxyModifiedEvent()
{
this->Implementation->InformationObsolete = true;
}
//-----------------------------------------------------------------------------
// Update information properties and domains. Since this is not a
// particularly fast operation, we update the information and domains
// only when the panel is selected or an already active panel is
// accepted.
void pqProxyPanel::updateInformationAndDomains()
{
if (this->Implementation->InformationObsolete)
{
vtkSMSourceProxy* sp;
sp = vtkSMSourceProxy::SafeDownCast(this->Implementation->Proxy);
if(sp)
{
sp->UpdatePipelineInformation();
}
else
{
this->Implementation->Proxy->UpdatePropertyInformation();
}
this->Implementation->InformationObsolete = false;
}
}
//-----------------------------------------------------------------------------
void pqProxyPanel::setModified()
{
emit this->modified();
}
bool pqProxyPanel::event(QEvent* e)
{
bool ret = QWidget::event(e);
// show tooltips for any server manager property
// doing it this way does not depend on the panel being created
// with tooltips set (auto panel or not)
// if a custom panel doesn't name its widgets to match the SM property,
// no tooltip will be shown
if(!e->isAccepted() && e->type() == QEvent::ToolTip)
{
QHelpEvent* he = static_cast<QHelpEvent*>(e);
// find the sm property this mouse is over
QWidget* w = QApplication::widgetAt(he->globalPos());
if(this->isAncestorOf(w))
{
vtkSMProperty* smProperty = NULL;
for(; !smProperty && w != this; w = w->parentWidget())
{
QString name = w->objectName();
int trimIndex = name.lastIndexOf(QRegExp("_[0-9]*$"));
if(trimIndex != -1)
{
name = name.left(trimIndex);
}
smProperty = this->Implementation->Proxy->GetProperty(name.toLatin1().data());
}
if(smProperty)
{
vtkSMDocumentation* doc = smProperty->GetDocumentation();
if(doc)
{
QToolTip::showText(he->globalPos(), QString("<p>%1</p>").arg(doc->GetDescription()), this);
ret = true;
e->setAccepted(true);
}
}
}
}
return ret;
}