Line data Source code
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 :
20 : #include "svx/dialcontrol.hxx"
21 :
22 : #include <math.h>
23 : #include <vcl/virdev.hxx>
24 : #include <vcl/svapp.hxx>
25 : #include <vcl/bitmap.hxx>
26 : #include <vcl/field.hxx>
27 : #include <svtools/colorcfg.hxx>
28 :
29 : namespace svx {
30 :
31 : // ============================================================================
32 :
33 : const long DIAL_OUTER_WIDTH = 8;
34 :
35 : // ============================================================================
36 :
37 0 : class DialControlBmp : public VirtualDevice
38 : {
39 : public:
40 : explicit DialControlBmp( Window& rParent );
41 :
42 : void InitBitmap( const Size& rSize, const Font& rFont );
43 : void CopyBackground( const DialControlBmp& rSrc );
44 : void DrawBackground( const Size& rSize, bool bEnabled );
45 : void DrawElements( const String& rText, sal_Int32 nAngle );
46 :
47 : private:
48 : const Color& GetBackgroundColor() const;
49 : const Color& GetTextColor() const;
50 : const Color& GetScaleLineColor() const;
51 : const Color& GetButtonLineColor() const;
52 : const Color& GetButtonFillColor( bool bMain ) const;
53 :
54 : void Init( const Size& rSize );
55 : void DrawBackground();
56 :
57 : Window& mrParent;
58 : Rectangle maRect;
59 : long mnCenterX;
60 : long mnCenterY;
61 : bool mbEnabled;
62 : };
63 :
64 : // ----------------------------------------------------------------------------
65 :
66 0 : DialControlBmp::DialControlBmp( Window& rParent ) :
67 : VirtualDevice( rParent, 0, 0 ),
68 : mrParent( rParent ),
69 : mnCenterX(0),
70 : mnCenterY(0),
71 0 : mbEnabled( true )
72 : {
73 0 : EnableRTL( sal_False );
74 0 : }
75 :
76 0 : void DialControlBmp::InitBitmap( const Size& rSize, const Font& rFont )
77 : {
78 0 : Init( rSize );
79 0 : SetFont( rFont );
80 0 : }
81 :
82 0 : void DialControlBmp::CopyBackground( const DialControlBmp& rSrc )
83 : {
84 0 : Init( rSrc.maRect.GetSize() );
85 0 : mbEnabled = rSrc.mbEnabled;
86 0 : Point aPos;
87 0 : DrawBitmapEx( aPos, rSrc.GetBitmapEx( aPos, maRect.GetSize() ) );
88 0 : }
89 :
90 0 : void DialControlBmp::DrawBackground( const Size& rSize, bool bEnabled )
91 : {
92 0 : Init( rSize );
93 0 : mbEnabled = bEnabled;
94 0 : DrawBackground();
95 0 : }
96 :
97 0 : void DialControlBmp::DrawElements( const String& rText, sal_Int32 nAngle )
98 : {
99 : // *** rotated text ***
100 :
101 0 : Font aFont( GetFont() );
102 0 : aFont.SetColor( GetTextColor() );
103 0 : aFont.SetOrientation( static_cast< short >( (nAngle + 5) / 10 ) ); // Font uses 1/10 degrees
104 0 : aFont.SetWeight( WEIGHT_BOLD );
105 0 : SetFont( aFont );
106 :
107 0 : double fAngle = nAngle * F_PI180 / 100.0;
108 0 : double fSin = sin( fAngle );
109 0 : double fCos = cos( fAngle );
110 0 : double fWidth = GetTextWidth( rText ) / 2.0;
111 0 : double fHeight = GetTextHeight() / 2.0;
112 0 : long nX = static_cast< long >( mnCenterX - fWidth * fCos - fHeight * fSin );
113 0 : long nY = static_cast< long >( mnCenterY + fWidth * fSin - fHeight * fCos );
114 0 : Rectangle aRect( nX, nY, 2 * mnCenterX - nX, 2 * mnCenterY - nY );
115 0 : DrawText( aRect, rText, mbEnabled ? 0 : TEXT_DRAW_DISABLE );
116 :
117 : // *** drag button ***
118 :
119 0 : bool bMain = (nAngle % 4500) != 0;
120 0 : SetLineColor( GetButtonLineColor() );
121 0 : SetFillColor( GetButtonFillColor( bMain ) );
122 :
123 0 : nX = mnCenterX - static_cast< long >( (DIAL_OUTER_WIDTH / 2 - mnCenterX) * fCos );
124 0 : nY = mnCenterY - static_cast< long >( (mnCenterY - DIAL_OUTER_WIDTH / 2) * fSin );
125 0 : long nSize = bMain ? (DIAL_OUTER_WIDTH / 4) : (DIAL_OUTER_WIDTH / 2 - 1);
126 0 : DrawEllipse( Rectangle( nX - nSize, nY - nSize, nX + nSize, nY + nSize ) );
127 0 : }
128 :
129 : // private --------------------------------------------------------------------
130 :
131 0 : const Color& DialControlBmp::GetBackgroundColor() const
132 : {
133 0 : return GetSettings().GetStyleSettings().GetDialogColor();
134 : }
135 :
136 0 : const Color& DialControlBmp::GetTextColor() const
137 : {
138 0 : return GetSettings().GetStyleSettings().GetLabelTextColor();
139 : }
140 :
141 0 : const Color& DialControlBmp::GetScaleLineColor() const
142 : {
143 0 : const StyleSettings& rSett = GetSettings().GetStyleSettings();
144 0 : return mbEnabled ? rSett.GetButtonTextColor() : rSett.GetDisableColor();
145 : }
146 :
147 0 : const Color& DialControlBmp::GetButtonLineColor() const
148 : {
149 0 : const StyleSettings& rSett = GetSettings().GetStyleSettings();
150 0 : return mbEnabled ? rSett.GetButtonTextColor() : rSett.GetDisableColor();
151 : }
152 :
153 0 : const Color& DialControlBmp::GetButtonFillColor( bool bMain ) const
154 : {
155 0 : const StyleSettings& rSett = GetSettings().GetStyleSettings();
156 0 : return mbEnabled ? (bMain ? rSett.GetMenuColor() : rSett.GetHighlightColor()) : rSett.GetDisableColor();
157 : }
158 :
159 0 : void DialControlBmp::Init( const Size& rSize )
160 : {
161 0 : SetSettings( mrParent.GetSettings() );
162 0 : maRect.SetPos( Point( 0, 0 ) );
163 0 : maRect.SetSize( rSize );
164 0 : mnCenterX = rSize.Width() / 2;
165 0 : mnCenterY = rSize.Height() / 2;
166 0 : SetOutputSize( rSize );
167 0 : SetBackground();
168 0 : }
169 :
170 0 : void DialControlBmp::DrawBackground()
171 : {
172 : // *** background with 3D effect ***
173 :
174 0 : SetLineColor();
175 0 : SetFillColor();
176 0 : Erase();
177 :
178 0 : EnableRTL( sal_True ); // draw 3D effect in correct direction
179 :
180 0 : sal_uInt8 nDiff = mbEnabled ? 0x18 : 0x10;
181 0 : Color aColor;
182 :
183 0 : aColor = GetBackgroundColor();
184 0 : SetFillColor( aColor );
185 0 : DrawPie( maRect, maRect.TopRight(), maRect.TopCenter() );
186 0 : DrawPie( maRect, maRect.BottomLeft(), maRect.BottomCenter() );
187 :
188 0 : aColor.DecreaseLuminance( nDiff );
189 0 : SetFillColor( aColor );
190 0 : DrawPie( maRect, maRect.BottomCenter(), maRect.TopRight() );
191 :
192 0 : aColor.DecreaseLuminance( nDiff );
193 0 : SetFillColor( aColor );
194 0 : DrawPie( maRect, maRect.BottomRight(), maRect.RightCenter() );
195 :
196 0 : aColor = GetBackgroundColor();
197 0 : aColor.IncreaseLuminance( nDiff );
198 0 : SetFillColor( aColor );
199 0 : DrawPie( maRect, maRect.TopCenter(), maRect.BottomLeft() );
200 :
201 0 : aColor.IncreaseLuminance( nDiff );
202 0 : SetFillColor( aColor );
203 0 : DrawPie( maRect, maRect.TopLeft(), maRect.LeftCenter() );
204 :
205 0 : EnableRTL( sal_False );
206 :
207 : // *** calibration ***
208 :
209 0 : Point aStartPos( mnCenterX, mnCenterY );
210 0 : Color aFullColor( GetScaleLineColor() );
211 0 : Color aLightColor( GetBackgroundColor() );
212 0 : aLightColor.Merge( aFullColor, 128 );
213 :
214 0 : for( int nAngle = 0; nAngle < 360; nAngle += 15 )
215 : {
216 0 : SetLineColor( (nAngle % 45) ? aLightColor : aFullColor );
217 0 : double fAngle = nAngle * F_PI180;
218 0 : long nX = static_cast< long >( -mnCenterX * cos( fAngle ) );
219 0 : long nY = static_cast< long >( mnCenterY * sin( fAngle ) );
220 0 : DrawLine( aStartPos, Point( mnCenterX - nX, mnCenterY - nY ) );
221 : }
222 :
223 : // *** clear inner area ***
224 :
225 0 : SetLineColor();
226 0 : SetFillColor( GetBackgroundColor() );
227 0 : DrawEllipse( Rectangle( maRect.Left() + DIAL_OUTER_WIDTH, maRect.Top() + DIAL_OUTER_WIDTH,
228 0 : maRect.Right() - DIAL_OUTER_WIDTH, maRect.Bottom() - DIAL_OUTER_WIDTH ) );
229 0 : }
230 :
231 : // ============================================================================
232 :
233 0 : struct DialControl_Impl
234 : {
235 : DialControlBmp maBmpEnabled;
236 : DialControlBmp maBmpDisabled;
237 : DialControlBmp maBmpBuffered;
238 : Link maModifyHdl;
239 : NumericField* mpLinkField;
240 : Size maWinSize;
241 : Font maWinFont;
242 : sal_Int32 mnAngle;
243 : sal_Int32 mnInitialAngle;
244 : sal_Int32 mnOldAngle;
245 : long mnCenterX;
246 : long mnCenterY;
247 : bool mbNoRot;
248 :
249 : explicit DialControl_Impl( Window& rParent );
250 : void Init( const Size& rWinSize, const Font& rWinFont );
251 : };
252 :
253 : // ----------------------------------------------------------------------------
254 :
255 0 : DialControl_Impl::DialControl_Impl( Window& rParent ) :
256 : maBmpEnabled( rParent ),
257 : maBmpDisabled( rParent ),
258 : maBmpBuffered( rParent ),
259 : mpLinkField( 0 ),
260 : mnAngle( 0 ),
261 : mnInitialAngle( 0 ),
262 : mnCenterX( 0 ),
263 : mnCenterY( 0 ),
264 0 : mbNoRot( false )
265 : {
266 0 : }
267 :
268 0 : void DialControl_Impl::Init( const Size& rWinSize, const Font& rWinFont )
269 : {
270 : // "(x - 1) | 1" creates odd value <= x, to have a well-defined center pixel position
271 0 : maWinSize = Size( (rWinSize.Width() - 1) | 1, (rWinSize.Height() - 1) | 1 );
272 0 : maWinFont = rWinFont;
273 :
274 0 : mnCenterX = maWinSize.Width() / 2;
275 0 : mnCenterY = maWinSize.Height() / 2;
276 0 : maWinFont.SetTransparent( sal_True );
277 :
278 0 : maBmpEnabled.DrawBackground( maWinSize, true );
279 0 : maBmpDisabled.DrawBackground( maWinSize, false );
280 0 : maBmpBuffered.InitBitmap( maWinSize, maWinFont );
281 0 : }
282 :
283 : // ============================================================================
284 :
285 0 : DialControl::DialControl( Window* pParent, const ResId& rResId ) :
286 : Control( pParent, rResId ),
287 0 : mpImpl( new DialControl_Impl( *this ) )
288 : {
289 0 : Init( GetOutputSizePixel() );
290 0 : }
291 :
292 0 : DialControl::~DialControl()
293 : {
294 0 : }
295 :
296 0 : void DialControl::Paint( const Rectangle& )
297 : {
298 0 : Point aPos;
299 0 : DrawBitmapEx( aPos, mpImpl->maBmpBuffered.GetBitmapEx( aPos, mpImpl->maWinSize ) );
300 0 : }
301 :
302 0 : void DialControl::StateChanged( StateChangedType nStateChange )
303 : {
304 0 : if( nStateChange == STATE_CHANGE_ENABLE )
305 0 : InvalidateControl();
306 :
307 : // update the linked edit field
308 0 : if( mpImpl->mpLinkField )
309 : {
310 0 : NumericField& rField = *mpImpl->mpLinkField;
311 0 : switch( nStateChange )
312 : {
313 0 : case STATE_CHANGE_VISIBLE: rField.Show( IsVisible() ); break;
314 0 : case STATE_CHANGE_ENABLE: rField.Enable( IsEnabled() ); break;
315 : }
316 : }
317 :
318 0 : Control::StateChanged( nStateChange );
319 0 : }
320 :
321 0 : void DialControl::DataChanged( const DataChangedEvent& rDCEvt )
322 : {
323 0 : if( (rDCEvt.GetType() == DATACHANGED_SETTINGS) && (rDCEvt.GetFlags() & SETTINGS_STYLE) )
324 : {
325 0 : Init( mpImpl->maWinSize, mpImpl->maWinFont );
326 0 : InvalidateControl();
327 : }
328 0 : Control::DataChanged( rDCEvt );
329 0 : }
330 :
331 0 : void DialControl::MouseButtonDown( const MouseEvent& rMEvt )
332 : {
333 0 : if( rMEvt.IsLeft() )
334 : {
335 0 : GrabFocus();
336 0 : CaptureMouse();
337 0 : mpImpl->mnOldAngle = mpImpl->mnAngle;
338 0 : HandleMouseEvent( rMEvt.GetPosPixel(), true );
339 : }
340 0 : Control::MouseButtonDown( rMEvt );
341 0 : }
342 :
343 0 : void DialControl::MouseMove( const MouseEvent& rMEvt )
344 : {
345 0 : if( IsMouseCaptured() && rMEvt.IsLeft() )
346 0 : HandleMouseEvent( rMEvt.GetPosPixel(), false );
347 0 : Control::MouseMove(rMEvt );
348 0 : }
349 :
350 0 : void DialControl::MouseButtonUp( const MouseEvent& rMEvt )
351 : {
352 0 : if( IsMouseCaptured() )
353 : {
354 0 : ReleaseMouse();
355 0 : if( mpImpl->mpLinkField )
356 0 : mpImpl->mpLinkField->GrabFocus();
357 : }
358 0 : Control::MouseButtonUp( rMEvt );
359 0 : }
360 :
361 0 : void DialControl::KeyInput( const KeyEvent& rKEvt )
362 : {
363 0 : const KeyCode& rKCode = rKEvt.GetKeyCode();
364 0 : if( !rKCode.GetModifier() && (rKCode.GetCode() == KEY_ESCAPE) )
365 0 : HandleEscapeEvent();
366 : else
367 0 : Control::KeyInput( rKEvt );
368 0 : }
369 :
370 0 : void DialControl::LoseFocus()
371 : {
372 : // release captured mouse
373 0 : HandleEscapeEvent();
374 0 : Control::LoseFocus();
375 0 : }
376 :
377 0 : bool DialControl::HasRotation() const
378 : {
379 0 : return !mpImpl->mbNoRot;
380 : }
381 :
382 0 : void DialControl::SetNoRotation()
383 : {
384 0 : if( !mpImpl->mbNoRot )
385 : {
386 0 : mpImpl->mbNoRot = true;
387 0 : InvalidateControl();
388 0 : if( mpImpl->mpLinkField )
389 0 : mpImpl->mpLinkField->SetText( String() );
390 : }
391 0 : }
392 :
393 0 : sal_Int32 DialControl::GetRotation() const
394 : {
395 0 : return mpImpl->mnAngle;
396 : }
397 :
398 0 : void DialControl::SetRotation( sal_Int32 nAngle )
399 : {
400 0 : ImplSetRotation( nAngle, false );
401 0 : }
402 :
403 0 : void DialControl::SetLinkedField( NumericField* pField )
404 : {
405 : // remove modify handler from old linked field
406 0 : ImplSetFieldLink( Link() );
407 : // remember the new linked field
408 0 : mpImpl->mpLinkField = pField;
409 : // set modify handler at new linked field
410 0 : ImplSetFieldLink( LINK( this, DialControl, LinkedFieldModifyHdl ) );
411 0 : }
412 :
413 0 : void DialControl::SaveValue()
414 : {
415 0 : mpImpl->mnInitialAngle = mpImpl->mnAngle;
416 0 : }
417 :
418 0 : bool DialControl::IsValueModified()
419 : {
420 0 : return mpImpl->mnInitialAngle != mpImpl->mnAngle;
421 : }
422 :
423 : // private --------------------------------------------------------------------
424 :
425 0 : void DialControl::Init( const Size& rWinSize, const Font& rWinFont )
426 : {
427 0 : mpImpl->Init( rWinSize, rWinFont );
428 0 : EnableRTL( sal_False ); // don't mirror mouse handling
429 0 : SetOutputSizePixel( mpImpl->maWinSize );
430 0 : SetBackground();
431 0 : }
432 :
433 0 : void DialControl::Init( const Size& rWinSize )
434 : {
435 : Font aFont( OutputDevice::GetDefaultFont(
436 0 : DEFAULTFONT_UI_SANS, Application::GetSettings().GetUILanguageTag().getLanguageType(), DEFAULTFONT_FLAGS_ONLYONE ) );
437 0 : Init( rWinSize, aFont );
438 0 : }
439 :
440 0 : void DialControl::InvalidateControl()
441 : {
442 0 : mpImpl->maBmpBuffered.CopyBackground( IsEnabled() ? mpImpl->maBmpEnabled : mpImpl->maBmpDisabled );
443 0 : if( !mpImpl->mbNoRot )
444 0 : mpImpl->maBmpBuffered.DrawElements( GetText(), mpImpl->mnAngle );
445 0 : Invalidate();
446 0 : }
447 :
448 0 : void DialControl::ImplSetRotation( sal_Int32 nAngle, bool bBroadcast )
449 : {
450 0 : bool bOldSel = mpImpl->mbNoRot;
451 0 : mpImpl->mbNoRot = false;
452 :
453 0 : while( nAngle < 0 ) nAngle += 36000;
454 0 : nAngle = (((nAngle + 50) / 100) * 100) % 36000;
455 0 : if( !bOldSel || (mpImpl->mnAngle != nAngle) )
456 : {
457 0 : mpImpl->mnAngle = nAngle;
458 0 : InvalidateControl();
459 0 : if( mpImpl->mpLinkField )
460 0 : mpImpl->mpLinkField->SetValue( static_cast< long >( GetRotation() / 100 ) );
461 0 : if( bBroadcast )
462 0 : mpImpl->maModifyHdl.Call( this );
463 : }
464 0 : }
465 :
466 0 : void DialControl::ImplSetFieldLink( const Link& rLink )
467 : {
468 0 : if( mpImpl->mpLinkField )
469 : {
470 0 : NumericField& rField = *mpImpl->mpLinkField;
471 0 : rField.SetModifyHdl( rLink );
472 0 : rField.SetUpHdl( rLink );
473 0 : rField.SetDownHdl( rLink );
474 0 : rField.SetFirstHdl( rLink );
475 0 : rField.SetLastHdl( rLink );
476 0 : rField.SetLoseFocusHdl( rLink );
477 : }
478 0 : }
479 :
480 0 : void DialControl::HandleMouseEvent( const Point& rPos, bool bInitial )
481 : {
482 0 : long nX = rPos.X() - mpImpl->mnCenterX;
483 0 : long nY = mpImpl->mnCenterY - rPos.Y();
484 0 : double fH = sqrt( static_cast< double >( nX ) * nX + static_cast< double >( nY ) * nY );
485 0 : if( fH != 0.0 )
486 : {
487 0 : double fAngle = acos( nX / fH );
488 0 : sal_Int32 nAngle = static_cast< sal_Int32 >( fAngle / F_PI180 * 100.0 );
489 0 : if( nY < 0 )
490 0 : nAngle = 36000 - nAngle;
491 0 : if( bInitial ) // round to entire 15 degrees
492 0 : nAngle = ((nAngle + 750) / 1500) * 1500;
493 0 : ImplSetRotation( nAngle, true );
494 : }
495 0 : }
496 :
497 0 : void DialControl::HandleEscapeEvent()
498 : {
499 0 : if( IsMouseCaptured() )
500 : {
501 0 : ReleaseMouse();
502 0 : ImplSetRotation( mpImpl->mnOldAngle, true );
503 0 : if( mpImpl->mpLinkField )
504 0 : mpImpl->mpLinkField->GrabFocus();
505 : }
506 0 : }
507 :
508 0 : IMPL_LINK( DialControl, LinkedFieldModifyHdl, NumericField*, pField )
509 : {
510 0 : if( pField )
511 0 : ImplSetRotation( static_cast< sal_Int32 >( pField->GetValue() * 100 ), false );
512 0 : return 0;
513 : }
514 :
515 : // ============================================================================
516 :
517 0 : DialControlWrapper::DialControlWrapper( DialControl& rDial ) :
518 0 : SingleControlWrapperType( rDial )
519 : {
520 0 : }
521 :
522 0 : bool DialControlWrapper::IsControlDontKnow() const
523 : {
524 0 : return !GetControl().HasRotation();
525 : }
526 :
527 0 : void DialControlWrapper::SetControlDontKnow( bool bSet )
528 : {
529 0 : if( bSet )
530 0 : GetControl().SetNoRotation();
531 0 : }
532 :
533 0 : sal_Int32 DialControlWrapper::GetControlValue() const
534 : {
535 0 : return GetControl().GetRotation();
536 : }
537 :
538 0 : void DialControlWrapper::SetControlValue( sal_Int32 nValue )
539 : {
540 0 : GetControl().SetRotation( nValue );
541 0 : }
542 :
543 : // ============================================================================
544 :
545 : } // namespace svx
546 :
547 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|