1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
4 *
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 *
9 * This file incorporates work covered by the following license notice:
10 *
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
18 */
19 #include <SectionWindow.hxx>
20 #include <ReportWindow.hxx>
21 #include <UITools.hxx>
22 #include <ReportController.hxx>
23 #include <ReportSection.hxx>
24 #include <DesignView.hxx>
25 #include <strings.hxx>
26 #include <core_resource.hxx>
27 #include <strings.hrc>
28 #include <StartMarker.hxx>
29 #include <EndMarker.hxx>
30 #include <ViewsWindow.hxx>
31
32 #include <functional>
33 #include <algorithm>
34 #include <vcl/svapp.hxx>
35 #include <vcl/settings.hxx>
36
37 namespace rptui
38 {
39 using namespace ::com::sun::star;
40 using namespace ::comphelper;
41
OSectionWindow(OViewsWindow * _pParent,const uno::Reference<report::XSection> & _xSection,const OUString & _sColorEntry)42 OSectionWindow::OSectionWindow( OViewsWindow* _pParent,const uno::Reference< report::XSection >& _xSection,const OUString& _sColorEntry)
43 : Window( _pParent,WB_DIALOGCONTROL)
44 ,OPropertyChangeListener(m_aMutex)
45 ,m_pParent(_pParent)
46 ,m_aStartMarker( VclPtr<rptui::OStartMarker>::Create(this,_sColorEntry))
47 ,m_aReportSection( VclPtr<rptui::OReportSection>::Create(this,_xSection))
48 ,m_aSplitter(VclPtr<Splitter>::Create(this))
49 ,m_aEndMarker( VclPtr<rptui::OEndMarker>::Create(this,_sColorEntry))
50 {
51 const MapMode& rMapMode = _pParent->GetMapMode();
52 SetMapMode( rMapMode );
53 ImplInitSettings();
54 // TRY
55 m_aSplitter->SetMapMode( MapMode( MapUnit::Map100thMM ) );
56 m_aSplitter->SetStartSplitHdl(LINK(this, OSectionWindow,StartSplitHdl));
57 m_aSplitter->SetSplitHdl(LINK(this, OSectionWindow,SplitHdl));
58 m_aSplitter->SetEndSplitHdl(LINK(this, OSectionWindow,EndSplitHdl));
59 m_aSplitter->SetBackground( Wallpaper( Application::GetSettings().GetStyleSettings().GetFaceColor() ));
60 m_aSplitter->SetSplitPosPixel(m_aSplitter->LogicToPixel(Size(0,_xSection->getHeight())).Height());
61
62
63 m_aStartMarker->setCollapsedHdl(LINK(this,OSectionWindow,Collapsed));
64
65 m_aStartMarker->zoom(rMapMode.GetScaleX());
66 setZoomFactor(rMapMode.GetScaleX(), *m_aReportSection);
67 setZoomFactor(rMapMode.GetScaleX(), *m_aSplitter);
68 setZoomFactor(rMapMode.GetScaleX(), *m_aEndMarker);
69
70 m_aSplitter->Show();
71 m_aStartMarker->Show();
72 m_aReportSection->Show();
73 m_aEndMarker->Show();
74 Show();
75
76 m_pSectionMulti = new OPropertyChangeMultiplexer(this,_xSection);
77 m_pSectionMulti->addProperty(PROPERTY_NAME);
78 m_pSectionMulti->addProperty(PROPERTY_HEIGHT);
79
80 beans::PropertyChangeEvent aEvent;
81 aEvent.Source = _xSection;
82 aEvent.PropertyName = PROPERTY_NAME;
83 uno::Reference< report::XGroup > xGroup(_xSection->getGroup());
84 if ( xGroup.is() )
85 {
86 m_pGroupMulti = new OPropertyChangeMultiplexer(this,xGroup);
87 m_pGroupMulti->addProperty(PROPERTY_EXPRESSION);
88 aEvent.Source = xGroup;
89 aEvent.PropertyName = PROPERTY_EXPRESSION;
90 }
91
92 _propertyChanged(aEvent);
93 }
94
~OSectionWindow()95 OSectionWindow::~OSectionWindow()
96 {
97 disposeOnce();
98 }
99
dispose()100 void OSectionWindow::dispose()
101 {
102 try
103 {
104 if ( m_pSectionMulti.is() )
105 m_pSectionMulti->dispose();
106 m_pSectionMulti.clear();
107 if ( m_pGroupMulti.is() )
108 m_pGroupMulti->dispose();
109 m_pGroupMulti.clear();
110 }
111 catch (uno::Exception&)
112 {
113 }
114 m_aStartMarker.disposeAndClear();
115 m_aReportSection.disposeAndClear();
116 m_aSplitter.disposeAndClear();
117 m_aEndMarker.disposeAndClear();
118 m_pParent.clear();
119 vcl::Window::dispose();
120 }
121
_propertyChanged(const beans::PropertyChangeEvent & _rEvent)122 void OSectionWindow::_propertyChanged(const beans::PropertyChangeEvent& _rEvent)
123 {
124 SolarMutexGuard g;
125 const uno::Reference< report::XSection > xSection(_rEvent.Source,uno::UNO_QUERY);
126 if ( xSection.is() )
127 {
128 if ( _rEvent.PropertyName == PROPERTY_HEIGHT )
129 {
130 m_pParent->getView()->SetUpdateMode(false);
131 //Resize();
132 m_pParent->getView()->notifySizeChanged();
133 m_pParent->resize(*this);
134 m_pParent->getView()->SetUpdateMode(true);
135 // getViewsWindow()->getView()->getReportView()->getController().resetZoomType();
136 }
137 else if ( _rEvent.PropertyName == PROPERTY_NAME && !xSection->getGroup().is() )
138 {
139 uno::Reference< report::XReportDefinition > xReport = xSection->getReportDefinition();
140 if ( setReportSectionTitle(xReport,RID_STR_REPORT_HEADER,::std::mem_fn(&OReportHelper::getReportHeader),::std::mem_fn(&OReportHelper::getReportHeaderOn))
141 || setReportSectionTitle(xReport,RID_STR_REPORT_FOOTER,::std::mem_fn(&OReportHelper::getReportFooter),::std::mem_fn(&OReportHelper::getReportFooterOn))
142 || setReportSectionTitle(xReport,RID_STR_PAGE_HEADER,::std::mem_fn(&OReportHelper::getPageHeader),::std::mem_fn(&OReportHelper::getPageHeaderOn))
143 || setReportSectionTitle(xReport,RID_STR_PAGE_FOOTER,::std::mem_fn(&OReportHelper::getPageFooter),::std::mem_fn(&OReportHelper::getPageFooterOn)) )
144 {
145 m_aStartMarker->Invalidate(InvalidateFlags::NoErase);
146 }
147 else
148 {
149 OUString sTitle = RptResId(RID_STR_DETAIL);
150 m_aStartMarker->setTitle(sTitle);
151 m_aStartMarker->Invalidate(InvalidateFlags::Children);
152 }
153 }
154 }
155 else if ( _rEvent.PropertyName == PROPERTY_EXPRESSION )
156 {
157 uno::Reference< report::XGroup > xGroup(_rEvent.Source,uno::UNO_QUERY);
158 if ( xGroup.is() && !setGroupSectionTitle(xGroup,RID_STR_HEADER,::std::mem_fn(&OGroupHelper::getHeader),::std::mem_fn(&OGroupHelper::getHeaderOn)))
159 {
160 setGroupSectionTitle(xGroup,RID_STR_FOOTER,::std::mem_fn(&OGroupHelper::getFooter),::std::mem_fn(&OGroupHelper::getFooterOn));
161 }
162 }
163 }
164
setReportSectionTitle(const uno::Reference<report::XReportDefinition> & _xReport,const char * pResId,const::std::function<uno::Reference<report::XSection> (OReportHelper *)> & _pGetSection,const::std::function<bool (OReportHelper *)> & _pIsSectionOn)165 bool OSectionWindow::setReportSectionTitle(
166 const uno::Reference<report::XReportDefinition>& _xReport, const char* pResId,
167 const ::std::function<uno::Reference<report::XSection>(OReportHelper*)>& _pGetSection,
168 const ::std::function<bool(OReportHelper*)>& _pIsSectionOn)
169 {
170 OReportHelper aReportHelper(_xReport);
171 const bool bRet = _pIsSectionOn(&aReportHelper) && _pGetSection(&aReportHelper) == m_aReportSection->getSection();
172 if ( bRet )
173 {
174 OUString sTitle = RptResId(pResId);
175 m_aStartMarker->setTitle(sTitle);
176 m_aStartMarker->Invalidate(InvalidateFlags::Children);
177 }
178 return bRet;
179 }
180
setGroupSectionTitle(const uno::Reference<report::XGroup> & _xGroup,const char * pResId,const::std::function<uno::Reference<report::XSection> (OGroupHelper *)> & _pGetSection,const::std::function<bool (OGroupHelper *)> & _pIsSectionOn)181 bool OSectionWindow::setGroupSectionTitle(
182 const uno::Reference<report::XGroup>& _xGroup, const char* pResId,
183 const ::std::function<uno::Reference<report::XSection>(OGroupHelper*)>& _pGetSection,
184 const ::std::function<bool(OGroupHelper*)>& _pIsSectionOn)
185 {
186 OGroupHelper aGroupHelper(_xGroup);
187 const bool bRet = _pIsSectionOn(&aGroupHelper) && _pGetSection(&aGroupHelper) == m_aReportSection->getSection() ;
188 if ( bRet )
189 {
190 OUString sExpression = _xGroup->getExpression();
191 OUString sLabel = getViewsWindow()->getView()->getReportView()->getController().getColumnLabel_throw(sExpression);
192 if ( !sLabel.isEmpty() )
193 {
194 sExpression = sLabel;
195 }
196
197 OUString sTitle(RptResId(pResId));
198 sTitle = sTitle.replaceFirst("#", sExpression);
199 m_aStartMarker->setTitle( sTitle );
200 m_aStartMarker->Invalidate(InvalidateFlags::Children);
201 }
202 return bRet;
203 }
204
ImplInitSettings()205 void OSectionWindow::ImplInitSettings()
206 {
207 EnableChildTransparentMode();
208 SetParentClipMode( ParentClipMode::NoClip );
209 SetPaintTransparent( true );
210 SetBackground( );
211 }
212
DataChanged(const DataChangedEvent & rDCEvt)213 void OSectionWindow::DataChanged( const DataChangedEvent& rDCEvt )
214 {
215 Window::DataChanged( rDCEvt );
216
217 if ( (rDCEvt.GetType() == DataChangedEventType::SETTINGS) &&
218 (rDCEvt.GetFlags() & AllSettingsFlags::STYLE) )
219 {
220 ImplInitSettings();
221 Invalidate();
222 }
223 }
224
Resize()225 void OSectionWindow::Resize()
226 {
227 Window::Resize();
228
229 Size aOutputSize = GetOutputSizePixel();
230 tools::Long nEndWidth = tools::Long(REPORT_ENDMARKER_WIDTH * GetMapMode().GetScaleX());
231
232 const Point aThumbPos = m_pParent->getView()->getThumbPos();
233 aOutputSize.AdjustWidth( -(aThumbPos.X()) );
234 aOutputSize.AdjustHeight( -m_aSplitter->GetSizePixel().Height() );
235
236 if ( m_aStartMarker->isCollapsed() )
237 {
238 Point aPos(0,0);
239 m_aStartMarker->SetPosSizePixel(aPos,aOutputSize);
240 }
241 else
242 {
243 const bool bShowEndMarker = m_pParent->getView()->GetTotalWidth() <= (aThumbPos.X() + aOutputSize.Width() );
244
245 tools::Long nStartWidth = tools::Long(REPORT_STARTMARKER_WIDTH * GetMapMode().GetScaleX());
246
247 // set start marker
248 m_aStartMarker->SetPosSizePixel(Point(0,0),Size(nStartWidth,aOutputSize.Height()));
249
250 // set report section
251 const uno::Reference< report::XSection> xSection = m_aReportSection->getSection();
252 Size aSectionSize = LogicToPixel( Size( 0,xSection->getHeight() ) );
253 Point aReportPos(nStartWidth,0);
254 aSectionSize.setWidth( aOutputSize.Width() - nStartWidth );
255 if ( bShowEndMarker )
256 aSectionSize.AdjustWidth( -nEndWidth );
257
258 m_aReportSection->SetPosSizePixel(aReportPos,aSectionSize);
259
260 // set splitter
261 aReportPos.AdjustY(aSectionSize.Height() );
262 m_aSplitter->SetPosSizePixel(aReportPos,Size(aSectionSize.Width(),m_aSplitter->GetSizePixel().Height()));
263 aSectionSize.setHeight( static_cast<tools::Long>(1000 * static_cast<double>(GetMapMode().GetScaleY())) );
264 m_aSplitter->SetDragRectPixel( tools::Rectangle(Point(nStartWidth,0),aSectionSize));
265
266 // set end marker
267 aReportPos.AdjustX(aSectionSize.Width() );
268 aReportPos.setY( 0 );
269 m_aEndMarker->Show(bShowEndMarker);
270 m_aEndMarker->SetPosSizePixel(aReportPos,Size(nEndWidth,aOutputSize.Height()));
271 }
272 }
273
setCollapsed(bool _bCollapsed)274 void OSectionWindow::setCollapsed(bool _bCollapsed)
275 {
276 if ( m_aStartMarker->isCollapsed() != _bCollapsed )
277 {
278 m_aStartMarker->setCollapsed(_bCollapsed);
279 }
280 }
281
showProperties()282 void OSectionWindow::showProperties()
283 {
284 m_pParent->getView()->showProperties( m_aReportSection->getSection() );
285 }
286
setMarked(bool _bMark)287 void OSectionWindow::setMarked(bool _bMark)
288 {
289 m_aStartMarker->setMarked(_bMark);
290 m_aEndMarker->setMarked(_bMark);
291 }
292
IMPL_LINK(OSectionWindow,Collapsed,OColorListener &,_rMarker,void)293 IMPL_LINK( OSectionWindow, Collapsed, OColorListener&, _rMarker, void )
294 {
295 bool bShow = !_rMarker.isCollapsed();
296 m_aReportSection->Show(bShow);
297 m_aEndMarker->Show(bShow);
298 m_aSplitter->Show(bShow);
299
300 m_pParent->resize(*this);
301 }
302
zoom(const Fraction & _aZoom)303 void OSectionWindow::zoom(const Fraction& _aZoom)
304 {
305 setZoomFactor(_aZoom,*this);
306 m_aStartMarker->zoom(_aZoom);
307
308 setZoomFactor(_aZoom, *m_aReportSection);
309 setZoomFactor(_aZoom, *m_aSplitter);
310 setZoomFactor(_aZoom, *m_aEndMarker);
311 Invalidate();
312 }
313
IMPL_LINK_NOARG(OSectionWindow,StartSplitHdl,Splitter *,void)314 IMPL_LINK_NOARG( OSectionWindow, StartSplitHdl, Splitter*, void)
315 {
316 const OUString sUndoAction( RptResId( RID_STR_UNDO_CHANGE_SIZE ) );
317 getViewsWindow()->getView()->getReportView()->getController().getUndoManager().EnterListAction( sUndoAction, OUString(), 0, ViewShellId(-1) );
318 }
319
IMPL_LINK_NOARG(OSectionWindow,EndSplitHdl,Splitter *,void)320 IMPL_LINK_NOARG( OSectionWindow, EndSplitHdl, Splitter*, void )
321 {
322 getViewsWindow()->getView()->getReportView()->getController().getUndoManager().LeaveListAction();
323 }
324
IMPL_LINK(OSectionWindow,SplitHdl,Splitter *,_pSplitter,void)325 IMPL_LINK( OSectionWindow, SplitHdl, Splitter*, _pSplitter, void )
326 {
327 if ( !getViewsWindow()->getView()->getReportView()->getController().isEditable() )
328 {
329 return;
330 }
331
332 sal_Int32 nSplitPos = _pSplitter->GetSplitPosPixel();
333
334 const uno::Reference< report::XSection> xSection = m_aReportSection->getSection();
335 nSplitPos = m_aSplitter->PixelToLogic(Size(0,nSplitPos)).Height();
336
337 const sal_Int32 nCount = xSection->getCount();
338 for (sal_Int32 i = 0; i < nCount; ++i)
339 {
340 uno::Reference<report::XReportComponent> xReportComponent(xSection->getByIndex(i),uno::UNO_QUERY);
341 if ( xReportComponent.is() )
342 {
343 nSplitPos = ::std::max(nSplitPos,xReportComponent->getPositionY() + xReportComponent->getHeight());
344 }
345 }
346
347 if ( nSplitPos < 0 )
348 nSplitPos = 0;
349
350 xSection->setHeight(nSplitPos);
351 m_aSplitter->SetSplitPosPixel(m_aSplitter->LogicToPixel(Size(0,nSplitPos)).Height());
352 }
353
lcl_scroll(vcl::Window & _rWindow,const Point & _aDelta)354 static void lcl_scroll(vcl::Window& _rWindow,const Point& _aDelta)
355 {
356 _rWindow.Scroll(-_aDelta.X(),-_aDelta.Y());
357 _rWindow.Invalidate(InvalidateFlags::Transparent);
358 }
359
lcl_setOrigin(vcl::Window & _rWindow,tools::Long _nX,tools::Long _nY)360 static void lcl_setOrigin(vcl::Window& _rWindow,tools::Long _nX, tools::Long _nY)
361 {
362 MapMode aMap = _rWindow.GetMapMode();
363 aMap.SetOrigin( Point(- _nX, - _nY));
364 _rWindow.SetMapMode( aMap );
365 }
366
scrollChildren(tools::Long _nX)367 void OSectionWindow::scrollChildren(tools::Long _nX)
368 {
369 const Point aDelta( _nX,0 );
370
371 MapMode aMapMode( m_aReportSection->GetMapMode() );
372 const Point aOld = aMapMode.GetOrigin();
373 lcl_setOrigin(*m_aReportSection, aDelta.X(), 0);
374
375 aMapMode = m_aReportSection->GetMapMode();
376 const Point aNew = aMapMode.GetOrigin();
377 const Point aDiff = aOld - aNew;
378 {
379 lcl_scroll(*m_aReportSection, aDiff);
380 }
381
382 lcl_scroll(*m_aEndMarker, m_aEndMarker->PixelToLogic(Point(_nX,0)));
383
384 lcl_setOrigin(*m_aSplitter,_nX, 0);
385 lcl_scroll(*m_aSplitter,aDiff);
386 }
387
388 } // rptui
389
390
391 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
392