mirror of
https://github.com/qgis/QGIS.git
synced 2025-04-05 00:05:32 -04:00
Merge pull request #4380 from nyalldawson/clone_annotations
Add methods to clone annotations
This commit is contained in:
commit
6ecb17f1fd
@ -25,6 +25,8 @@ class QgsAnnotation : QObject
|
||||
|
||||
QgsAnnotation( QObject* parent /TransferThis/ = nullptr );
|
||||
|
||||
virtual QgsAnnotation *clone() const = 0 /Factory/;
|
||||
|
||||
bool isVisible() const;
|
||||
void setVisible( bool visible );
|
||||
|
||||
@ -81,4 +83,6 @@ class QgsAnnotation : QObject
|
||||
void _writeXml( QDomElement& itemElem, QDomDocument& doc ) const;
|
||||
void _readXml( const QDomElement& annotationElem, const QDomDocument& doc );
|
||||
|
||||
void copyCommonProperties( QgsAnnotation *target ) const;
|
||||
|
||||
};
|
||||
|
@ -68,6 +68,16 @@ class QgsAnnotationManager : QObject
|
||||
QList< QgsAnnotation * > annotations() const;
|
||||
%Docstring
|
||||
Returns a list of all annotations contained in the manager.
|
||||
\see cloneAnnotations()
|
||||
:rtype: list of QgsAnnotation
|
||||
%End
|
||||
|
||||
QList< QgsAnnotation * > cloneAnnotations() const /Factory/;
|
||||
%Docstring
|
||||
Returns a list containing clones of all annotations contained
|
||||
in the manager. The caller takes responsibility for deleting
|
||||
all returned annotations.
|
||||
\see annotations()
|
||||
:rtype: list of QgsAnnotation
|
||||
%End
|
||||
|
||||
|
@ -31,6 +31,8 @@ class QgsHtmlAnnotation: QgsAnnotation
|
||||
|
||||
~QgsHtmlAnnotation();
|
||||
|
||||
virtual QgsHtmlAnnotation *clone() const /Factory/;
|
||||
|
||||
virtual QSizeF minimumFrameSize() const;
|
||||
|
||||
void setSourceFile( const QString &htmlFile );
|
||||
|
@ -26,6 +26,8 @@ class QgsSvgAnnotation: QgsAnnotation
|
||||
Constructor for QgsSvgAnnotation.
|
||||
%End
|
||||
|
||||
virtual QgsSvgAnnotation *clone() const /Factory/;
|
||||
|
||||
virtual void writeXml( QDomElement &elem, QDomDocument &doc ) const;
|
||||
virtual void readXml( const QDomElement &itemElem, const QDomDocument &doc );
|
||||
|
||||
|
@ -26,6 +26,8 @@ class QgsTextAnnotation: QgsAnnotation
|
||||
Constructor for QgsTextAnnotation.
|
||||
%End
|
||||
|
||||
virtual QgsTextAnnotation *clone() const /Factory/;
|
||||
|
||||
const QTextDocument *document() const;
|
||||
%Docstring
|
||||
Returns the text document which will be rendered
|
||||
|
@ -7,6 +7,8 @@ class QgsFormAnnotation : QgsAnnotation
|
||||
|
||||
QgsFormAnnotation( QObject* parent /TransferThis/ = nullptr );
|
||||
|
||||
virtual QgsFormAnnotation *clone() const /Factory/;
|
||||
|
||||
QSizeF minimumFrameSize() const;
|
||||
QSizeF preferredFrameSize() const;
|
||||
|
||||
|
@ -431,3 +431,22 @@ void QgsAnnotation::_readXml( const QDomElement &annotationElem, const QDomDocum
|
||||
emit mapLayerChanged();
|
||||
}
|
||||
|
||||
void QgsAnnotation::copyCommonProperties( QgsAnnotation *target ) const
|
||||
{
|
||||
target->mVisible = mVisible;
|
||||
target->mHasFixedMapPosition = mHasFixedMapPosition;
|
||||
target->mMapPosition = mMapPosition;
|
||||
target->mMapPositionCrs = mMapPositionCrs;
|
||||
target->mRelativePosition = mRelativePosition;
|
||||
target->mOffsetFromReferencePoint = mOffsetFromReferencePoint;
|
||||
target->mFrameSize = mFrameSize;
|
||||
target->mMarkerSymbol.reset( mMarkerSymbol ? mMarkerSymbol->clone() : nullptr );
|
||||
target->mContentsMargins = mContentsMargins;
|
||||
target->mFillSymbol.reset( mFillSymbol ? mFillSymbol->clone() : nullptr );
|
||||
target->mBalloonSegment = mBalloonSegment;
|
||||
target->mBalloonSegmentPoint1 = mBalloonSegmentPoint1;
|
||||
target->mBalloonSegmentPoint2 = mBalloonSegmentPoint2;
|
||||
target->mMapLayer = mMapLayer;
|
||||
target->mFeature = mFeature;
|
||||
}
|
||||
|
||||
|
@ -57,6 +57,12 @@ class CORE_EXPORT QgsAnnotation : public QObject
|
||||
*/
|
||||
QgsAnnotation( QObject *parent = nullptr );
|
||||
|
||||
/**
|
||||
* Clones the annotation, returning a new copy of the annotation
|
||||
* reflecting the annotation's current state.
|
||||
*/
|
||||
virtual QgsAnnotation *clone() const = 0;
|
||||
|
||||
/**
|
||||
* Returns true if the annotation is visible and should be rendered.
|
||||
* \see setVisible()
|
||||
@ -293,6 +299,14 @@ class CORE_EXPORT QgsAnnotation : public QObject
|
||||
*/
|
||||
void _readXml( const QDomElement &annotationElem, const QDomDocument &doc );
|
||||
|
||||
/**
|
||||
* Copies common annotation properties to the \a targe
|
||||
* annotation.
|
||||
* Can be used within QgsAnnotation::clone() implementations
|
||||
* to assist with creating copies.
|
||||
*/
|
||||
void copyCommonProperties( QgsAnnotation *target ) const;
|
||||
|
||||
private:
|
||||
|
||||
//! Check where to attach the balloon connection between frame and map point
|
||||
|
@ -73,6 +73,16 @@ QList<QgsAnnotation *> QgsAnnotationManager::annotations() const
|
||||
return mAnnotations;
|
||||
}
|
||||
|
||||
QList<QgsAnnotation *> QgsAnnotationManager::cloneAnnotations() const
|
||||
{
|
||||
QList<QgsAnnotation *> results;
|
||||
Q_FOREACH ( const QgsAnnotation *a, mAnnotations )
|
||||
{
|
||||
results << a->clone();
|
||||
}
|
||||
return results;
|
||||
}
|
||||
|
||||
bool QgsAnnotationManager::readXml( const QDomElement &element, const QDomDocument &doc )
|
||||
{
|
||||
clear();
|
||||
|
@ -79,9 +79,18 @@ class CORE_EXPORT QgsAnnotationManager : public QObject
|
||||
|
||||
/**
|
||||
* Returns a list of all annotations contained in the manager.
|
||||
* \see cloneAnnotations()
|
||||
*/
|
||||
QList< QgsAnnotation * > annotations() const;
|
||||
|
||||
/**
|
||||
* Returns a list containing clones of all annotations contained
|
||||
* in the manager. The caller takes responsibility for deleting
|
||||
* all returned annotations.
|
||||
* \see annotations()
|
||||
*/
|
||||
QList< QgsAnnotation * > cloneAnnotations() const SIP_FACTORY;
|
||||
|
||||
/**
|
||||
* Reads the manager's state from a DOM element, restoring all annotations
|
||||
* present in the XML document.
|
||||
|
@ -47,6 +47,14 @@ QgsHtmlAnnotation::QgsHtmlAnnotation( QObject *parent )
|
||||
connect( mWebPage->mainFrame(), &QWebFrame::javaScriptWindowObjectCleared, this, &QgsHtmlAnnotation::javascript );
|
||||
}
|
||||
|
||||
QgsHtmlAnnotation *QgsHtmlAnnotation::clone() const
|
||||
{
|
||||
std::unique_ptr< QgsHtmlAnnotation > c( new QgsHtmlAnnotation() );
|
||||
copyCommonProperties( c.get() );
|
||||
c->setSourceFile( mHtmlFile );
|
||||
return c.release();
|
||||
}
|
||||
|
||||
void QgsHtmlAnnotation::setSourceFile( const QString &htmlFile )
|
||||
{
|
||||
QFile file( htmlFile );
|
||||
|
@ -44,6 +44,8 @@ class CORE_EXPORT QgsHtmlAnnotation: public QgsAnnotation
|
||||
|
||||
~QgsHtmlAnnotation() = default;
|
||||
|
||||
QgsHtmlAnnotation *clone() const override SIP_FACTORY;
|
||||
|
||||
QSizeF minimumFrameSize() const override;
|
||||
|
||||
/**
|
||||
|
@ -27,6 +27,14 @@ QgsSvgAnnotation::QgsSvgAnnotation( QObject *parent )
|
||||
|
||||
}
|
||||
|
||||
QgsSvgAnnotation *QgsSvgAnnotation::clone() const
|
||||
{
|
||||
std::unique_ptr< QgsSvgAnnotation > c( new QgsSvgAnnotation() );
|
||||
copyCommonProperties( c.get() );
|
||||
c->setFilePath( mFilePath );
|
||||
return c.release();
|
||||
}
|
||||
|
||||
void QgsSvgAnnotation::writeXml( QDomElement &elem, QDomDocument &doc ) const
|
||||
{
|
||||
QDomElement svgAnnotationElem = doc.createElement( QStringLiteral( "SVGAnnotationItem" ) );
|
||||
|
@ -39,6 +39,8 @@ class CORE_EXPORT QgsSvgAnnotation: public QgsAnnotation
|
||||
*/
|
||||
QgsSvgAnnotation( QObject *parent SIP_TRANSFERTHIS = nullptr );
|
||||
|
||||
QgsSvgAnnotation *clone() const override SIP_FACTORY;
|
||||
|
||||
virtual void writeXml( QDomElement &elem, QDomDocument &doc ) const override;
|
||||
virtual void readXml( const QDomElement &itemElem, const QDomDocument &doc ) override;
|
||||
|
||||
|
@ -26,6 +26,14 @@ QgsTextAnnotation::QgsTextAnnotation( QObject *parent )
|
||||
mDocument->setUseDesignMetrics( true );
|
||||
}
|
||||
|
||||
QgsTextAnnotation *QgsTextAnnotation::clone() const
|
||||
{
|
||||
std::unique_ptr< QgsTextAnnotation > c( new QgsTextAnnotation() );
|
||||
copyCommonProperties( c.get() );
|
||||
c->setDocument( mDocument ? mDocument->clone() : nullptr );
|
||||
return c.release();
|
||||
}
|
||||
|
||||
const QTextDocument *QgsTextAnnotation::document() const
|
||||
{
|
||||
return mDocument.get();
|
||||
|
@ -39,6 +39,8 @@ class CORE_EXPORT QgsTextAnnotation: public QgsAnnotation
|
||||
*/
|
||||
QgsTextAnnotation( QObject *parent SIP_TRANSFERTHIS = nullptr );
|
||||
|
||||
QgsTextAnnotation *clone() const override SIP_FACTORY;
|
||||
|
||||
/**
|
||||
* Returns the text document which will be rendered
|
||||
* within the annotation.
|
||||
|
@ -40,6 +40,14 @@ QgsFormAnnotation::QgsFormAnnotation( QObject *parent )
|
||||
: QgsAnnotation( parent )
|
||||
{}
|
||||
|
||||
QgsFormAnnotation *QgsFormAnnotation::clone() const
|
||||
{
|
||||
std::unique_ptr< QgsFormAnnotation > c( new QgsFormAnnotation() );
|
||||
copyCommonProperties( c.get() );
|
||||
c->setDesignerForm( mDesignerForm );
|
||||
return c.release();
|
||||
}
|
||||
|
||||
void QgsFormAnnotation::setDesignerForm( const QString &uiFile )
|
||||
{
|
||||
mDesignerForm = uiFile;
|
||||
|
@ -39,6 +39,8 @@ class GUI_EXPORT QgsFormAnnotation: public QgsAnnotation
|
||||
*/
|
||||
QgsFormAnnotation( QObject *parent = nullptr );
|
||||
|
||||
QgsFormAnnotation *clone() const override SIP_FACTORY;
|
||||
|
||||
QSizeF minimumFrameSize() const override;
|
||||
//! Returns the optimal frame size
|
||||
QSizeF preferredFrameSize() const;
|
||||
@ -63,7 +65,7 @@ class GUI_EXPORT QgsFormAnnotation: public QgsAnnotation
|
||||
/**
|
||||
* Returns a new QgsFormAnnotation object.
|
||||
*/
|
||||
static QgsFormAnnotation *create() { return new QgsFormAnnotation(); }
|
||||
static QgsFormAnnotation *create() SIP_FACTORY { return new QgsFormAnnotation(); }
|
||||
|
||||
protected:
|
||||
|
||||
|
15
tests/src/python/test_qgsannotation.py
Normal file → Executable file
15
tests/src/python/test_qgsannotation.py
Normal file → Executable file
@ -63,6 +63,11 @@ class TestQgsAnnotation(unittest.TestCase):
|
||||
im = self.renderAnnotation(a, QPointF(20, 30))
|
||||
self.assertTrue(self.imageCheck('text_annotation', 'text_annotation', im))
|
||||
|
||||
# check clone
|
||||
clone = a.clone()
|
||||
im = self.renderAnnotation(a, QPointF(20, 30))
|
||||
self.assertTrue(self.imageCheck('text_annotation', 'text_annotation', im))
|
||||
|
||||
def testSvgAnnotation(self):
|
||||
""" test rendering a svg annotation"""
|
||||
a = QgsSvgAnnotation()
|
||||
@ -73,6 +78,11 @@ class TestQgsAnnotation(unittest.TestCase):
|
||||
im = self.renderAnnotation(a, QPointF(20, 30))
|
||||
self.assertTrue(self.imageCheck('svg_annotation', 'svg_annotation', im))
|
||||
|
||||
# check clone
|
||||
clone = a.clone()
|
||||
im = self.renderAnnotation(a, QPointF(20, 30))
|
||||
self.assertTrue(self.imageCheck('svg_annotation', 'svg_annotation', im))
|
||||
|
||||
def testHtmlAnnotation(self):
|
||||
""" test rendering a html annotation"""
|
||||
a = QgsHtmlAnnotation()
|
||||
@ -83,6 +93,11 @@ class TestQgsAnnotation(unittest.TestCase):
|
||||
im = self.renderAnnotation(a, QPointF(20, 30))
|
||||
self.assertTrue(self.imageCheck('html_annotation', 'html_annotation', im))
|
||||
|
||||
# check clone
|
||||
clone = a.clone()
|
||||
im = self.renderAnnotation(a, QPointF(20, 30))
|
||||
self.assertTrue(self.imageCheck('html_annotation', 'html_annotation', im))
|
||||
|
||||
def testHtmlAnnotationWithFeature(self):
|
||||
""" test rendering a html annotation with a feature"""
|
||||
layer = QgsVectorLayer("Point?crs=EPSG:3111&field=station:string&field=suburb:string",
|
||||
|
Loading…
x
Reference in New Issue
Block a user