/* LabelModel.cpp * * Copyright (C) 2013-2016 Jim Evins * * This file is part of gLabels-qt. * * gLabels-qt is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * gLabels-qt is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with gLabels-qt. If not, see . */ #include "LabelModel.h" #include #include #include #include #include #include "LabelModelObject.h" #include "LabelModelTextObject.h" #include "LabelModelImageObject.h" #include "Region.h" #include "Size.h" #include "XmlLabelCreator.h" #include "XmlLabelParser.h" #include "Merge/None.h" namespace glabels { // // Private // namespace { const QString MIME_TYPE = "application/x-glabels-objects"; } /// /// Default constructor. /// LabelModel::LabelModel() : mUntitledInstance(0), mModified(true), mTmplate(0), mRotate(false) { mMerge = new merge::None(); } /// /// Save model state /// LabelModel* LabelModel::save() const { LabelModel* savedModel = new LabelModel; savedModel->restore( this ); return savedModel; } /// /// Restore model state /// void LabelModel::restore( const LabelModel *savedModel ) { // Clear current object list foreach ( LabelModelObject* object, mObjectList ) { delete object; } mObjectList.clear(); // Now copy state mUntitledInstance = savedModel->mUntitledInstance; mModified = savedModel->mModified; mFileName = savedModel->mFileName; mCompressionLevel = savedModel->mCompressionLevel; mTmplate = savedModel->mTmplate; mFrame = savedModel->mFrame; mRotate = savedModel->mRotate; foreach ( LabelModelObject* savedObject, savedModel->mObjectList ) { LabelModelObject* object = savedObject->clone(); object->setParent( this ); mObjectList << object; connect( object, SIGNAL(changed()), this, SLOT(onObjectChanged()) ); connect( object, SIGNAL(moved()), this, SLOT(onObjectMoved()) ); } delete mMerge; mMerge = savedModel->mMerge->clone(); // Emit signals based on potential changes emit changed(); emit selectionChanged(); emit modifiedChanged(); emit nameChanged(); emit sizeChanged(); emit mergeChanged(); emit mergeSourceChanged(); emit mergeSelectionChanged(); } /// /// Is model modified? /// bool LabelModel::isModified() const { return mModified; } /// /// Get filename /// const QString& LabelModel::fileName() const { return mFileName; } /// /// Set filename /// void LabelModel::setFileName( const QString &fileName ) { if ( mFileName != fileName ) { mFileName = fileName; emit nameChanged(); } } /// /// Get compression level /// int LabelModel::compressionLevel() const { return mCompressionLevel; } /// /// Set compression level /// void LabelModel::setCompressionLevel( int compressionLevel ) { mCompressionLevel = compressionLevel; } /// /// Get template /// const Template* LabelModel::tmplate() const { return mTmplate; } /// /// Get frame /// const Frame* LabelModel::frame() const { return mFrame; } /// /// Set template /// void LabelModel::setTmplate( const Template* tmplate ) { if (mTmplate != tmplate) { mTmplate = tmplate; mFrame = tmplate->frames().first(); setModified(); emit changed(); emit sizeChanged(); Settings::addToRecentTemplateList( tmplate->name() ); } } /// /// Get rotation /// bool LabelModel::rotate() const { return mRotate; } /// /// Set rotation /// void LabelModel::setRotate( bool rotate ) { if (mRotate != rotate) { mRotate = rotate; setModified(); emit changed(); emit sizeChanged(); } } /// /// Get width /// Distance LabelModel::w() const { return mRotate ? mFrame->h() : mFrame->w(); } /// /// Get height /// Distance LabelModel::h() const { return mRotate ? mFrame->w() : mFrame->h(); } /// /// Get object list /// const QList& LabelModel::objectList() const { return mObjectList; } /// /// Get short name. /// QString LabelModel::shortName() { static int untitledCount = 0; if ( mFileName.isEmpty() ) { if ( mUntitledInstance == 0 ) { mUntitledInstance = ++untitledCount; } QString numString; numString.setNum(mUntitledInstance);; return tr("Untitled") + numString; } else { QFileInfo fileInfo( mFileName ); return fileInfo.baseName(); } } /// /// Get merge object /// merge::Merge* LabelModel::merge() const { return mMerge; } /// /// Set merge object /// void LabelModel::setMerge( merge::Merge* merge ) { if ( merge != mMerge ) { delete mMerge; mMerge = merge; connect( mMerge, SIGNAL(sourceChanged()), this, SLOT(onMergeSourceChanged()) ); connect( mMerge, SIGNAL(selectionChanged()), this, SLOT(onMergeSelectionChanged()) ); setModified(); emit changed(); emit mergeChanged(); emit mergeSourceChanged(); } } /// /// Set modified status /// void LabelModel::setModified() { mModified = true; emit modifiedChanged(); } /// /// Clear modified status /// void LabelModel::clearModified() { mModified = false; emit modifiedChanged(); } /// /// Add object. /// void LabelModel::addObject( LabelModelObject* object ) { object->setParent( this ); mObjectList << object; connect( object, SIGNAL(changed()), this, SLOT(onObjectChanged()) ); connect( object, SIGNAL(moved()), this, SLOT(onObjectMoved()) ); setModified(); emit changed(); } /// /// Delete Object /// void LabelModel::deleteObject( LabelModelObject* object ) { object->unselect(); mObjectList.removeOne( object ); disconnect( object, 0, this, 0 ); setModified(); emit changed(); delete object; } /// /// Object at x,y /// LabelModelObject* LabelModel::objectAt( double scale, const Distance& x, const Distance& y ) const { /* Search object list in reverse order. I.e. from top to bottom. */ QList::const_iterator it = mObjectList.end(); while ( it != mObjectList.begin() ) { it--; LabelModelObject* object = *it; if ( object->isLocatedAt( scale, x, y ) ) { return object; } } return 0; } /// /// Handle at x,y /// Handle* LabelModel::handleAt( double scale, const Distance& x, const Distance& y ) const { foreach( LabelModelObject* object, mObjectList ) { Handle* handle = object->handleAt( scale, x, y ); if ( handle ) { return handle; } } return 0; } /// /// Object Changed Slot /// void LabelModel::onObjectChanged() { setModified(); emit changed(); } /// /// Object Moved Slot /// void LabelModel::onObjectMoved() { setModified(); emit changed(); } /// /// Merge Source Changed Slot /// void LabelModel::onMergeSourceChanged() { setModified(); emit changed(); emit mergeSourceChanged(); } /// /// Merge Selection Changed Slot /// void LabelModel::onMergeSelectionChanged() { emit changed(); emit mergeSelectionChanged(); } /// /// Select Object /// void LabelModel::selectObject( LabelModelObject* object ) { object->select(); emit selectionChanged(); } /// /// Unselect Object /// void LabelModel::unselectObject( LabelModelObject* object ) { object->unselect(); emit selectionChanged(); } /// /// Select All Objects /// void LabelModel::selectAll() { foreach ( LabelModelObject* object, mObjectList ) { object->select(); } emit selectionChanged(); } /// /// Unselect All Objects /// void LabelModel::unselectAll() { foreach ( LabelModelObject* object, mObjectList ) { object->unselect(); } emit selectionChanged(); } /// /// Select Region /// void LabelModel::selectRegion( const Region ®ion ) { Distance rX1 = min( region.x1(), region.x2() ); Distance rY1 = min( region.y1(), region.y2() ); Distance rX2 = max( region.x1(), region.x2() ); Distance rY2 = max( region.y1(), region.y2() ); foreach ( LabelModelObject* object, mObjectList ) { Region objectExtent = object->getExtent(); if ( (objectExtent.x1() >= rX1) && (objectExtent.x2() <= rX2) && (objectExtent.y1() >= rY1) && (objectExtent.y2() <= rY2) ) { object->select(); } } emit selectionChanged(); } /// /// Is Selection Empty? /// bool LabelModel::isSelectionEmpty() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { return false; } } return true; } /// /// Is Selection Atomic? /// bool LabelModel::isSelectionAtomic() { int nSelected = 0; foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { nSelected++; if ( nSelected > 1 ) { return false; } } } return nSelected == 1; } /// /// Get List of Selected Objects /// QList LabelModel::getSelection() { QList selectedList; foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { selectedList << object; } } return selectedList; } /// /// Get First Object in Selection List /// LabelModelObject* LabelModel::getFirstSelectedObject() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { return object; } } } /// /// Can Any Objects in Selection Accept Text Properties? /// bool LabelModel::canSelectionText() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() && object->canText() ) { return true; } } return false; } /// /// Can Any Objects in Selection Accept Fill Property? /// bool LabelModel::canSelectionFill() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() && object->canFill() ) { return true; } } return false; } /// /// Can Any Objects in Selection Accept Line Color Property? /// bool LabelModel::canSelectionLineColor() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() && object->canLineColor() ) { return true; } } return false; } /// /// Can Any Objects in Selection Accept Line Width Property? /// bool LabelModel::canSelectionLineWidth() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() && object->canLineWidth() ) { return true; } } return false; } /// /// Delete Selected Objects /// void LabelModel::deleteSelection() { QList selectedList = getSelection(); foreach ( LabelModelObject* object, selectedList ) { deleteObject( object ); } setModified(); emit changed(); emit selectionChanged(); } /// /// Raise Selected Objects To Top /// void LabelModel::raiseSelectionToTop() { QList selectedList = getSelection(); foreach ( LabelModelObject* object, selectedList ) { mObjectList.removeOne( object ); } // Move to end of list, representing top most object. foreach ( LabelModelObject* object, selectedList ) { mObjectList.push_back( object ); } setModified(); emit changed(); } /// /// Lower Selected Objects To Bottom /// void LabelModel::lowerSelectionToBottom() { QList selectedList = getSelection(); foreach ( LabelModelObject* object, selectedList ) { mObjectList.removeOne( object ); } // Move to front of list, representing bottom most object. foreach ( LabelModelObject* object, selectedList ) { mObjectList.push_front( object ); } setModified(); emit changed(); } /// /// Rotate Selected Objects /// void LabelModel::rotateSelection( double thetaDegs ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->rotate( thetaDegs ); } } setModified(); emit changed(); } /// /// Rotate Selected Objects Left 90 degrees /// void LabelModel::rotateSelectionLeft() { rotateSelection( -90.0 ); } /// /// Rotate Selected Objects Right 90 degrees /// void LabelModel::rotateSelectionRight() { rotateSelection( 90.0 ); } /// /// Flip Selected Objects Horizontally /// void LabelModel::flipSelectionHoriz() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->flipHoriz(); } } setModified(); emit changed(); } /// /// Flip Selected Objects Vertically /// void LabelModel::flipSelectionVert() { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->flipVert(); } } setModified(); emit changed(); } /// /// Align Selected Objects To Their Left Edges /// void LabelModel::alignSelectionLeft() { if ( isSelectionEmpty() || isSelectionAtomic() ) { return; } QList selectedList = getSelection(); // Find left-most edge. Distance x1_min = 7200; // Start with a very large value: 7200pts = 100in foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); if ( r.x1() < x1_min ) x1_min = r.x1(); } // Now adjust the object positions to line up the left edges at left-most edge. foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dx = x1_min - r.x1(); object->setPositionRelative( dx, 0 ); } setModified(); emit changed(); } /// /// Align Selected Objects To Their Right Edges /// void LabelModel::alignSelectionRight() { if ( isSelectionEmpty() || isSelectionAtomic() ) { return; } QList selectedList = getSelection(); // Find right-most edge. Distance x1_max = -7200; // Start with a very large negative value: 7200pts = 100in foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); if ( r.x1() > x1_max ) x1_max = r.x1(); } // Now adjust the object positions to line up the right edges at right-most edge. foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dx = x1_max - r.x1(); object->setPositionRelative( dx, 0 ); } setModified(); emit changed(); } /// /// Align Selected Objects To Their Horizontal Centers /// void LabelModel::alignSelectionHCenter() { if ( isSelectionEmpty() || isSelectionAtomic() ) { return; } QList selectedList = getSelection(); // Find average center of objects. Distance xsum = 0; int n = 0; foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); xsum += (r.x1() + r.x2()) / 2.0; n++; } Distance xavg = xsum / n; // Find object closest to average center of objects. Distance xcenter = 7200; // Start with very large value. Distance dxmin = fabs( xavg - xcenter ); foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dx = fabs( xavg - (r.x1() + r.x2())/2.0 ); if ( dx < dxmin ) { dxmin = dx; xcenter = (r.x1() + r.x2()) / 2.0; } } // Now adjust the object positions to line up with the center of this object. foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dx = xcenter - (r.x1() + r.x2())/2.0; object->setPositionRelative( dx, 0 ); } setModified(); emit changed(); } /// /// Align Selected Objects To Their Top Edges /// void LabelModel::alignSelectionTop() { if ( isSelectionEmpty() || isSelectionAtomic() ) { return; } QList selectedList = getSelection(); // Find top-most edge. Distance y1_min = 7200; // Start with a very large value: 7200pts = 100in foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); if ( r.y1() < y1_min ) y1_min = r.y1(); } // Now adjust the object positions to line up the top edges at top-most edge. foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dy = y1_min - r.y1(); object->setPositionRelative( 0, dy ); } setModified(); emit changed(); } /// /// Align Selected Objects To Their Bottom Edges /// void LabelModel::alignSelectionBottom() { if ( isSelectionEmpty() || isSelectionAtomic() ) { return; } QList selectedList = getSelection(); // Find bottom-most edge. Distance y1_max = -7200; // Start with a very large negative value: 7200pts = 100in foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); if ( r.y1() > y1_max ) y1_max = r.y1(); } // Now adjust the object positions to line up the bottom edges at bottom-most edge. foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dy = y1_max - r.y1(); object->setPositionRelative( 0, dy ); } setModified(); emit changed(); } /// /// Align Selected Objects To Their Vertical Centers Edges /// void LabelModel::alignSelectionVCenter() { if ( isSelectionEmpty() || isSelectionAtomic() ) { return; } QList selectedList = getSelection(); // Find average center of objects. Distance ysum = 0; int n = 0; foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); ysum += (r.y1() + r.y2()) / 2.0; n++; } Distance yavg = ysum / n; // Find object closest to average center of objects. Distance ycenter = 7200; // Start with very large value. Distance dymin = fabs( yavg - ycenter ); foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dy = fabs( yavg - (r.y1() + r.y2())/2.0 ); if ( dy < dymin ) { dymin = dy; ycenter = (r.y1() + r.y2()) / 2.0; } } // Now adjust the object positions to line up with the center of this object. foreach ( LabelModelObject* object, selectedList ) { Region r = object->getExtent(); Distance dy = ycenter - (r.y1() + r.y2())/2.0; object->setPositionRelative( 0, dy ); } setModified(); emit changed(); } /// /// Align Selected Objects To Center Of Label Horizontally /// void LabelModel::centerSelectionHoriz() { Distance xLabelCenter = w() / 2.0; foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { Region r = object->getExtent(); Distance xObjectCenter = (r.x1() + r.x2()) / 2.0; Distance dx = xLabelCenter - xObjectCenter; object->setPositionRelative( dx, 0 ); } } setModified(); emit changed(); } /// /// Align Selected Objects To Center Of Label Vertically /// void LabelModel::centerSelectionVert() { Distance yLabelCenter = h() / 2.0; foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { Region r = object->getExtent(); Distance yObjectCenter = (r.y1() + r.y2()) / 2.0; Distance dy = yLabelCenter - yObjectCenter; object->setPositionRelative( 0, dy ); } } setModified(); emit changed(); } /// /// Move Selected Objects By dx,dy /// void LabelModel::moveSelection( const Distance& dx, const Distance& dy ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setPositionRelative( dx, dy ); } } setModified(); emit changed(); } /// /// Set Font Family Of Selected Objects /// void LabelModel::setSelectionFontFamily( const QString &fontFamily ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setFontFamily( fontFamily ); } } setModified(); emit changed(); } /// /// Set Font Size Of Selected Objects /// void LabelModel::setSelectionFontSize( double fontSize ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setFontSize( fontSize ); } } setModified(); emit changed(); } /// /// Set Font Weight Of Selected Objects /// void LabelModel::setSelectionFontWeight( QFont::Weight fontWeight ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setFontWeight( fontWeight ); } } setModified(); emit changed(); } /// /// Set Font Italic Flag Of Selected Objects /// void LabelModel::setSelectionFontItalicFlag( bool fontItalicFlag ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setFontItalicFlag( fontItalicFlag ); } } setModified(); emit changed(); } /// /// Set Text Horizontal Alignment Of Selected Objects /// void LabelModel::setSelectionTextHAlign( Qt::Alignment textHAlign ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setTextHAlign( textHAlign ); } } setModified(); emit changed(); } /// /// Set Text Vertical Alignment Of Selected Objects /// void LabelModel::setSelectionTextVAlign( Qt::Alignment textVAlign ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setTextVAlign( textVAlign ); } } setModified(); emit changed(); } /// /// Set Text Line Spacing Of Selected Objects /// void LabelModel::setSelectionTextLineSpacing( double textLineSpacing ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setTextLineSpacing( textLineSpacing ); } } setModified(); emit changed(); } /// /// Set Text Color Node Of Selected Objects /// void LabelModel::setSelectionTextColorNode( ColorNode textColorNode ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setTextColorNode( textColorNode ); } } setModified(); emit changed(); } /// /// Set Line Width Of Selected Objects /// void LabelModel::setSelectionLineWidth( const Distance& lineWidth ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setLineWidth( lineWidth ); } } setModified(); emit changed(); } /// /// Set Line Color Node Of Selected Objects /// void LabelModel::setSelectionLineColorNode( ColorNode lineColorNode ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setLineColorNode( lineColorNode ); } } setModified(); emit changed(); } /// /// Set Fill Color Node Of Selected Objects /// void LabelModel::setSelectionFillColorNode( ColorNode fillColorNode ) { foreach ( LabelModelObject* object, mObjectList ) { if ( object->isSelected() ) { object->setFillColorNode( fillColorNode ); } } setModified(); emit changed(); } /// /// Copy selection to clipboard /// void LabelModel::copySelection() { if ( !isSelectionEmpty() ) { QClipboard *clipboard = QApplication::clipboard(); QByteArray buffer; XmlLabelCreator::serializeObjects( getSelection(), buffer ); QMimeData *mimeData = new QMimeData; mimeData->setData( MIME_TYPE, buffer ); clipboard->setMimeData( mimeData ); } } /// /// Cut selection to clipboard /// void LabelModel::cutSelection() { copySelection(); deleteSelection(); } /// /// Can we paste? /// bool LabelModel::canPaste() { const QClipboard *clipboard = QApplication::clipboard(); const QMimeData *mimeData = clipboard->mimeData(); if ( mimeData->hasFormat( MIME_TYPE ) ) { return true; } else if ( mimeData->hasImage() ) { return true; } else if ( mimeData->hasText() ) { return true; } return false; } /// /// Paste from clipboard /// void LabelModel::paste() { const QClipboard *clipboard = QApplication::clipboard(); const QMimeData *mimeData = clipboard->mimeData(); if ( mimeData->hasFormat( MIME_TYPE ) ) { // Native objects QByteArray buffer = mimeData->data( MIME_TYPE ); QList objects = XmlLabelParser::deserializeObjects( buffer ); unselectAll(); foreach ( LabelModelObject* object, objects ) { addObject( object ); selectObject( object ); } } else if ( mimeData->hasImage() ) { // Create object from clipboard image LabelModelImageObject* object = new LabelModelImageObject(); object->setImage( qvariant_cast(mimeData->imageData()) ); object->setSize( object->naturalSize() ); object->setPosition( (w()-object->w())/2.0, (h()-object->h())/2.0 ); addObject( object ); unselectAll(); selectObject( object ); } else if ( mimeData->hasText() ) { // Create object from clipboard text LabelModelTextObject* object = new LabelModelTextObject(); object->setText( mimeData->text() ); object->setSize( object->naturalSize() ); object->setPosition( (w()-object->w())/2.0, (h()-object->h())/2.0 ); addObject( object ); unselectAll(); selectObject( object ); } } /// /// Draw label objects /// void LabelModel::draw( QPainter* painter, bool inEditor, merge::Record* record ) const { foreach ( LabelModelObject* object, mObjectList ) { object->draw( painter, inEditor, record ); } } }