QGIS API Documentation 3.28.14-Firenze (exported)
Loading...
Searching...
No Matches
qgsmeshlayerproperties.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsmeshlayerproperties.cpp
3 --------------------------
4 begin : Jun 2018
5 copyright : (C) 2018 by Peter Petrik
6 email : zilolv at gmail dot com
7 ***************************************************************************/
8
9/***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
18#include <limits>
19#include <typeinfo>
20
21#include "qgsapplication.h"
22#include "qgsfileutils.h"
23#include "qgshelp.h"
24#include "qgslogger.h"
25#include "qgsmapcanvas.h"
28#include "qgsmeshlayer.h"
31#include "qgsproject.h"
34#include "qgssettings.h"
37#include "qgsgui.h"
38#include "qgsnative.h"
39#include "qgsmetadatawidget.h"
40
41#include <QDesktopServices>
42#include <QFileDialog>
43#include <QMessageBox>
44
45QgsMeshLayerProperties::QgsMeshLayerProperties( QgsMapLayer *lyr, QgsMapCanvas *canvas, QWidget *parent, Qt::WindowFlags fl )
46 : QgsOptionsDialogBase( QStringLiteral( "MeshLayerProperties" ), parent, fl )
47 , mMeshLayer( qobject_cast<QgsMeshLayer *>( lyr ) )
48 , mCanvas( canvas )
49{
50 Q_ASSERT( mMeshLayer );
51
52 setupUi( this );
53 mRendererMeshPropertiesWidget = new QgsRendererMeshPropertiesWidget( mMeshLayer, canvas, this );
54 mConfigWidgets << mRendererMeshPropertiesWidget;
55 mOptsPage_StyleContent->layout()->addWidget( mRendererMeshPropertiesWidget );
56
57 mSimplifyReductionFactorSpinBox->setClearValue( 10.0 );
58 mSimplifyMeshResolutionSpinBox->setClearValue( 5 );
59
60 mStaticDatasetWidget->setLayer( mMeshLayer );
61 mIsMapSettingsTemporal = mMeshLayer && canvas && canvas->mapSettings().isTemporal();
62
63 mTemporalProviderTimeUnitComboBox->addItem( tr( "Seconds" ), QgsUnitTypes::TemporalSeconds );
64 mTemporalProviderTimeUnitComboBox->addItem( tr( "Minutes" ), QgsUnitTypes::TemporalMinutes );
65 mTemporalProviderTimeUnitComboBox->addItem( tr( "Hours" ), QgsUnitTypes::TemporalHours );
66 mTemporalProviderTimeUnitComboBox->addItem( tr( "Days" ), QgsUnitTypes::TemporalDays );
67
68 connect( mCrsSelector, &QgsProjectionSelectionWidget::crsChanged, this, &QgsMeshLayerProperties::changeCrs );
69 connect( mDatasetGroupTreeWidget, &QgsMeshDatasetGroupTreeWidget::datasetGroupAdded, this, &QgsMeshLayerProperties::syncToLayer );
70
71 // QgsOptionsDialogBase handles saving/restoring of geometry, splitter and current tab states,
72 // switching vertical tabs between icon/text to icon-only modes (splitter collapsed to left),
73 // and connecting QDialogButtonBox's accepted/rejected signals to dialog's accept/reject slots
74 initOptionsBase( false );
75
76 connect( lyr->styleManager(), &QgsMapLayerStyleManager::currentStyleChanged, this, &QgsMeshLayerProperties::syncAndRepaint );
77
78 connect( this, &QDialog::accepted, this, &QgsMeshLayerProperties::apply );
79 connect( this, &QDialog::rejected, this, &QgsMeshLayerProperties::onCancel );
80 connect( buttonBox->button( QDialogButtonBox::Apply ), &QAbstractButton::clicked, this, &QgsMeshLayerProperties::apply );
81
82 connect( mMeshLayer, &QgsMeshLayer::dataChanged, this, &QgsMeshLayerProperties::syncAndRepaint );
83 connect( buttonBox, &QDialogButtonBox::helpRequested, this, &QgsMeshLayerProperties::showHelp );
84
85 connect( mTemporalReloadButton, &QPushButton::clicked, this, &QgsMeshLayerProperties::reloadTemporalProperties );
86 connect( mTemporalDateTimeReference, &QDateTimeEdit::dateTimeChanged, this, &QgsMeshLayerProperties::onTimeReferenceChange );
89
90 mScaleRangeWidget->setMapCanvas( mCanvas );
91 chkUseScaleDependentRendering->setChecked( lyr->hasScaleBasedVisibility() );
92 mScaleRangeWidget->setScaleRange( lyr->minimumScale(), lyr->maximumScale() );
93
94 connect( mAlwaysTimeFromSourceCheckBox, &QCheckBox::stateChanged, this, [this]
95 {
96 mTemporalDateTimeReference->setEnabled( !mAlwaysTimeFromSourceCheckBox->isChecked() );
97 if ( mAlwaysTimeFromSourceCheckBox->isChecked() )
98 reloadTemporalProperties();
99 } );
100
101 mComboBoxTemporalDatasetMatchingMethod->addItem( tr( "Find Closest Dataset Before Requested Time" ),
103 mComboBoxTemporalDatasetMatchingMethod->addItem( tr( "Find Closest Dataset From Requested Time (After or Before)" ),
105
106 QVBoxLayout *layout = new QVBoxLayout( metadataFrame );
107 layout->setContentsMargins( 0, 0, 0, 0 );
108 metadataFrame->setContentsMargins( 0, 0, 0, 0 );
109 mMetadataWidget = new QgsMetadataWidget( this, mMeshLayer );
110 mMetadataWidget->layout()->setContentsMargins( 0, 0, 0, 0 );
111 mMetadataWidget->setMapCanvas( mCanvas );
112 layout->addWidget( mMetadataWidget );
113 metadataFrame->setLayout( layout );
114 mOptsPage_Metadata->setContentsMargins( 0, 0, 0, 0 );
115 mBackupCrs = mMeshLayer->crs();
116
117 mTemporalDateTimeStart->setDisplayFormat( "yyyy-MM-dd HH:mm:ss" );
118 mTemporalDateTimeEnd->setDisplayFormat( "yyyy-MM-dd HH:mm:ss" );
119 mTemporalDateTimeReference->setDisplayFormat( "yyyy-MM-dd HH:mm:ss" );
120
121 // update based on lyr's current state
122 syncToLayer();
123
124 QgsSettings settings;
125 // if dialog hasn't been opened/closed yet, default to Styles tab, which is used most often
126 // this will be read by restoreOptionsBaseUi()
127 if ( !settings.contains( QStringLiteral( "/Windows/MeshLayerProperties/tab" ) ) )
128 {
129 settings.setValue( QStringLiteral( "Windows/MeshLayerProperties/tab" ),
130 mOptStackedWidget->indexOf( mOptsPage_Style ) );
131 }
132
133 //Add help page references
134 mOptsPage_Information->setProperty( "helpPage", QStringLiteral( "working_with_mesh/mesh_properties.html#information-properties" ) );
135 mOptsPage_Source->setProperty( "helpPage", QStringLiteral( "working_with_mesh/mesh_properties.html#source-properties" ) );
136 mOptsPage_Style->setProperty( "helpPage", QStringLiteral( "working_with_mesh/mesh_properties.html#symbology-properties" ) );
137 mOptsPage_Rendering->setProperty( "helpPage", QStringLiteral( "working_with_mesh/mesh_properties.html#rendering-properties" ) );
138 mOptsPage_Temporal->setProperty( "helpPage", QStringLiteral( "working_with_mesh/mesh_properties.html#temporal-properties" ) );
139 mOptsPage_Metadata->setProperty( "helpPage", QStringLiteral( "working_with_mesh/mesh_properties.html#metadata-properties" ) );
140
141 mBtnStyle = new QPushButton( tr( "Style" ) );
142 QMenu *menuStyle = new QMenu( this );
143 menuStyle->addAction( tr( "Load Style…" ), this, &QgsMeshLayerProperties::loadStyle );
144 menuStyle->addAction( tr( "Save Style…" ), this, &QgsMeshLayerProperties::saveStyleAs );
145 menuStyle->addSeparator();
146 menuStyle->addAction( tr( "Save as Default" ), this, &QgsMeshLayerProperties::saveDefaultStyle );
147 menuStyle->addAction( tr( "Restore Default" ), this, &QgsMeshLayerProperties::loadDefaultStyle );
148 mBtnStyle->setMenu( menuStyle );
149 connect( menuStyle, &QMenu::aboutToShow, this, &QgsMeshLayerProperties::aboutToShowStyleMenu );
150
151 buttonBox->addButton( mBtnStyle, QDialogButtonBox::ResetRole );
152
153 mBtnMetadata = new QPushButton( tr( "Metadata" ), this );
154 QMenu *menuMetadata = new QMenu( this );
155 mActionLoadMetadata = menuMetadata->addAction( tr( "Load Metadata…" ), this, &QgsMeshLayerProperties::loadMetadata );
156 mActionSaveMetadataAs = menuMetadata->addAction( tr( "Save Metadata…" ), this, &QgsMeshLayerProperties::saveMetadataAs );
157 mBtnMetadata->setMenu( menuMetadata );
158 buttonBox->addButton( mBtnMetadata, QDialogButtonBox::ResetRole );
159
160 QString title = tr( "Layer Properties — %1" ).arg( lyr->name() );
161
162 if ( !mMeshLayer->styleManager()->isDefault( mMeshLayer->styleManager()->currentStyle() ) )
163 title += QStringLiteral( " (%1)" ).arg( mMeshLayer->styleManager()->currentStyle() );
164 restoreOptionsBaseUi( title );
165}
166
168{
169 if ( !factory->supportsLayer( mMeshLayer ) || !factory->supportLayerPropertiesDialog() )
170 {
171 return;
172 }
173
174 QgsMapLayerConfigWidget *page = factory->createWidget( mMeshLayer, mCanvas, false, this );
175 mConfigWidgets << page;
176
177 const QString beforePage = factory->layerPropertiesPagePositionHint();
178 if ( beforePage.isEmpty() )
179 addPage( factory->title(), factory->title(), factory->icon(), page );
180 else
181 insertPage( factory->title(), factory->title(), factory->icon(), page, beforePage );
182
183 page->syncToLayer( mMeshLayer );
184
185}
186
188{
190
191 bool isMetadataPanel = ( index == mOptStackedWidget->indexOf( mOptsPage_Metadata ) );
192 mBtnStyle->setVisible( ! isMetadataPanel );
193 mBtnMetadata->setVisible( isMetadataPanel );
194}
195
196void QgsMeshLayerProperties::syncToLayer()
197{
198 Q_ASSERT( mRendererMeshPropertiesWidget );
199
200 QgsDebugMsgLevel( QStringLiteral( "populate general information tab" ), 4 );
201 /*
202 * Information Tab
203 */
204 QString myStyle = QgsApplication::reportStyleSheet();
205 myStyle.append( QStringLiteral( "body { margin: 10px; }\n " ) );
206 mInformationTextBrowser->clear();
207 mInformationTextBrowser->document()->setDefaultStyleSheet( myStyle );
208 mInformationTextBrowser->setHtml( mMeshLayer->htmlMetadata() );
209 mInformationTextBrowser->setOpenLinks( false );
210 connect( mInformationTextBrowser, &QTextBrowser::anchorClicked, this, &QgsMeshLayerProperties::urlClicked );
211
212 QgsDebugMsgLevel( QStringLiteral( "populate source tab" ), 4 );
213 /*
214 * Source Tab
215 */
216 mLayerOrigNameLineEd->setText( mMeshLayer->name() );
217 whileBlocking( mCrsSelector )->setCrs( mMeshLayer->crs() );
218
219 mDatasetGroupTreeWidget->syncToLayer( mMeshLayer );
220
221 QgsDebugMsgLevel( QStringLiteral( "populate config tab" ), 4 );
222 for ( QgsMapLayerConfigWidget *w : std::as_const( mConfigWidgets ) )
223 w->syncToLayer( mMeshLayer );
224
225 QgsDebugMsgLevel( QStringLiteral( "populate rendering tab" ), 4 );
226 if ( mMeshLayer->isEditable() )
227 mSimplifyMeshGroupBox->setEnabled( false );
228
229 QgsMeshSimplificationSettings simplifySettings = mMeshLayer->meshSimplificationSettings();
230 mSimplifyMeshGroupBox->setChecked( simplifySettings.isEnabled() );
231 mSimplifyReductionFactorSpinBox->setValue( simplifySettings.reductionFactor() );
232 mSimplifyMeshResolutionSpinBox->setValue( simplifySettings.meshResolution() );
233
234 QgsDebugMsgLevel( QStringLiteral( "populate temporal tab" ), 4 );
235 const QgsMeshLayerTemporalProperties *temporalProperties = qobject_cast< const QgsMeshLayerTemporalProperties * >( mMeshLayer->temporalProperties() );
236 whileBlocking( mTemporalDateTimeReference )->setDateTime( temporalProperties->referenceTime() );
237 const QgsDateTimeRange timeRange = temporalProperties->timeExtent();
238 mTemporalDateTimeStart->setDateTime( timeRange.begin() );
239 mTemporalDateTimeEnd->setDateTime( timeRange.end() );
240 if ( mMeshLayer->dataProvider() )
241 {
242 mTemporalProviderTimeUnitComboBox->setCurrentIndex(
243 mTemporalProviderTimeUnitComboBox->findData( mMeshLayer->dataProvider()->temporalCapabilities()->temporalUnit() ) );
244 }
245 mAlwaysTimeFromSourceCheckBox->setChecked( temporalProperties->alwaysLoadReferenceTimeFromSource() );
246 mComboBoxTemporalDatasetMatchingMethod->setCurrentIndex(
247 mComboBoxTemporalDatasetMatchingMethod->findData( temporalProperties->matchingMethod() ) );
248
249 mStaticDatasetWidget->syncToLayer();
250 mStaticDatasetGroupBox->setChecked( !mMeshLayer->temporalProperties()->isActive() );
251}
252
253void QgsMeshLayerProperties::loadDefaultStyle()
254{
255 bool defaultLoadedFlag = false;
256 QString myMessage = mMeshLayer->loadDefaultStyle( defaultLoadedFlag );
257 // reset if the default style was loaded OK only
258 if ( defaultLoadedFlag )
259 {
260 syncToLayer();
261 }
262 else
263 {
264 // otherwise let the user know what went wrong
265 QMessageBox::information( this,
266 tr( "Default Style" ),
267 myMessage
268 );
269 }
270}
271
272void QgsMeshLayerProperties::saveDefaultStyle()
273{
274 apply(); // make sure the style to save is up-to-date
275
276 // a flag passed by reference
277 bool defaultSavedFlag = false;
278 // TODO Once the deprecated `saveDefaultStyle()` method is gone, just
279 // remove the NOWARN_DEPRECATED tags
281 // after calling this the above flag will be set true for success
282 // or false if the save operation failed
283 QString myMessage = mMeshLayer->saveDefaultStyle( defaultSavedFlag );
285 if ( !defaultSavedFlag )
286 {
287 // let the user know what went wrong
288 QMessageBox::information( this,
289 tr( "Default Style" ),
290 myMessage
291 );
292 }
293}
294
295void QgsMeshLayerProperties::loadStyle()
296{
297 QgsSettings settings;
298 QString lastUsedDir = settings.value( QStringLiteral( "style/lastStyleDir" ), QDir::homePath() ).toString();
299
300 QString fileName = QFileDialog::getOpenFileName(
301 this,
302 tr( "Load rendering setting from style file" ),
303 lastUsedDir,
304 tr( "QGIS Layer Style File" ) + " (*.qml)" );
305 if ( fileName.isEmpty() )
306 return;
307
308 // ensure the user never omits the extension from the file name
309 if ( !fileName.endsWith( QLatin1String( ".qml" ), Qt::CaseInsensitive ) )
310 fileName += QLatin1String( ".qml" );
311
312 mOldStyle = mMeshLayer->styleManager()->style( mMeshLayer->styleManager()->currentStyle() );
313
314 bool defaultLoadedFlag = false;
315 QString message = mMeshLayer->loadNamedStyle( fileName, defaultLoadedFlag );
316 if ( defaultLoadedFlag )
317 {
318 settings.setValue( QStringLiteral( "style/lastStyleDir" ), QFileInfo( fileName ).absolutePath() );
319 syncToLayer();
320 }
321 else
322 {
323 QMessageBox::information( this, tr( "Load Style" ), message );
324 }
325}
326
327void QgsMeshLayerProperties::saveStyleAs()
328{
329 QgsSettings settings;
330 QString lastUsedDir = settings.value( QStringLiteral( "style/lastStyleDir" ), QDir::homePath() ).toString();
331
332 QString outputFileName = QFileDialog::getSaveFileName(
333 this,
334 tr( "Save layer properties as style file" ),
335 lastUsedDir,
336 tr( "QGIS Layer Style File" ) + " (*.qml)" );
337 if ( outputFileName.isEmpty() )
338 return;
339
340 // ensure the user never omits the extension from the file name
341 outputFileName = QgsFileUtils::ensureFileNameHasExtension( outputFileName, QStringList() << QStringLiteral( "qml" ) );
342
343 apply(); // make sure the style to save is up-to-date
344
345 // then export style
346 bool defaultLoadedFlag = false;
347 QString message;
348 message = mMeshLayer->saveNamedStyle( outputFileName, defaultLoadedFlag );
349
350 if ( defaultLoadedFlag )
351 {
352 settings.setValue( QStringLiteral( "style/lastStyleDir" ), QFileInfo( outputFileName ).absolutePath() );
353 }
354 else
355 QMessageBox::information( this, tr( "Save Style" ), message );
356}
357
358void QgsMeshLayerProperties::apply()
359{
360 Q_ASSERT( mRendererMeshPropertiesWidget );
361
362 QgsDebugMsgLevel( QStringLiteral( "processing general tab" ), 4 );
363 /*
364 * General Tab
365 */
366 mMeshLayer->setName( mLayerOrigNameLineEd->text() );
367
368 QgsDebugMsgLevel( QStringLiteral( "processing source tab" ), 4 );
369 /*
370 * Source Tab
371 */
372 mDatasetGroupTreeWidget->apply();
373
374 QgsDebugMsgLevel( QStringLiteral( "processing config tabs" ), 4 );
375
376 for ( QgsMapLayerConfigWidget *w : std::as_const( mConfigWidgets ) )
377 w->apply();
378
379 QgsDebugMsgLevel( QStringLiteral( "processing rendering tab" ), 4 );
380 /*
381 * Rendering Tab
382 */
383 QgsMeshSimplificationSettings simplifySettings;
384 simplifySettings.setEnabled( mSimplifyMeshGroupBox->isChecked() );
385 simplifySettings.setReductionFactor( mSimplifyReductionFactorSpinBox->value() );
386 simplifySettings.setMeshResolution( mSimplifyMeshResolutionSpinBox->value() );
387 bool needMeshUpdating = ( ( simplifySettings.isEnabled() != mMeshLayer->meshSimplificationSettings().isEnabled() ) ||
388 ( simplifySettings.reductionFactor() != mMeshLayer->meshSimplificationSettings().reductionFactor() ) );
389
390 mMeshLayer->setMeshSimplificationSettings( simplifySettings );
391
392 mMeshLayer->setScaleBasedVisibility( chkUseScaleDependentRendering->isChecked() );
393 mMeshLayer->setMinimumScale( mScaleRangeWidget->minimumScale() );
394 mMeshLayer->setMaximumScale( mScaleRangeWidget->maximumScale() );
395
396 QgsDebugMsgLevel( QStringLiteral( "processing temporal tab" ), 4 );
397 /*
398 * Temporal Tab
399 */
400
401 mMeshLayer->setReferenceTime( mTemporalDateTimeReference->dateTime() );
402 if ( mMeshLayer->dataProvider() )
403 mMeshLayer->dataProvider()->setTemporalUnit(
404 static_cast<QgsUnitTypes::TemporalUnit>( mTemporalProviderTimeUnitComboBox->currentData().toInt() ) );
405
406 mStaticDatasetWidget->apply();
407 bool needEmitRendererChanged = mMeshLayer->temporalProperties()->isActive() == mStaticDatasetGroupBox->isChecked();
408 mMeshLayer->temporalProperties()->setIsActive( !mStaticDatasetGroupBox->isChecked() );
410 mComboBoxTemporalDatasetMatchingMethod->currentData().toInt() ) );
411 static_cast<QgsMeshLayerTemporalProperties *>(
412 mMeshLayer->temporalProperties() )->setAlwaysLoadReferenceTimeFromSource( mAlwaysTimeFromSourceCheckBox->isChecked() );
413
414 mMetadataWidget->acceptMetadata();
415
416 mBackupCrs = mMeshLayer->crs();
417
418 if ( needMeshUpdating )
419 mMeshLayer->reload();
420
421 if ( needEmitRendererChanged )
422 emit mMeshLayer->rendererChanged();
423
424 //make sure the layer is redrawn
425 mMeshLayer->triggerRepaint();
426
427 // notify the project we've made a change
429
430 // Resync what have to be resync (widget that can be changed by other properties part)
431 mStaticDatasetWidget->syncToLayer();
432 for ( QgsMapLayerConfigWidget *w : std::as_const( mConfigWidgets ) )
433 w->syncToLayer( mMeshLayer );
434}
435
436void QgsMeshLayerProperties::changeCrs( const QgsCoordinateReferenceSystem &crs )
437{
438 QgsDatumTransformDialog::run( crs, QgsProject::instance()->crs(), this, mCanvas, tr( "Select Transformation" ) );
439 mMeshLayer->setCrs( crs );
440}
441
442void QgsMeshLayerProperties::syncAndRepaint()
443{
444 syncToLayer();
445 mMeshLayer->triggerRepaint();
446}
447
448void QgsMeshLayerProperties::showHelp()
449{
450 const QVariant helpPage = mOptionsStackedWidget->currentWidget()->property( "helpPage" );
451
452 if ( helpPage.isValid() )
453 {
454 QgsHelp::openHelp( helpPage.toString() );
455 }
456 else
457 {
458 QgsHelp::openHelp( QStringLiteral( "working_with_mesh/mesh_properties.html" ) );
459 }
460}
461
462void QgsMeshLayerProperties::aboutToShowStyleMenu()
463{
464 QMenu *m = qobject_cast<QMenu *>( sender() );
465
467 // re-add style manager actions!
468 m->addSeparator();
470}
471
472void QgsMeshLayerProperties::reloadTemporalProperties()
473{
474 if ( !mMeshLayer->dataProvider() )
475 return;
476 QgsMeshDataProviderTemporalCapabilities *temporalCapabalities = mMeshLayer->dataProvider()->temporalCapabilities();
477 QgsDateTimeRange timeExtent;
478 QDateTime referenceTime = temporalCapabalities->referenceTime();
479 if ( referenceTime.isValid() )
480 {
481 timeExtent = temporalCapabalities->timeExtent();
482 whileBlocking( mTemporalDateTimeReference )->setDateTime( referenceTime );
483 }
484 else
485 // The reference time already here is used again to define the time extent
486 timeExtent = temporalCapabalities->timeExtent( mTemporalDateTimeReference->dateTime() );
487
488 mTemporalDateTimeStart->setDateTime( timeExtent.begin() );
489 mTemporalDateTimeEnd->setDateTime( timeExtent.end() );
490}
491
492void QgsMeshLayerProperties::onTimeReferenceChange()
493{
494 if ( !mMeshLayer->dataProvider() )
495 return;
496 const QgsDateTimeRange &timeExtent = mMeshLayer->dataProvider()->temporalCapabilities()->timeExtent( mTemporalDateTimeReference->dateTime() );
497 mTemporalDateTimeStart->setDateTime( timeExtent.begin() );
498 mTemporalDateTimeEnd->setDateTime( timeExtent.end() );
499}
500
501void QgsMeshLayerProperties::urlClicked( const QUrl &url )
502{
503 QFileInfo file( url.toLocalFile() );
504 if ( file.exists() && !file.isDir() )
505 QgsGui::nativePlatformInterface()->openFileExplorerAndSelectFile( url.toLocalFile() );
506 else
507 QDesktopServices::openUrl( url );
508}
509
510void QgsMeshLayerProperties::loadMetadata()
511{
512 QgsSettings myQSettings; // where we keep last used filter in persistent state
513 QString myLastUsedDir = myQSettings.value( QStringLiteral( "style/lastStyleDir" ), QDir::homePath() ).toString();
514
515 QString myFileName = QFileDialog::getOpenFileName( this, tr( "Load layer metadata from metadata file" ), myLastUsedDir,
516 tr( "QGIS Layer Metadata File" ) + " (*.qmd)" );
517 if ( myFileName.isNull() )
518 {
519 return;
520 }
521
522 QString myMessage;
523 bool defaultLoadedFlag = false;
524 myMessage = mMeshLayer->loadNamedMetadata( myFileName, defaultLoadedFlag );
525
526 //reset if the default style was loaded OK only
527 if ( defaultLoadedFlag )
528 {
529 mMetadataWidget->setMetadata( &mMeshLayer->metadata() );
530 }
531 else
532 {
533 //let the user know what went wrong
534 QMessageBox::warning( this, tr( "Load Metadata" ), myMessage );
535 }
536
537 QFileInfo myFI( myFileName );
538 QString myPath = myFI.path();
539 myQSettings.setValue( QStringLiteral( "style/lastStyleDir" ), myPath );
540
541 activateWindow(); // set focus back to properties dialog
542}
543
544void QgsMeshLayerProperties::saveMetadataAs()
545{
546 QgsSettings myQSettings; // where we keep last used filter in persistent state
547 QString myLastUsedDir = myQSettings.value( QStringLiteral( "style/lastStyleDir" ), QDir::homePath() ).toString();
548
549 QString myOutputFileName = QFileDialog::getSaveFileName( this, tr( "Save Layer Metadata as QMD" ),
550 myLastUsedDir, tr( "QMD File" ) + " (*.qmd)" );
551 if ( myOutputFileName.isNull() ) //dialog canceled
552 {
553 return;
554 }
555
556 mMetadataWidget->acceptMetadata();
557
558 //ensure the user never omitted the extension from the file name
559 if ( !myOutputFileName.endsWith( QgsMapLayer::extensionPropertyType( QgsMapLayer::Metadata ), Qt::CaseInsensitive ) )
560 {
562 }
563
564 bool defaultLoadedFlag = false;
565 QString message = mMeshLayer->saveNamedMetadata( myOutputFileName, defaultLoadedFlag );
566 if ( defaultLoadedFlag )
567 myQSettings.setValue( QStringLiteral( "style/lastStyleDir" ), QFileInfo( myOutputFileName ).absolutePath() );
568 else
569 QMessageBox::information( this, tr( "Save Metadata" ), message );
570}
571
572void QgsMeshLayerProperties::onCancel()
573{
574 if ( mBackupCrs != mMeshLayer->crs() )
575 mMeshLayer->setCrs( mBackupCrs );
576}
static QString reportStyleSheet(QgsApplication::StyleSheetType styleSheetType=QgsApplication::StyleSheetType::Qt)
Returns a css style sheet for reports, the styleSheetType argument determines what type of stylesheet...
This class represents a coordinate reference system (CRS).
static bool run(const QgsCoordinateReferenceSystem &sourceCrs=QgsCoordinateReferenceSystem(), const QgsCoordinateReferenceSystem &destinationCrs=QgsCoordinateReferenceSystem(), QWidget *parent=nullptr, QgsMapCanvas *mapCanvas=nullptr, const QString &windowTitle=QString())
Runs the dialog (if required) prompting for the desired transform to use from sourceCrs to destinatio...
static QString ensureFileNameHasExtension(const QString &fileName, const QStringList &extensions)
Ensures that a fileName ends with an extension from the provided list of extensions.
static QgsNative * nativePlatformInterface()
Returns the global native interface, which offers abstraction to the host OS's underlying public inte...
Definition qgsgui.cpp:73
static void openHelp(const QString &key)
Opens help topic for the given help key using default system web browser.
Definition qgshelp.cpp:38
Map canvas is a class for displaying all GIS data types on a canvas.
const QgsMapSettings & mapSettings() const
Gets access to properties used for map rendering.
Factory class for creating custom map layer property pages.
virtual bool supportsLayer(QgsMapLayer *layer) const
Check if the layer is supported for this widget.
virtual QIcon icon() const
The icon that will be shown in the UI for the panel.
virtual QgsMapLayerConfigWidget * createWidget(QgsMapLayer *layer, QgsMapCanvas *canvas, bool dockWidget=true, QWidget *parent=nullptr) const =0
Factory function to create the widget on demand as needed by the dock.
virtual QString title() const
The title of the panel.
virtual bool supportLayerPropertiesDialog() const
Flag if widget is supported for use in layer properties dialog.
virtual QString layerPropertiesPagePositionHint() const
Returns a tab name hinting at where this page should be inserted into the layer properties tab list.
A panel widget that can be shown in the map style dock.
virtual void syncToLayer(QgsMapLayer *layer)
Reset to original (vector layer) values.
void removesExtraMenuSeparators(QMenu *m)
removes extra separators from the menu
void addStyleManagerActions(QMenu *m, QgsMapLayer *layer)
adds actions to the menu in accordance to the layer
static QgsMapLayerStyleGuiUtils * instance()
returns a singleton instance of this class
QString currentStyle() const
Returns name of the current style.
static bool isDefault(const QString &styleName)
Returns true if this is the default style.
QgsMapLayerStyle style(const QString &name) const
Returns data of a stored style - accessed by its unique name.
void currentStyleChanged(const QString &currentName)
Emitted when the current style has been changed.
Base class for all map layer types.
Definition qgsmaplayer.h:73
QString name
Definition qgsmaplayer.h:76
void setMinimumScale(double scale)
Sets the minimum map scale (i.e.
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:79
virtual QString loadNamedMetadata(const QString &uri, bool &resultFlag)
Retrieve a named metadata for this layer if one exists (either as a .qmd file on disk or as a record ...
void triggerRepaint(bool deferredUpdate=false)
Will advise the map canvas (and any other interested party) that this layer requires to be repainted.
void setMaximumScale(double scale)
Sets the maximum map scale (i.e.
QgsLayerMetadata metadata
Definition qgsmaplayer.h:78
void rendererChanged()
Signal emitted when renderer is changed.
void setScaleBasedVisibility(bool enabled)
Sets whether scale based visibility is enabled for the layer.
bool hasScaleBasedVisibility() const
Returns whether scale based visibility is enabled for the layer.
virtual QString loadNamedStyle(const QString &uri, bool &resultFlag, QgsMapLayer::StyleCategories categories=QgsMapLayer::AllStyleCategories)
Retrieve a named style for this layer if one exists (either as a .qml file on disk or as a record in ...
void dataChanged()
Data of layer changed.
static QString extensionPropertyType(PropertyType type)
Returns the extension of a Property.
void setName(const QString &name)
Set the display name of the layer.
QString saveNamedMetadata(const QString &uri, bool &resultFlag)
Save the current metadata of this layer as a named metadata (either as a .qmd file on disk or as a re...
double minimumScale() const
Returns the minimum map scale (i.e.
QgsMapLayerStyleManager * styleManager() const
Gets access to the layer's style manager.
virtual QString saveNamedStyle(const QString &uri, bool &resultFlag, StyleCategories categories=AllStyleCategories)
Save the properties of this layer as a named style (either as a .qml file on disk or as a record in t...
virtual QString saveDefaultStyle(bool &resultFlag, StyleCategories categories)
Save the properties of this layer as the default style (either as a .qml file on disk or as a record ...
double maximumScale() const
Returns the maximum map scale (i.e.
void setCrs(const QgsCoordinateReferenceSystem &srs, bool emitSignal=true)
Sets layer's spatial reference system.
Class for handling properties relating to a mesh data provider's temporal capabilities.
QgsUnitTypes::TemporalUnit temporalUnit() const
Returns the temporal unit used to read data by the data provider.
QDateTime referenceTime() const
Returns the reference time.
QgsDateTimeRange timeExtent() const
Returns the time extent using the internal reference time and the first and last times available from...
MatchingTemporalDatasetMethod
Method for selection of temporal mesh dataset from a range time.
@ FindClosestDatasetFromStartRangeTime
Finds the closest dataset which have its time before the requested start range time.
QgsMeshDataProviderTemporalCapabilities * temporalCapabilities() override
Returns the provider's temporal capabilities.
void setTemporalUnit(QgsUnitTypes::TemporalUnit unit)
Sets the temporal unit of the provider and reload data if it changes.
void optionsStackedWidget_CurrentChanged(int index) override
QgsMeshLayerProperties(QgsMapLayer *lyr, QgsMapCanvas *canvas, QWidget *parent=nullptr, Qt::WindowFlags=QgsGuiUtils::ModalDialogFlags)
Constructor.
void addPropertiesPageFactory(const QgsMapLayerConfigWidgetFactory *factory)
Adds properties page from a factory.
Implementation of map layer temporal properties for mesh layers.
QDateTime referenceTime() const
Returns the reference time.
QgsMeshDataProviderTemporalCapabilities::MatchingTemporalDatasetMethod matchingMethod() const
Returns the method used to match dataset from temporal capabilities.
bool alwaysLoadReferenceTimeFromSource() const
Returns whether the time proporties are automatically reloaded from provider when project is opened o...
QgsDateTimeRange timeExtent() const
Returns the time extent.
Represents a mesh layer supporting display of data on structured or unstructured meshes.
void setMeshSimplificationSettings(const QgsMeshSimplificationSettings &meshSimplificationSettings)
Sets mesh simplification settings.
void activeScalarDatasetGroupChanged(int index)
Emitted when active scalar group dataset is changed.
void activeVectorDatasetGroupChanged(int index)
Emitted when active vector group dataset is changed.
void reload() override
Synchronises with changes in the datasource.
QString loadDefaultStyle(bool &resultFlag) FINAL
Retrieve the default style for this layer if one exists (either as a .qml file on disk or as a record...
QString htmlMetadata() const override
Obtain a formatted HTML string containing assorted metadata for this layer.
QgsMeshSimplificationSettings meshSimplificationSettings() const
Returns mesh simplification settings.
bool isEditable() const override
Returns true if the layer can be edited.
QgsMeshDataProvider * dataProvider() override
Returns the layer's data provider, it may be nullptr.
QgsMapLayerTemporalProperties * temporalProperties() override
Returns the layer's temporal properties.
void setReferenceTime(const QDateTime &referenceTime)
Sets the reference time of the layer.
void setTemporalMatchingMethod(const QgsMeshDataProviderTemporalCapabilities::MatchingTemporalDatasetMethod &matchingMethod)
Sets the method used to match the temporal dataset from a requested time, see activeVectorDatasetAtTi...
Represents an overview renderer settings.
void setMeshResolution(int meshResolution)
Sets the mesh resolution i.e., the minimum size (average) of triangles in pixels This value is used d...
void setEnabled(bool isEnabled)
Sets if the overview is active.
double reductionFactor() const
Returns the reduction factor used to build simplified mesh.
bool isEnabled() const
Returns if the overview is active.
int meshResolution() const
Returns the mesh resolution i.e., the minimum size (average) of triangles in pixels.
void setReductionFactor(double value)
Sets the reduction factor used to build simplified mesh.
void setScalarDatasetGroup(int index)
Sets the scalar dataset group.
void setVectorDatasetGroup(int index)
Sets the vector dataset group.
A wizard to edit metadata on a map layer.
void acceptMetadata()
Saves the metadata to the layer.
void setMapCanvas(QgsMapCanvas *canvas)
Sets a map canvas associated with the widget.
void setMetadata(const QgsAbstractMetadataBase *metadata)
Sets the metadata to display in the widget.
A base dialog for options and properties dialogs that offers vertical tabs.
void addPage(const QString &title, const QString &tooltip, const QIcon &icon, QWidget *widget, const QStringList &path=QStringList())
Adds a new page to the dialog pages.
virtual void optionsStackedWidget_CurrentChanged(int index)
Select relevant tab on current page change.
void insertPage(const QString &title, const QString &tooltip, const QIcon &icon, QWidget *widget, const QString &before, const QStringList &path=QStringList())
Inserts a new page into the dialog pages.
void restoreOptionsBaseUi(const QString &title=QString())
Restore the base ui.
QStackedWidget * mOptStackedWidget
void initOptionsBase(bool restoreUi=true, const QString &title=QString())
Set up the base ui connections for vertical tabs.
static QgsProject * instance()
Returns the QgsProject singleton instance.
void setDirty(bool b=true)
Flag the project as dirty (modified).
void crsChanged(const QgsCoordinateReferenceSystem &)
Emitted when the selected CRS is changed.
Widget for renderer properties of the mesh, contours (scalars) and vectors data associated with the m...
This class is a composition of two QSettings instances:
Definition qgssettings.h:62
QVariant value(const QString &key, const QVariant &defaultValue=QVariant(), Section section=NoSection) const
Returns the value for setting key.
bool contains(const QString &key, QgsSettings::Section section=QgsSettings::NoSection) const
Returns true if there exists a setting called key; returns false otherwise.
void setValue(const QString &key, const QVariant &value, QgsSettings::Section section=QgsSettings::NoSection)
Sets the value of setting key to value.
bool isActive() const
Returns true if the temporal property is active.
void setIsActive(bool active)
Sets whether the temporal property is active.
bool isTemporal() const
Returns true if the object's temporal range is enabled, and the object will be filtered when renderin...
T begin() const
Returns the beginning of the range.
Definition qgsrange.h:394
T end() const
Returns the upper bound of the range.
Definition qgsrange.h:401
TemporalUnit
Temporal units.
@ TemporalDays
Days.
@ TemporalSeconds
Seconds.
@ TemporalMinutes
Minutes.
@ TemporalHours
Hours.
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:3061
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:3060
QgsSignalBlocker< Object > whileBlocking(Object *object)
Temporarily blocks signals from a QObject while calling a single method from the object.
Definition qgis.h:2453
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:39
const QgsCoordinateReferenceSystem & crs