/*========================================================================= medInria Copyright (c) INRIA 2013 - 2018. All rights reserved. See LICENSE.txt for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. =========================================================================*/ #include #include #include #include #include "vtkFiberDataSetManager.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include class medVtkFibersDataInteractorPrivate { public: template void setROI (medAbstractData *data); medAbstractData *data; medAbstractImageView *view; medAbstractData *projectionData; vtkSmartPointer manager; vtkSmartPointer roiManager; vtkSmartPointer dataset; vtkImageView2D *view2d; vtkImageView3D *view3d; vtkRenderWindow *render; vtkSmartPointer actor; vtkSmartPointer opacityProperty; QWidget *toolboxWidget; QPointer bundleToolboxWidget; QMap meanLengthList; QMap minLengthList; QMap maxLengthList; QMap varLengthList; QList parameters; medStringListParameterL *colorFiberParameter; medStringListParameterL *LutFiberParameter; medBoolParameterL *alphaTransparencyParameter; medDoubleParameterL *minIntensityParameter; medDoubleParameterL *maxIntensityParameter; medBoolGroupParameterL *shapesDisplayedGroupParameter; medBoolParameterL *displayPolylinesParameter; medBoolParameterL *displayRibbonsParameter; medBoolParameterL *displayTubesParameter; medIntParameterL *radiusParameter; medDropSite *dropOrOpenRoi; QComboBox *roiComboBox; QMap roiLabels; QUuid roiImportUuid; medBoolParameterL *andParameter; medBoolParameterL *notParameter; medBoolParameterL *nullParameter; medTriggerParameterL *saveParameter; medTriggerParameterL *validateParameter; medTriggerParameterL *resetParameter; medBoolParameterL *addParameter; medTriggerParameterL *tagParameter; medBoolGroupParameterL *bundleOperationGroupParameter; medDoubleParameterL* opacityParam; medBoolParameterL *showAllBundleParameter; medBoolParameterL *showBundleBox; QStandardItemModel *bundlingModel; QTreeView *bundlingList; medIntParameterL *slicingParameter; vtkSmartPointer spoDefaultLut; std::vector< std::vector > oScalarRangeVect; QWidget * poLutWidget; }; template void medVtkFibersDataInteractorPrivate::setROI (medAbstractData *data) { if (!data) return; typedef itk::Image InputROIType; typedef itk::Image ROIType; typename InputROIType::Pointer tmpROIImage = dynamic_cast(static_cast(data->data())); if (tmpROIImage.IsNull()) return; typedef itk::CastImageFilter CastFilterType; typename CastFilterType::Pointer castFilter = CastFilterType::New(); castFilter->SetInput(tmpROIImage); castFilter->Update(); typename ROIType::Pointer roiImage = castFilter->GetOutput(); roiImage->DisconnectPipeline(); QList labels; typedef itk::ImageRegionConstIterator ROIITeratorType; ROIITeratorType roiItr(roiImage,roiImage->GetLargestPossibleRegion()); while (!roiItr.IsAtEnd()) { if ((roiItr.Get() != 0)&&(!labels.contains(roiItr.Get()))) labels.append(roiItr.Get()); ++roiItr; } typename itk::ImageToVTKImageFilter::Pointer converter = itk::ImageToVTKImageFilter::New(); converter->SetReferenceCount(2); converter->SetInput(roiImage); converter->Update(); typename ROIType::DirectionType directions = roiImage->GetDirection(); vtkMatrix4x4 *matrix = vtkMatrix4x4::New(); matrix->Identity(); for (int i=0; i<3; i++) for (int j=0; j<3; j++) matrix->SetElement (i, j, directions (i,j)); manager->Reset(); manager->GetROILimiter()->SetMaskImage (converter->GetOutput()); manager->GetROILimiter()->SetDirectionMatrix (matrix); typename ROIType::PointType origin = roiImage->GetOrigin(); vtkMatrix4x4 *matrix2 = vtkMatrix4x4::New(); matrix2->Identity(); for (int i=0; i<3; i++) for (int j=0; j<3; j++) matrix2->SetElement (i, j, directions (i,j)); double v_origin[4], v_origin2[4]; for (int i=0; i<3; i++) v_origin[i] = origin[i]; v_origin[3] = 1.0; matrix->MultiplyPoint (v_origin, v_origin2); for (int i=0; i<3; i++) matrix2->SetElement (i, 3, v_origin[i]-v_origin2[i]); roiManager->SetInput (converter->GetOutput()); roiManager->SetDirectionMatrix (matrix2); roiManager->GenerateData(); std::sort(labels.begin(), labels.end()); roiLabels.clear(); if (roiComboBox) { roiComboBox->blockSignals(true); roiComboBox->clear(); unsigned int i = 0; for(int label : labels) { roiLabels[i] = label; roiComboBox->addItem("ROI " + QString::number(label)); ++i; } roiComboBox->blockSignals(false); roiComboBox->setCurrentIndex(0); } matrix->Delete(); matrix2->Delete(); view->render(); } medVtkFibersDataInteractor::medVtkFibersDataInteractor(medAbstractView *parent): medAbstractImageViewInteractor(parent), d(new medVtkFibersDataInteractorPrivate) { d->data = nullptr; d->dataset = nullptr; d->view = dynamic_cast(parent); medVtkViewBackend* backend = static_cast(parent->backend()); d->view2d = backend->view2D; d->view3d = backend->view3D; d->render = backend->renWin; d->manager = vtkFiberDataSetManager::New(); d->manager->SetHelpMessageVisibility(0); d->roiManager = vtkIsosurfaceManager::New(); d->spoDefaultLut = d->manager->GetLookupTable(); d->manager->SetRenderWindowInteractor(d->render->GetInteractor()); d->roiManager->SetRenderWindowInteractor(d->render->GetInteractor()); d->toolboxWidget = nullptr; d->bundleToolboxWidget = nullptr; d->colorFiberParameter = new medStringListParameterL("colorFiberParameter", this); d->colorFiberParameter->setToolTip(tr("Choose the coloring method of the fibers.")); d->colorFiberParameter->getLabel()->setText(tr("Color fibers by:")); d->colorFiberParameter->addItem("Local orientation"); d->colorFiberParameter->addItem("Global orientation"); d->parameters << d->colorFiberParameter; d->poLutWidget = nullptr; d->LutFiberParameter = new medStringListParameterL("LutParameter", this); d->LutFiberParameter->setToolTip("Choose type of LUT"); d->LutFiberParameter->getLabel()->setText(tr("Choose type of LUT:")); d->LutFiberParameter->addItems(vtkLookupTableManager::GetAvailableLookupTablesInStringList()); d->LutFiberParameter->setValue("Loni"); d->parameters << d->LutFiberParameter; d->alphaTransparencyParameter = new medBoolParameterL("ActivateTransparencyParameter", this); d->alphaTransparencyParameter->setToolTip(tr("Activate transparency.")); d->alphaTransparencyParameter->setText(tr("Activate transparency")); d->alphaTransparencyParameter->getCheckBox()->setEnabled(true); d->parameters << d->alphaTransparencyParameter; d->minIntensityParameter = new medDoubleParameterL("Min Intensity", this); d->minIntensityParameter->setToolTip(tr("Min Intensity.")); d->minIntensityParameter->getLabel()->setText(tr("Min Intensity:")); d->minIntensityParameter->setRange(0.0, 1.0); d->minIntensityParameter->setValue(0); d->parameters << d->minIntensityParameter; d->maxIntensityParameter = new medDoubleParameterL("Max Intensity", this); d->maxIntensityParameter->setToolTip(tr("Max Intensity.")); d->maxIntensityParameter->getLabel()->setText(tr("Max Intensity:")); d->maxIntensityParameter->setRange(0.0, 1.0); d->maxIntensityParameter->setValue(1); d->parameters << d->maxIntensityParameter; connect(d->maxIntensityParameter, SIGNAL(valueChanged(double)), this, SLOT(updateRange())); connect(d->minIntensityParameter, SIGNAL(valueChanged(double)), this, SLOT(updateRange())); connect(d->alphaTransparencyParameter, SIGNAL(valueChanged(bool)), this, SLOT(updateLut())); connect(d->LutFiberParameter, SIGNAL(valueChanged(QString)), this, SLOT(updateLut())); d->shapesDisplayedGroupParameter = new medBoolGroupParameterL("shapesDisplayedGroupParameter", this); d->parameters << d->shapesDisplayedGroupParameter; d->displayPolylinesParameter = new medBoolParameterL("displayPolylinesParameter", this); d->displayPolylinesParameter->setToolTip(tr("Use polylines to draw the fibers.")); d->displayPolylinesParameter->setText(tr("Display fibers as polylines")); d->displayPolylinesParameter->getLabel()->hide(); d->shapesDisplayedGroupParameter->addParameter(d->displayPolylinesParameter); d->displayRibbonsParameter = new medBoolParameterL("displayRibbonsParameter", this); d->displayRibbonsParameter->setToolTip(tr("Use ribbons to draw the fibers.")); d->displayRibbonsParameter->setText(tr("Display fibers as ribbons")); d->displayRibbonsParameter->getLabel()->hide(); d->shapesDisplayedGroupParameter->addParameter(d->displayRibbonsParameter); d->displayTubesParameter = new medBoolParameterL("displayTubesParameter", this); d->displayTubesParameter->setToolTip(tr("Use tubes to draw the fibers.")); d->displayTubesParameter->setText(tr("Display fibers as tubes")); d->displayTubesParameter->getLabel()->hide(); d->shapesDisplayedGroupParameter->addParameter(d->displayTubesParameter); d->displayPolylinesParameter->setValue(true); d->radiusParameter = new medIntParameterL("fibersRadiusParameter", this); d->radiusParameter->setToolTip(tr("Increase of decrease the radius of the fibers (except if there are being drawn with polylines).")); d->radiusParameter->getLabel()->setText(tr("Fibers radius:")); d->radiusParameter->setRange(1, 10); connect (d->colorFiberParameter, SIGNAL(valueChanged(QString)), this, SLOT(setFiberColorMode(QString))); connect (d->displayPolylinesParameter, SIGNAL(valueChanged(bool)), this, SLOT(selectLineMode (bool))); connect (d->displayRibbonsParameter, SIGNAL(valueChanged(bool)), this, SLOT(selectRibbonMode(bool))); connect (d->displayTubesParameter, SIGNAL(valueChanged(bool)), this, SLOT(selectTubeMode(bool))); connect (d->radiusParameter, SIGNAL(valueChanged(int)), this, SLOT(setRadius(int))); //--Bundling d->andParameter = new medBoolParameterL("andFiberParameter", this); d->andParameter->setToolTip(tr("If \"AND\" is selected fibers will need to overlap with this ROI to be displayed.")); d->andParameter->setText("AND"); d->andParameter->setValue(true); d->andParameter->getLabel()->hide(); d->notParameter = new medBoolParameterL("notFiberParameter", this); d->notParameter->setToolTip(tr("If \"NOT\" is selected fibers overlapping with this ROI will not be displayed.")); d->notParameter->setText("NOT"); d->notParameter->getLabel()->hide(); d->nullParameter = new medBoolParameterL("nullFiberParameter", this); d->nullParameter->setToolTip(tr("If \"NULL\" is selected this ROI won't be taken into account.")); d->nullParameter->setText("NULL"); d->nullParameter->getLabel()->hide(); d->bundleOperationGroupParameter = new medBoolGroupParameterL("bundleOperationGroupParameter", this); d->bundleOperationGroupParameter->addParameter(d->andParameter); d->bundleOperationGroupParameter->addParameter(d->notParameter); d->bundleOperationGroupParameter->addParameter(d->nullParameter); d->tagParameter = new medTriggerParameterL("tagFiberParameter", this); d->tagParameter->setToolTip(tr("Tag the currently shown bundle to let the application memorize it and another, new bundling box, will appear.\nThis new box will also isolate fibers but will also consider the previously \"tagged\" fibers.")); d->tagParameter->setButtonText("Tag"); d->addParameter = new medBoolParameterL("addFiberParameter", this); d->addParameter->setToolTip(tr("Select to either include the fibers that overlap with the bundling box, or to include the ones that do not overlap.")); d->addParameter->setText("Add"); d->addParameter->setValue(true); d->resetParameter = new medTriggerParameterL("resetFiberParameter", this); d->resetParameter->setToolTip(tr("Reset all previously tagged bundling boxes.")); d->resetParameter->setButtonText("Reset"); d->validateParameter = new medTriggerParameterL("validateFiberParameter", this); d->validateParameter->setToolTip(tr("Save the current shown bundle and show useful information about it.")); d->validateParameter->setButtonText("Validate"); d->saveParameter = new medTriggerParameterL("saveFiberParameter", this); d->saveParameter->setToolTip(tr("Save all bundles to database")); d->saveParameter->setButtonText("Save"); d->showAllBundleParameter = new medBoolParameterL("showAllBundleFiberParameter", this); d->showAllBundleParameter->setValue(true); d->showAllBundleParameter->setToolTip(tr("Uncheck if you do not want the previously validated bundles to be displayed.")); d->showAllBundleParameter->setText(tr("Show all bundles")); d->parameters << d->showAllBundleParameter; d->showBundleBox = new medBoolParameterL("showBundleFiberBox", this); d->showBundleBox->setToolTip(tr("Select to activate and show the fiber bundling box on the screen.")); d->showBundleBox->setText("Activate bundling box"); d->showBundleBox->setValue(false); d->bundlingModel = new QStandardItemModel(0, 1, this); d->bundlingModel->setHeaderData(0, Qt::Horizontal, tr("Fiber bundles")); d->opacityParam = new medDoubleParameterL("Opacity", this); d->opacityParam->setRange(0,1); d->opacityParam->setSingleStep(0.01); d->opacityParam->setValue(1); d->parameters << d->opacityParam; connect(d->opacityParam, SIGNAL(valueChanged(double)), this, SLOT(setOpacity(double))); connect (d->validateParameter, SIGNAL(triggered()), this, SLOT(validateBundling())); connect (d->showBundleBox, SIGNAL(valueChanged(bool)), this, SLOT(setBoxVisibility (bool))); connect (d->addParameter, SIGNAL(valueChanged(bool)), this, SLOT(setBoxBooleanOperation(bool))); connect (d->showAllBundleParameter, SIGNAL(valueChanged(bool)), this, SLOT(setAllBundlesVisibility (bool))); connect (d->tagParameter, SIGNAL(triggered()),this, SLOT(tagSelection())); connect (d->resetParameter, SIGNAL(triggered()), this, SLOT(resetSelection())); connect (d->saveParameter, SIGNAL(triggered()), this, SLOT(saveBundlesInDataBase())); connect (d->andParameter, SIGNAL(valueChanged(bool)), this, SLOT(setRoiAddOperation(bool))); connect (d->notParameter, SIGNAL(valueChanged(bool)), this, SLOT(setRoiNotOperation(bool))); connect (d->nullParameter, SIGNAL(valueChanged(bool)), this, SLOT(setRoiNullOperation(bool))); connect (d->bundlingModel, SIGNAL(itemChanged(QStandardItem*)), this, SLOT(changeBundlingItem(QStandardItem*))); connect(d->view, SIGNAL(orientationChanged()), this, SLOT(updateWidgets())); if(d->view->layer(d->data) == 0) { switch(d->view2d->GetViewOrientation()) { case vtkImageView2D::VIEW_ORIENTATION_AXIAL: d->view->setOrientation(medImageView::VIEW_ORIENTATION_AXIAL); break; case vtkImageView2D::VIEW_ORIENTATION_SAGITTAL: d->view->setOrientation(medImageView::VIEW_ORIENTATION_SAGITTAL); break; case vtkImageView2D::VIEW_ORIENTATION_CORONAL: d->view->setOrientation(medImageView::VIEW_ORIENTATION_CORONAL); break; } } d->slicingParameter = new medIntParameterL("Slicing", this); connect(d->slicingParameter, SIGNAL(valueChanged(int)), this, SLOT(moveToSlice(int))); connect(d->view->positionBeingViewedParameter(), SIGNAL(valueChanged(QVector3D)), this, SLOT(updateSlicingParam())); } medVtkFibersDataInteractor::~medVtkFibersDataInteractor() { delete d; d = nullptr; } QString medVtkFibersDataInteractor::description() const { return tr("Interactor to help visualising Fibers"); } QString medVtkFibersDataInteractor::identifier() const { return "medVtkFibersDataInteractor"; } QStringList medVtkFibersDataInteractor::handled() const { return medVtkFibersDataInteractor::dataHandled(); } QStringList medVtkFibersDataInteractor::dataHandled() { QStringList d = QStringList() << "medVtkFibersData"; return d; } bool medVtkFibersDataInteractor::isDataTypeHandled(QString dataType) const { if (dataType == "medVtkFibersData") return true; return false; } bool medVtkFibersDataInteractor::registered() { medViewFactory *factory = medViewFactory::instance(); factory->registerInteractor("medVtkFibersDataInteractor", QStringList () << "medVtkView" << medVtkFibersDataInteractor::dataHandled()); return true; } void medVtkFibersDataInteractor::setInputData(medAbstractData *data) { medAbstractInteractor::setInputData(data); if (!data) return; if(data->identifier() != "medVtkFibersData") return; if (vtkFiberDataSet *dataset = static_cast(data->data())) { d->dataset = dataset; d->manager->SetInput(d->dataset); // Update color fiber parameter d->colorFiberParameter->blockSignals(true); d->colorFiberParameter->clear(); d->colorFiberParameter->addItem("Local orientation"); d->colorFiberParameter->addItem("Global orientation"); vtkPolyData *fibersData = d->dataset->GetFibers(); unsigned int numArrays = fibersData->GetPointData()->GetNumberOfArrays(); d->oScalarRangeVect.resize(numArrays); for (unsigned int i = 0;i < numArrays;++i) { vtkDataArray *tmpArray = fibersData->GetPointData()->GetArray(i); double *pdfScalarRangeTmp = tmpArray->GetRange(); d->oScalarRangeVect[i].resize(2); d->oScalarRangeVect[i][0] = pdfScalarRangeTmp[0]; d->oScalarRangeVect[i][1] = pdfScalarRangeTmp[1]; if (tmpArray->GetNumberOfComponents() > 1) continue; d->colorFiberParameter->addItem(fibersData->GetPointData()->GetArrayName(i)); } d->colorFiberParameter->blockSignals(false); double pdfTmp[2]; pdfTmp[0] = 0.0; pdfTmp[1] = 0.0; if (numArrays > 0) { pdfTmp[0] = d->oScalarRangeVect[0][0]; pdfTmp[1] = d->oScalarRangeVect[0][1]; } initWindowLevelParameters(pdfTmp); data->setMetaData("BundleList", QStringList()); data->setMetaData("BundleColorList", QStringList()); d->actor = d->manager->GetOutput(); d->opacityProperty = vtkSmartPointer ::New(); d->actor->SetProperty(d->opacityProperty); d->data = data; d->view2d->SetInput(d->actor, d->view->layer(d->data)); //TODO - harmonise all of this setInput method in vtkImageView. d->view3d->GetRenderer()->AddActor(d->actor); this->updateWidgets(); } d->parameters << visibilityParameter(); } void medVtkFibersDataInteractor::initWindowLevelParameters(double * range) { double window = range[1] - range[0]; double halfWidth = 0.5 * window; double levelMin = range[0] - halfWidth; double levelMax = range[1] + halfWidth; double intensityStep = qMin(0.1, (levelMax - levelMin) / 1000); d->minIntensityParameter->blockSignals(true); d->maxIntensityParameter->blockSignals(true); d->minIntensityParameter->setSingleStep(intensityStep); d->minIntensityParameter->setDecimals(6); d->maxIntensityParameter->setSingleStep(intensityStep); d->maxIntensityParameter->setDecimals(6); d->minIntensityParameter->setRange(levelMin, levelMax); d->maxIntensityParameter->setRange(levelMin, levelMax); d->minIntensityParameter->setValue(range[0]); d->maxIntensityParameter->setValue(range[1]); d->maxIntensityParameter->blockSignals(false); d->minIntensityParameter->blockSignals(false); } void medVtkFibersDataInteractor::changeBundlingItem(QStandardItem *item) { QString itemOldName = item->data(Qt::UserRole+1).toString(); if (itemOldName != item->text()) { this->changeBundleName(itemOldName,item->text()); item->setData(item->text(),Qt::UserRole+1); } this->setBundleVisibility(item->text(), item->checkState()); } void medVtkFibersDataInteractor::changeBundleName(QString oldName, QString newName) { if ((!d->dataset)||(!d->manager)) return; d->manager->ChangeBundleName(oldName.toStdString(), newName.toStdString()); } void medVtkFibersDataInteractor::setVisibility(bool visible) { d->manager->SetVisibility(visible); d->view->render(); } void medVtkFibersDataInteractor::setBoxVisibility(bool visible) { if (d->view && d->view->orientation() != medImageView::VIEW_ORIENTATION_3D) { medMessageController::instance().showError("View must be in 3D mode to activate the bundling box", 3000); d->manager->SetBoxWidget(false); return; } d->manager->SetBoxWidget(visible); } void medVtkFibersDataInteractor::selectLineMode(bool value) { if(value) this->setRenderingMode(medVtkFibersDataInteractor::Lines); } void medVtkFibersDataInteractor::selectRibbonMode(bool value) { if(value) this->setRenderingMode(medVtkFibersDataInteractor::Ribbons); } void medVtkFibersDataInteractor::selectTubeMode(bool value) { if(value) this->setRenderingMode(medVtkFibersDataInteractor::Tubes); } void medVtkFibersDataInteractor::setRenderingMode(RenderingMode mode) { switch(mode) { case medVtkFibersDataInteractor::Lines: d->manager->SetRenderingModeToPolyLines(); break; case medVtkFibersDataInteractor::Ribbons: d->manager->SetRenderingModeToRibbons(); break; case medVtkFibersDataInteractor::Tubes: d->manager->SetRenderingModeToTubes(); break; default: dtkDebug() << "medVtkFibersDataInteractor: unknown rendering mode"; } d->view->render(); } void medVtkFibersDataInteractor::setFiberColorMode(QString mode) { if (mode == "Local orientation") { d->poLutWidget->hide(); d->manager->SetLookupTable(d->spoDefaultLut); d->manager->SetColorModeToLocalFiberOrientation(); } else if (mode == "Global orientation") { d->poLutWidget->hide(); d->manager->SetLookupTable(d->spoDefaultLut); d->manager->SetColorModelToGlobalFiberOrientation(); } else { switchMinMaxRangeSource(mode); updateCustomLUT(mode); d->manager->GetLookupTable()->SetRange(d->minIntensityParameter->value(), d->maxIntensityParameter->value()); d->poLutWidget->show(); } d->view->render(); } void medVtkFibersDataInteractor::updateRange() { vtkLookupTable * lut = vtkLookupTable::SafeDownCast(d->manager->GetLookupTable()); if (lut) { medDoubleParameterL *sender = dynamic_cast(this->sender()); if (sender) { if (sender == d->minIntensityParameter && d->minIntensityParameter->value() >= d->maxIntensityParameter->value()) { d->maxIntensityParameter->blockSignals(true); d->maxIntensityParameter->setValue(d->minIntensityParameter->value()); d->maxIntensityParameter->blockSignals(false); } else if (sender == d->maxIntensityParameter && d->maxIntensityParameter->value() <= d->minIntensityParameter->value()) { d->minIntensityParameter->blockSignals(true); d->minIntensityParameter->setValue(d->maxIntensityParameter->value()); d->minIntensityParameter->blockSignals(false); } } lut->SetRange(d->minIntensityParameter->value(), d->maxIntensityParameter->value()); d->view->render(); } } void medVtkFibersDataInteractor::updateCustomLUT(QString mode) { QString lutName = d->LutFiberParameter->value(); vtkLookupTable *lut = vtkLookupTableManager::GetLookupTable(lutName.toStdString()); lut->SetRange(d->minIntensityParameter->value(), d->maxIntensityParameter->value()); if (!d->alphaTransparencyParameter->value()) { // remove the alpha channel from the LUT, it messes up the mesh lut = vtkLookupTableManager::removeLUTAlphaChannel(lut); } d->manager->SetLookupTable(lut); d->manager->SetColorModeToLocalFiberOrientation(); for (int i = 0; i < d->manager->GetNumberOfPointArrays(); i++) { if (d->manager->GetPointArrayName(i)) { QString pointArrayName = d->manager->GetPointArrayName(i); if (pointArrayName == mode) { d->manager->SetColorModeToPointArray(i); break; } } } d->view3d->GetScalarBar()->SetLookupTable(lut); d->view2d->GetScalarBar()->SetLookupTable(lut); } void medVtkFibersDataInteractor::switchMinMaxRangeSource(QString mode) { vtkPolyData *fibersData = d->dataset->GetFibers(); unsigned int numArrays = fibersData->GetPointData()->GetNumberOfArrays(); for (unsigned int i = 0; i < numArrays; i++) { if (fibersData->GetPointData()->GetArrayName(i) == mode) { double dfTmp2[2]; dfTmp2[0] = d->oScalarRangeVect[i][0]; dfTmp2[1] = d->oScalarRangeVect[i][1]; initWindowLevelParameters(dfTmp2); break; } } } void medVtkFibersDataInteractor::updateLut() { updateCustomLUT(d->colorFiberParameter->value()); d->view->render(); } void medVtkFibersDataInteractor::setBoxBooleanOperation(bool value) { if (value) this->setBoxBooleanOperation(medVtkFibersDataInteractor::Plus); else this->setBoxBooleanOperation(medVtkFibersDataInteractor::Minus); } void medVtkFibersDataInteractor::setBoxBooleanOperation(BooleanOperation op) { switch(op) { case medVtkFibersDataInteractor::Plus: d->manager->GetVOILimiter()->SetBooleanOperationToAND(); break; case medVtkFibersDataInteractor::Minus: d->manager->GetVOILimiter()->SetBooleanOperationToNOT(); break; default: dtkDebug() << "medVtkFibersDataInteractor: Unknown boolean operations"; } d->manager->GetVOILimiter()->Modified(); d->view->render(); } void medVtkFibersDataInteractor::tagSelection() { d->manager->SwapInputOutput(); d->view->render(); } void medVtkFibersDataInteractor::resetSelection() { d->manager->Reset(); d->view->render(); } void medVtkFibersDataInteractor::validateSelection(const QString &name, const QColor &color) { if (!d->data) return; double color_d[3] = {(double)color.red()/255.0, (double)color.green()/255.0, (double)color.blue()/255.0}; d->manager->Validate (name.toLatin1().constData(), color_d); d->view2d->SetInput(d->manager->GetBundleActor(name.toLatin1().constData()), d->view->layer(d->data)); d->view3d->GetRenderer()->AddActor(d->manager->GetBundleActor(name.toLatin1().constData())); d->data->setMetaData("BundleList", name); d->data->setMetaData("BundleColorList", color.name()); // reset to initial navigation state d->manager->Reset(); d->view->render(); } void medVtkFibersDataInteractor::saveBundlesInDataBase() { if (!d->dataset) return; vtkFiberDataSet::vtkFiberBundleListType bundles = d->dataset->GetBundleList(); vtkFiberDataSet::vtkFiberBundleListType::iterator it = bundles.begin(); medAbstractData *tmpBundle; while (it!=bundles.end()) { tmpBundle = medAbstractDataFactory::instance()->create("medVtkFibersData"); if (!tmpBundle) return; vtkSmartPointer bundle = vtkFiberDataSet::New(); bundle->SetFibers((*it).second.Bundle); tmpBundle->setData(bundle); QString newSeriesDescription = d->data->metadata ( medMetaDataKeys::SeriesDescription.key() ); newSeriesDescription += " "; newSeriesDescription += (*it).first.c_str(); tmpBundle->setMetaData ( medMetaDataKeys::SeriesDescription.key(), newSeriesDescription ); for( QString metaData : d->data->metaDataList() ) { if ((metaData == "BundleList")||(metaData == "BundleColorList")) continue; tmpBundle->setMetaData (metaData, d->data->metaDataValues (metaData)); } for( QString property : d->data->propertyList() ) { tmpBundle->addProperty ( property,d->data->propertyValues ( property ) ); } QString generatedID = QUuid::createUuid().toString().replace("{","").replace("}",""); tmpBundle->setMetaData ( medMetaDataKeys::SeriesID.key(), generatedID ); medDataManager::instance().importData(tmpBundle); ++it; } } void medVtkFibersDataInteractor::bundleImageStatistics (const QString &bundleName, QMap &mean, QMap &min, QMap &max, QMap &var) { vtkPolyData *bundleData = d->dataset->GetBundle(bundleName.toLatin1().constData()).Bundle; if (!bundleData) return; if (bundleData->GetPointData()->HasArray("Tensors")) { // Specific TTK case itk::FiberBundleStatisticsCalculator::Pointer statCalculator = itk::FiberBundleStatisticsCalculator::New(); statCalculator->SetInput (bundleData); try { statCalculator->Compute(); } catch(itk::ExceptionObject &e) { dtkDebug() << e.GetDescription(); return; } double meanVal, minVal, maxVal, varVal; statCalculator->GetADCStatistics(meanVal, minVal, maxVal, varVal); mean["ADC"] = meanVal; min["ADC"] = minVal; max["ADC"] = maxVal; var["ADC"] = varVal; statCalculator->GetFAStatistics(meanVal, minVal, maxVal, varVal); mean["FA"] = meanVal; min["FA"] = minVal; max["FA"] = maxVal; var["FA"] = varVal; } vtkPointData *bundlePointData = bundleData->GetPointData(); unsigned int numberOfArrays = bundlePointData->GetNumberOfArrays(); vtkPoints* points = bundleData->GetPoints(); vtkCellArray* lines = bundleData->GetLines(); lines->InitTraversal(); for (unsigned int i = 0;i < numberOfArrays;++i) { // To do, explore fibers one by one, compute mean value on each fiber, then deduce all stats // Put them inside maps QString arrayName = bundlePointData->GetArrayName(i); vtkDataArray *imageCoefficients = bundlePointData->GetArray(i); unsigned int numComponents = imageCoefficients->GetNumberOfComponents(); if (numComponents != 1) continue; vtkIdType npts = 0; vtkIdType* ptids = 0; vtkIdType test = lines->GetNextCell (npts, ptids); double sumData = 0; double squareSumData = 0; double minValue = HUGE_VAL; double maxValue = - HUGE_VAL; unsigned int numberOfLines = 0; // go over lines, each cell is a line while (test) { double meanFiberData = 0; for (int k=0; kGetPoint (ptids[k], pt); double tupleValue = 0; imageCoefficients->GetTuple(ptids[k], &tupleValue); if (k == 0) { minValue = tupleValue; maxValue = tupleValue; } if (minValue > tupleValue) minValue = tupleValue; if (maxValue < tupleValue) maxValue = tupleValue; meanFiberData += tupleValue; } meanFiberData /= npts; if (minValue > meanFiberData) minValue = meanFiberData; if (maxValue < meanFiberData) maxValue = meanFiberData; sumData += meanFiberData; squareSumData += meanFiberData * meanFiberData; ++numberOfLines; test = lines->GetNextCell (npts, ptids); } if (std::isfinite (sumData / numberOfLines)) { mean[arrayName] = sumData / numberOfLines; min[arrayName] = minValue; max[arrayName] = maxValue; var[arrayName] = 0; if (numberOfLines > 1) var[arrayName] = (squareSumData - sumData * sumData / numberOfLines) / (numberOfLines - 1.0); } } } void medVtkFibersDataInteractor::computeBundleLengthStatistics (const QString &name, double &mean, double &min, double &max, double &var) { vtkPolyData *bundleData = d->dataset->GetBundle(name.toLatin1().constData()).Bundle; if (!bundleData) return; vtkPoints* points = bundleData->GetPoints(); vtkCellArray* lines = bundleData->GetLines(); lines->InitTraversal(); vtkIdType npts = 0; vtkIdType* ptids = 0; vtkIdType test = lines->GetNextCell (npts, ptids); double sumData = 0; double squareSumData = 0; double minValue = HUGE_VAL; double maxValue = - HUGE_VAL; unsigned int numberOfLines = 0; // go over lines, each cell is a line while (test) { double lengthFiber = 0; for (int k=1; kGetPoint (ptids[k-1], pt1); points->GetPoint (ptids[k], pt2); double normDiff = 0; for (unsigned int l = 0;l < 3;++l) normDiff += (pt2[l] - pt1[l])*(pt2[l] - pt1[l]); lengthFiber += sqrt(normDiff); } if (minValue > lengthFiber) minValue = lengthFiber; if (maxValue < lengthFiber) maxValue = lengthFiber; sumData += lengthFiber; squareSumData += lengthFiber * lengthFiber; ++numberOfLines; test = lines->GetNextCell (npts, ptids); } if (std::isfinite (sumData / numberOfLines)) { mean = sumData / numberOfLines; min = minValue; max = maxValue; var = 0; if (numberOfLines > 1) var = (squareSumData - sumData * sumData / numberOfLines) / (numberOfLines - 1.0); } else { mean = 0; min = 0; max = 0; var = 0; } } void medVtkFibersDataInteractor::bundleLengthStatistics(const QString &name, double &mean, double &min, double &max, double &var) { if (!d->meanLengthList.contains(name)) { this->computeBundleLengthStatistics(name, mean, min, max, var); d->meanLengthList[name] = mean; d->minLengthList[name] = min; d->maxLengthList[name] = max; d->varLengthList[name] = var; } else { mean = d->meanLengthList[name]; min = d->minLengthList[name]; max = d->maxLengthList[name]; var = d->varLengthList[name]; } } void medVtkFibersDataInteractor::clearStatistics(void) { d->meanLengthList.clear(); d->minLengthList.clear(); d->maxLengthList.clear(); d->varLengthList.clear(); } void medVtkFibersDataInteractor::setProjection(const QString& value) { if (!d->view) return; if (value=="true") { d->view2d->AddDataSet( d->manager->GetCallbackOutput() ); } } void medVtkFibersDataInteractor::setRadius (int value) { d->manager->SetRadius (value); d->view->render(); } void medVtkFibersDataInteractor::setRoiBoolean(int roi, int meaning) { d->manager->GetROILimiter()->SetBooleanOperation (roi, meaning); d->view->render(); } int medVtkFibersDataInteractor::roiBoolean(int roi) { return d->manager->GetROILimiter()->GetBooleanOperationVector()[roi]; } void medVtkFibersDataInteractor::setBundleVisibility(const QString &name, bool visibility) { d->manager->SetBundleVisibility(name.toLatin1().constData(), (int)visibility); d->view->render(); } bool medVtkFibersDataInteractor::bundleVisibility(const QString &name) const { return d->manager->GetBundleVisibility(name.toLatin1().constData()); } void medVtkFibersDataInteractor::setAllBundlesVisibility(bool visibility) { if (visibility) { d->manager->ShowAllBundles(); for(int i = 0; i < d->bundlingModel->rowCount(); ++i) d->bundlingModel->item(i)->setCheckState(Qt::Checked); } else { d->manager->HideAllBundles(); for(int i = 0; i < d->bundlingModel->rowCount(); ++i) d->bundlingModel->item(i)->setCheckState(Qt::Unchecked); } d->view->render(); } void medVtkFibersDataInteractor::validateBundling() { unsigned int bundleNumber = 1; QString text = tr("Fiber bundle #") + QString::number(bundleNumber); // Handle potential deleted/renamed bundles while (!d->bundlingModel->findItems(text).empty()) { bundleNumber++; text = tr("Fiber bundle #") + QString::number(bundleNumber); } QColor color = QColor::fromHsv(qrand()%360, 255, 210); this->validateSelection(text, color); this->addBundle(text, color); } void medVtkFibersDataInteractor::addBundle (const QString &name, const QColor &color) { int row = d->bundlingModel->rowCount(); QStandardItem *item = new QStandardItem (name); item->setCheckable(true); item->setTristate(false); item->setEditable(true); item->setCheckState(Qt::Checked); QMap meanData; QMap minData; QMap maxData; QMap varData; double meanLength = 0.0; double minLength = 0.0; double maxLength = 0.0; double varLength = 0.0; this->bundleImageStatistics(name, meanData, minData, maxData, varData); this->bundleLengthStatistics(name, meanLength, minLength, maxLength, varLength); for(QString key : meanData.keys()) { QStandardItem *childItem1 = new QStandardItem (key + ": " + QString::number(meanData[key])); childItem1->setEditable(false); childItem1->appendRow(new QStandardItem (tr("mean: ") + QString::number(meanData[key]))); childItem1->appendRow(new QStandardItem (tr("variance: ") + QString::number(varData[key]))); childItem1->appendRow(new QStandardItem (tr("min: ") + QString::number(minData[key]))); childItem1->appendRow(new QStandardItem (tr("max: ") + QString::number(maxData[key]))); item->appendRow(childItem1); } QStandardItem *childItem2 = new QStandardItem (tr("Length: ") + QString::number(meanLength)); childItem2->setEditable(false); childItem2->appendRow(new QStandardItem (tr("mean: ") + QString::number(meanLength))); childItem2->appendRow(new QStandardItem (tr("variance: ") + QString::number(varLength))); childItem2->appendRow(new QStandardItem (tr("min: ") + QString::number(minLength))); childItem2->appendRow(new QStandardItem (tr("max: ") + QString::number(maxLength))); item->appendRow(childItem2); item->setData(name,Qt::UserRole+1); d->bundlingModel->setItem(row, item); d->bundlingModel->setData(d->bundlingModel->index(row, 0, QModelIndex()), true, Qt::CheckStateRole); d->bundlingModel->setData(d->bundlingModel->index(row, 0, QModelIndex()), color, Qt::DecorationRole); } void medVtkFibersDataInteractor::setRoiAddOperation(bool value) { if (d->roiLabels.isEmpty()) return; int roi = d->roiLabels[d->roiComboBox->currentIndex()]; if (value) this->setRoiBoolean(roi, 2); } void medVtkFibersDataInteractor::setRoiNotOperation(bool value) { if (d->roiLabels.isEmpty()) return; int roi = d->roiLabels[d->roiComboBox->currentIndex()]; if (value) this->setRoiBoolean(roi, 1); } void medVtkFibersDataInteractor::setRoiNullOperation(bool value) { if (d->roiLabels.isEmpty()) return; int roi = d->roiLabels[d->roiComboBox->currentIndex()]; if (value) this->setRoiBoolean(roi, 0); } void medVtkFibersDataInteractor::loadRoiFromFile() { if (!d->view) return; QString roiFileName = QFileDialog::getOpenFileName(0, tr("Open ROI"), "", tr("Image file (*.*)")); if (roiFileName.isEmpty()) return; d->roiImportUuid = medDataManager::instance().importPath(roiFileName,false); connect(&medDataManager::instance(), SIGNAL(dataImported(medDataIndex,QUuid)), this, SLOT(importROI(medDataIndex,QUuid)), Qt::UniqueConnection); } void medVtkFibersDataInteractor::importROI(const medDataIndex& index, QUuid uuid) { if (uuid != d->roiImportUuid) return; this->importROI(index); } void medVtkFibersDataInteractor::importROI(const medDataIndex& index) { dtkSmartPointer data = medDataManager::instance().retrieveData(index); // we accept only ROIs (itkDataImageUChar3) // TODO try dynamic_cast of medAbstractMaskData would be better - RDE if (!data || (data->identifier() != "itkDataImageUChar3" && data->identifier() != "itkDataImageChar3" && data->identifier() != "itkDataImageUShort3" && data->identifier() != "itkDataImageShort3" && data->identifier() != "itkDataImageUInt3" && data->identifier() != "itkDataImageInt3" && data->identifier() != "itkDataImageFloat3" && data->identifier() != "itkDataImageDouble3")) { medMessageController::instance().showError(tr("Unable to load ROI, format not supported yet"), 3000); return; } d->dropOrOpenRoi->setPixmap(medDataManager::instance().thumbnail(index).scaled(d->dropOrOpenRoi->sizeHint())); d->setROI(data); d->setROI(data); d->setROI(data); d->setROI(data); d->setROI(data); d->setROI(data); d->setROI(data); d->setROI(data); } void medVtkFibersDataInteractor::clearRoi(void) { d->manager->GetROILimiter()->SetMaskImage (0); d->manager->GetROILimiter()->SetDirectionMatrix (0); d->roiManager->ResetData(); // clear medDropSite and put text again d->dropOrOpenRoi->clear(); d->dropOrOpenRoi->setText(tr("Drag-and-drop\nfrom the database\nto open a ROI.")); if (d->roiComboBox) d->roiComboBox->clear(); d->roiLabels.clear(); d->view->render(); } void medVtkFibersDataInteractor::selectRoi(int value) { if (d->roiLabels.isEmpty()) return; int boolean = this->roiBoolean (d->roiLabels[value]); switch (boolean) { case 2: d->andParameter->blockSignals(true); d->andParameter->setValue(true); d->andParameter->blockSignals(false); break; case 1: d->notParameter->blockSignals(true); d->notParameter->setValue(true); d->notParameter->blockSignals(false); break; case 0: default: d->nullParameter->blockSignals(true); d->nullParameter->setValue(true); d->nullParameter->blockSignals(false); break; } } void medVtkFibersDataInteractor::setOpacity(double opacity) { d->opacityProperty->SetOpacity(opacity); d->view->render(); } void medVtkFibersDataInteractor::setVisible(bool visible) { int v = (visible) ? 1 : 0; d->actor->SetVisibility(v); d->view->render(); } void medVtkFibersDataInteractor::setWindowLevel (QHash) { } void medVtkFibersDataInteractor::moveToSlice (int slice) { //TODO find a way to get woorldCoordinate for slice from vtkInria. // instead of moving to the slice corresponding on the first layer dropped. if(d->view->is2D() && slice != d->view2d->GetSlice()) { d->view2d->SetSlice(slice); d->view2d->Render(); } } void medVtkFibersDataInteractor::removeData() { d->view2d->RemoveLayer(d->view->layer(d->data)); d->manager->Delete(); } QWidget* medVtkFibersDataInteractor::buildToolBoxWidget() { d->toolboxWidget = new QWidget; QVBoxLayout *toolBoxLayout = new QVBoxLayout(d->toolboxWidget); toolBoxLayout->addWidget(d->colorFiberParameter->getLabel()); toolBoxLayout->addWidget(d->colorFiberParameter->getComboBox()); //////////////////////////////////////////////////////////////// // LUT and WindowLevel range for it QGroupBox *poGroupBoxLut = new QGroupBox(/*tr("LUT: ")*/); poGroupBoxLut->setAlignment(Qt::AlignLeft); d->poLutWidget = poGroupBoxLut; d->poLutWidget->hide(); QVBoxLayout *lutLayout = new QVBoxLayout(poGroupBoxLut); QWidget *poMinMaxWidget = new QWidget(); QFormLayout *minMaxLayout = new QFormLayout(poMinMaxWidget); lutLayout->addWidget(d->LutFiberParameter->getLabel()); lutLayout->addWidget(d->LutFiberParameter->getComboBox()); lutLayout->addWidget(d->alphaTransparencyParameter->getCheckBox()); lutLayout->addWidget(poMinMaxWidget); d->minIntensityParameter->getSlider()->setOrientation(Qt::Horizontal); d->maxIntensityParameter->getSlider()->setOrientation(Qt::Horizontal); QHBoxLayout *minRangeLayout = new QHBoxLayout(); QHBoxLayout *maxRangeLayout = new QHBoxLayout(); minRangeLayout->addWidget(d->minIntensityParameter->getSlider()); minRangeLayout->addWidget(d->minIntensityParameter->getSpinBox()); maxRangeLayout->addWidget(d->maxIntensityParameter->getSlider()); maxRangeLayout->addWidget(d->maxIntensityParameter->getSpinBox()); minMaxLayout->addRow(d->minIntensityParameter->getLabel(), minRangeLayout); minMaxLayout->addRow(d->maxIntensityParameter->getLabel(), maxRangeLayout); toolBoxLayout->addWidget(poGroupBoxLut/*poMinMaxGlobalWidget*/); // Finish LUT and WindowLevel range for it //////////////////////////////////////////////////////////////// toolBoxLayout->addWidget(d->shapesDisplayedGroupParameter->getRadioButtonGroup()); toolBoxLayout->addWidget(d->radiusParameter->getLabel()); d->radiusParameter->getSlider()->setOrientation(Qt::Horizontal); toolBoxLayout->addWidget(d->radiusParameter->getSlider()); d->bundleToolboxWidget = new QWidget(d->toolboxWidget); QVBoxLayout *bundleToolboxLayout = new QVBoxLayout(d->bundleToolboxWidget); toolBoxLayout->addWidget(d->bundleToolboxWidget); d->dropOrOpenRoi = new medDropSite(d->toolboxWidget); d->dropOrOpenRoi->setToolTip(tr("Drag-and-drop A ROI from the database.")); d->dropOrOpenRoi->setText(tr("Drag-and-drop\nfrom the database\nto open a ROI.")); d->dropOrOpenRoi->setCanAutomaticallyChangeAppereance(false); QPushButton *clearRoiButton = new QPushButton("Clear ROI", d->toolboxWidget); clearRoiButton->setToolTip(tr("Clear previously loaded ROIs.")); d->roiComboBox = new QComboBox(d->bundleToolboxWidget); if (!d->roiLabels.isEmpty()) { QMap::const_iterator i = d->roiLabels.constBegin(); while (i != d->roiLabels.constEnd()) { d->roiComboBox->addItem("ROI " + QString::number(i.value())); i++; } d->roiComboBox->setCurrentIndex(0); } d->roiComboBox->setToolTip(tr("Select a ROI to modify how its interaction with the fibers affects whether they are displayed.")); bundleToolboxLayout->addWidget(d->dropOrOpenRoi, 0, Qt::AlignCenter); bundleToolboxLayout->addWidget(clearRoiButton, 0, Qt::AlignCenter); connect (d->dropOrOpenRoi, SIGNAL(objectDropped(const medDataIndex&)), this, SLOT(importROI(const medDataIndex&))); connect (d->dropOrOpenRoi, SIGNAL(clicked()), this, SLOT(loadRoiFromFile())); connect (clearRoiButton, SIGNAL(clicked()), this, SLOT(clearRoi())); connect (d->roiComboBox, SIGNAL(currentIndexChanged(int)), this, SLOT(selectRoi(int))); QHBoxLayout *roiLayout = new QHBoxLayout(d->bundleToolboxWidget); roiLayout->addWidget(d->roiComboBox); d->bundleOperationGroupParameter->setRadioButtonDirection(QBoxLayout::LeftToRight); roiLayout->addWidget(d->bundleOperationGroupParameter->getRadioButtonGroup()); bundleToolboxLayout->addLayout(roiLayout); bundleToolboxLayout->addWidget(d->showBundleBox->getCheckBox()); QHBoxLayout *bundlingLayout = new QHBoxLayout; bundlingLayout->addWidget(d->tagParameter->getPushButton()); bundlingLayout->addWidget(d->addParameter->getPushButton()); bundlingLayout->addWidget(d->resetParameter->getPushButton()); bundlingLayout->addWidget(d->validateParameter->getPushButton()); bundleToolboxLayout->addLayout(bundlingLayout); bundleToolboxLayout->addWidget(d->saveParameter->getPushButton()); d->bundlingList = new QTreeView(d->toolboxWidget); d->bundlingList->setAlternatingRowColors(true); d->bundlingList->setMinimumHeight(150); d->bundlingList->setModel(d->bundlingModel); d->bundlingList->setEditTriggers(QAbstractItemView::SelectedClicked); d->bundlingList->setContextMenuPolicy(Qt::CustomContextMenu); connect(d->bundlingList,SIGNAL(customContextMenuRequested(const QPoint &)), this, SLOT(bundlingListCustomContextMenu(const QPoint &))); bundleToolboxLayout->addWidget(d->bundlingList); bundleToolboxLayout->addWidget(d->showAllBundleParameter->getPushButton()); this->updateWidgets(); return d->toolboxWidget; } void medVtkFibersDataInteractor::bundlingListCustomContextMenu(const QPoint &point) { QMenu *menu = new QMenu; QAction *saveAction = new QAction(tr("Save"), this); saveAction->setIcon(QIcon(":icons/save.png")); connect(saveAction, SIGNAL(triggered()), this, SLOT(saveCurrentBundle())); menu->addAction(saveAction); QAction *removeAction = new QAction(tr("Remove"), this); removeAction->setIcon(QIcon(":icons/cross.svg")); connect(removeAction, SIGNAL(triggered()), this, SLOT(removeCurrentBundle())); menu->addAction(removeAction); menu->exec(QCursor::pos()); } void medVtkFibersDataInteractor::saveCurrentBundle() { QModelIndex index = d->bundlingList->currentIndex(); unsigned int dataIndex = index.row(); if (!d->dataset) return; vtkFiberDataSet::vtkFiberBundleListType bundles = d->dataset->GetBundleList(); vtkFiberDataSet::vtkFiberBundleListType::iterator it = bundles.begin(); unsigned int i = 0; while (i < dataIndex) { i++; it++; } medAbstractData *savedBundle = medAbstractDataFactory::instance()->create("medVtkFibersData"); if (!savedBundle) return; vtkSmartPointer bundle = vtkFiberDataSet::New(); vtkSmartPointer spoTmpFiberBundle = vtkPolyData::New(); spoTmpFiberBundle->DeepCopy((*it).second.Bundle); spoTmpFiberBundle->GetCellData()->SetScalars(d->dataset->GetFibers()->GetCellData()->GetScalars()); bundle->SetFibers(spoTmpFiberBundle); savedBundle->setData(bundle); QString newSeriesDescription = d->data->metadata ( medMetaDataKeys::SeriesDescription.key() ); newSeriesDescription += " "; newSeriesDescription += (*it).first.c_str(); savedBundle->setMetaData ( medMetaDataKeys::SeriesDescription.key(), newSeriesDescription ); for( QString metaData : d->data->metaDataList() ) { if ((metaData == "BundleList")||(metaData == "BundleColorList")) { continue; } if (!savedBundle->hasMetaData(metaData)) { savedBundle->addMetaData (metaData, d->data->metaDataValues (metaData)); } } for( QString property : d->data->propertyList() ) { savedBundle->addProperty ( property,d->data->propertyValues ( property ) ); } QString generatedID = QUuid::createUuid().toString().replace("{","").replace("}",""); savedBundle->setMetaData ( medMetaDataKeys::SeriesID.key(), generatedID ); medDataManager::instance().importData(savedBundle); } void medVtkFibersDataInteractor::removeCurrentBundle() { QModelIndex index = d->bundlingList->currentIndex(); QString bundleName = d->bundlingModel->item(index.row())->data(Qt::UserRole+1).toString(); d->view2d->RemoveLayerActor(d->manager->GetBundleActor(bundleName.toLatin1().constData()),d->view->layer(d->data)); d->view3d->GetRenderer()->RemoveActor(d->manager->GetBundleActor(bundleName.toLatin1().constData())); d->manager->RemoveBundle(bundleName.toLatin1().constData()); // TO DO : better handle bundle list: how to remove metadata from object ? //d->data->addMetaData("BundleList", name); //d->data->addMetaData("BundleColorList", color.name()); // reset to initial navigation state d->manager->Reset(); d->view->render(); d->bundlingModel->removeRow(index.row()); } QWidget* medVtkFibersDataInteractor::buildLayerWidget() { QSlider *slider = d->opacityParam->getSlider(); slider->setOrientation(Qt::Horizontal); return slider; } QWidget* medVtkFibersDataInteractor::buildToolBarWidget() { d->slicingParameter->getSlider()->setOrientation(Qt::Horizontal); return d->slicingParameter->getSlider(); } void medVtkFibersDataInteractor::setUpViewForThumbnail() { d->view->setOrientation(medImageView::VIEW_ORIENTATION_3D); d->view->reset(); d->view3d->ShowAnnotationsOff(); d->view3d->ShowCubeOff(); } QList medVtkFibersDataInteractor::linkableParameters() { return d->parameters; } QList medVtkFibersDataInteractor::mouseInteractionParameters() { // no parameters related to mouse interactions return QList(); } void medVtkFibersDataInteractor::updateWidgets() { if(!d->view->is2D()) { if(!d->bundleToolboxWidget.isNull()) d->bundleToolboxWidget->show(); d->slicingParameter->getSlider()->setEnabled(false); } else { if(!d->bundleToolboxWidget.isNull()) d->bundleToolboxWidget->hide(); d->slicingParameter->getSlider()->setEnabled(true); this->updateSlicingParam(); } } void medVtkFibersDataInteractor::updateSlicingParam() { if(!d->view->is2D()) return; //TODO Should be set according to the real number of slice of this data and // not according to vtkInria (ie. first layer dropped) - RDE d->slicingParameter->blockSignals(true); d->slicingParameter->setRange(d->view2d->GetSliceMin(), d->view2d->GetSliceMax()); d->slicingParameter->blockSignals(false); d->slicingParameter->setValue(d->view2d->GetSlice()); }