/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include "svx/connctrl.hxx" #include "svx/dlgutil.hxx" #include #include #include #include #include #include #include #include #include #include #include #include SvxXConnectionPreview::SvxXConnectionPreview( vcl::Window* pParent, WinBits nStyle) : Control(pParent, nStyle) , pEdgeObj(nullptr) , pObjList(nullptr) , pView(nullptr) { SetMapMode( MapUnit::Map100thMM ); SetStyles(); } VCL_BUILDER_FACTORY_CONSTRUCTOR(SvxXConnectionPreview, 0) SvxXConnectionPreview::~SvxXConnectionPreview() { disposeOnce(); } void SvxXConnectionPreview::dispose() { delete pObjList; Control::dispose(); } void SvxXConnectionPreview::Resize() { Control::Resize(); AdaptSize(); Invalidate(); } Size SvxXConnectionPreview::GetOptimalSize() const { return LogicToPixel(Size(118 , 121), MapMode(MapUnit::MapAppFont)); } void SvxXConnectionPreview::AdaptSize() { // Adapt size if( pObjList ) { SetMapMode( MapUnit::Map100thMM ); OutputDevice* pOD = pView->GetFirstOutputDevice(); // GetWin( 0 ); tools::Rectangle aRect = pObjList->GetAllObjBoundRect(); MapMode aMapMode = GetMapMode(); aMapMode.SetMapUnit( pOD->GetMapMode().GetMapUnit() ); SetMapMode( aMapMode ); MapMode aDisplayMap( aMapMode ); Point aNewPos; Size aNewSize; const Size aWinSize = PixelToLogic( GetOutputSizePixel(), aDisplayMap ); const long nWidth = aWinSize.Width(); const long nHeight = aWinSize.Height(); if (aRect.GetHeight() == 0) return; double fRectWH = (double) aRect.GetWidth() / aRect.GetHeight(); if (nHeight == 0) return; double fWinWH = (double) nWidth / nHeight; // Adapt bitmap to Thumb size (not here!) if ( fRectWH < fWinWH) { aNewSize.Width() = (long) ( (double) nHeight * fRectWH ); aNewSize.Height()= nHeight; } else { aNewSize.Width() = nWidth; aNewSize.Height()= (long) ( (double) nWidth / fRectWH ); } Fraction aFrac1( aWinSize.Width(), aRect.GetWidth() ); Fraction aFrac2( aWinSize.Height(), aRect.GetHeight() ); Fraction aMinFrac( aFrac1 <= aFrac2 ? aFrac1 : aFrac2 ); // Implement MapMode aDisplayMap.SetScaleX( aMinFrac ); aDisplayMap.SetScaleY( aMinFrac ); // Centering aNewPos.X() = ( nWidth - aNewSize.Width() ) >> 1; aNewPos.Y() = ( nHeight - aNewSize.Height() ) >> 1; aDisplayMap.SetOrigin( LogicToLogic( aNewPos, aMapMode, aDisplayMap ) ); SetMapMode( aDisplayMap ); // Origin aNewPos = aDisplayMap.GetOrigin(); aNewPos -= Point( aRect.TopLeft().X(), aRect.TopLeft().Y() ); aDisplayMap.SetOrigin( aNewPos ); SetMapMode( aDisplayMap ); Point aPos; MouseEvent aMEvt( aPos, 1, MouseEventModifiers::NONE, MOUSE_RIGHT ); MouseButtonDown( aMEvt ); } } void SvxXConnectionPreview::Construct() { DBG_ASSERT( pView, "No valid view is passed on! "); const SdrMarkList& rMarkList = pView->GetMarkedObjectList(); const size_t nMarkCount = rMarkList.GetMarkCount(); if( nMarkCount >= 1 ) { bool bFound = false; for( size_t i = 0; i < nMarkCount && !bFound; ++i ) { const SdrObject* pObj = rMarkList.GetMark( i )->GetMarkedSdrObj(); SdrInventor nInv = pObj->GetObjInventor(); sal_uInt16 nId = pObj->GetObjIdentifier(); if( nInv == SdrInventor::Default && nId == OBJ_EDGE ) { bFound = true; const SdrEdgeObj* pTmpEdgeObj = static_cast(pObj); pEdgeObj = pTmpEdgeObj->Clone(); SdrObjConnection& rConn1 = (SdrObjConnection&)pEdgeObj->GetConnection( true ); SdrObjConnection& rConn2 = (SdrObjConnection&)pEdgeObj->GetConnection( false ); rConn1 = pTmpEdgeObj->GetConnection( true ); rConn2 = pTmpEdgeObj->GetConnection( false ); SdrObject* pTmpObj1 = pTmpEdgeObj->GetConnectedNode( true ); SdrObject* pTmpObj2 = pTmpEdgeObj->GetConnectedNode( false ); // potential memory leak here (!). Create SdrObjList only when there is // not yet one. if(!pObjList) { pObjList = new SdrObjList( pView->GetModel(), nullptr ); } if( pTmpObj1 ) { SdrObject* pObj1 = pTmpObj1->Clone(); pObjList->InsertObject( pObj1 ); pEdgeObj->ConnectToNode( true, pObj1 ); } if( pTmpObj2 ) { SdrObject* pObj2 = pTmpObj2->Clone(); pObjList->InsertObject( pObj2 ); pEdgeObj->ConnectToNode( false, pObj2 ); } pObjList->InsertObject( pEdgeObj ); } } } if( !pEdgeObj ) pEdgeObj = new SdrEdgeObj(); AdaptSize(); } void SvxXConnectionPreview::Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle&) { if (pObjList) { // This will not work anymore. To not start at Adam and Eve, i will // ATM not try to change all this stuff to really using an own model // and a view. I will just try to provide a mechanism to paint such // objects without own model and without a page/view with the new // mechanism. // New stuff: Use a ObjectContactOfObjListPainter. sdr::contact::SdrObjectVector aObjectVector; for (size_t a = 0; a < pObjList->GetObjCount(); ++a) { SdrObject* pObject = pObjList->GetObj(a); DBG_ASSERT(pObject, "SvxXConnectionPreview::Paint: Corrupt ObjectList (!)"); aObjectVector.push_back(pObject); } sdr::contact::ObjectContactOfObjListPainter aPainter(rRenderContext, aObjectVector, nullptr); sdr::contact::DisplayInfo aDisplayInfo; // do processing aPainter.ProcessDisplay(aDisplayInfo); } } void SvxXConnectionPreview::SetAttributes( const SfxItemSet& rInAttrs ) { pEdgeObj->SetMergedItemSetAndBroadcast(rInAttrs); Invalidate(); } // Get number of lines which are offset based on the preview object sal_uInt16 SvxXConnectionPreview::GetLineDeltaCount() { const SfxItemSet& rSet = pEdgeObj->GetMergedItemSet(); sal_uInt16 nCount(0); if(SfxItemState::DONTCARE != rSet.GetItemState(SDRATTR_EDGELINEDELTAANZ)) nCount = static_cast(rSet.Get(SDRATTR_EDGELINEDELTAANZ)).GetValue(); return nCount; } void SvxXConnectionPreview::MouseButtonDown( const MouseEvent& rMEvt ) { bool bZoomIn = rMEvt.IsLeft() && !rMEvt.IsShift(); bool bZoomOut = rMEvt.IsRight() || rMEvt.IsShift(); bool bCtrl = rMEvt.IsMod1(); if( bZoomIn || bZoomOut ) { MapMode aMapMode = GetMapMode(); Fraction aXFrac = aMapMode.GetScaleX(); Fraction aYFrac = aMapMode.GetScaleY(); std::unique_ptr pMultFrac; if( bZoomIn ) { if( bCtrl ) pMultFrac.reset(new Fraction( 3, 2 )); else pMultFrac.reset(new Fraction( 11, 10 )); } else { if( bCtrl ) pMultFrac.reset(new Fraction( 2, 3 )); else pMultFrac.reset(new Fraction( 10, 11 )); } aXFrac *= *pMultFrac; aYFrac *= *pMultFrac; if( (double)aXFrac > 0.001 && (double)aXFrac < 1000.0 && (double)aYFrac > 0.001 && (double)aYFrac < 1000.0 ) { aMapMode.SetScaleX( aXFrac ); aMapMode.SetScaleY( aYFrac ); SetMapMode( aMapMode ); Size aOutSize( GetOutputSize() ); Point aPt( aMapMode.GetOrigin() ); long nX = (long)( ( (double)aOutSize.Width() - ( (double)aOutSize.Width() * (double)*pMultFrac ) ) / 2.0 + 0.5 ); long nY = (long)( ( (double)aOutSize.Height() - ( (double)aOutSize.Height() * (double)*pMultFrac ) ) / 2.0 + 0.5 ); aPt.X() += nX; aPt.Y() += nY; aMapMode.SetOrigin( aPt ); SetMapMode( aMapMode ); Invalidate(); } } } void SvxXConnectionPreview::SetStyles() { const StyleSettings& rStyles = Application::GetSettings().GetStyleSettings(); SetDrawMode( GetSettings().GetStyleSettings().GetHighContrastMode() ? OUTPUT_DRAWMODE_CONTRAST : OUTPUT_DRAWMODE_COLOR ); SetBackground( Wallpaper( Color( rStyles.GetFieldColor() ) ) ); } void SvxXConnectionPreview::DataChanged( const DataChangedEvent& rDCEvt ) { Control::DataChanged( rDCEvt ); if ((rDCEvt.GetType() == DataChangedEventType::SETTINGS) && (rDCEvt.GetFlags() & AllSettingsFlags::STYLE) ) { SetStyles(); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */