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 :
21 : #include "ViewTabBar.hxx"
22 :
23 : #define USE_TAB_CONTROL
24 :
25 : #include "ViewShell.hxx"
26 : #include "ViewShellBase.hxx"
27 : #include "DrawViewShell.hxx"
28 : #include "FrameView.hxx"
29 : #include "EventMultiplexer.hxx"
30 : #include "framework/FrameworkHelper.hxx"
31 : #include "framework/Pane.hxx"
32 : #include "DrawController.hxx"
33 :
34 : #include "sdresid.hxx"
35 : #include "strings.hrc"
36 : #include "helpids.h"
37 : #include "Client.hxx"
38 : #include <vcl/svapp.hxx>
39 : #include <vcl/tabpage.hxx>
40 : #include <osl/mutex.hxx>
41 : #include <sfx2/viewfrm.hxx>
42 : #include <com/sun/star/drawing/framework/ResourceId.hpp>
43 : #include <com/sun/star/drawing/framework/XControllerManager.hpp>
44 : #include <com/sun/star/lang/XUnoTunnel.hpp>
45 : #include <com/sun/star/lang/DisposedException.hpp>
46 : #include <comphelper/processfactory.hxx>
47 : #include <comphelper/servicehelper.hxx>
48 : #include <tools/diagnose_ex.h>
49 :
50 : using namespace ::com::sun::star;
51 : using namespace ::com::sun::star::uno;
52 : using namespace ::com::sun::star::drawing::framework;
53 : using ::sd::framework::FrameworkHelper;
54 : using ::sd::tools::EventMultiplexerEvent;
55 : using ::rtl::OUString;
56 :
57 : namespace sd {
58 :
59 : namespace {
60 0 : bool IsEqual (const TabBarButton& rButton1, const TabBarButton& rButton2)
61 : {
62 : return (
63 0 : (rButton1.ResourceId.is()
64 0 : && rButton2.ResourceId.is()
65 0 : && rButton1.ResourceId->compareTo(rButton2.ResourceId)==0)
66 0 : || rButton1.ButtonLabel == rButton2.ButtonLabel);
67 : }
68 :
69 0 : class TabBarControl : public ::TabControl
70 : {
71 : public:
72 : TabBarControl (
73 : ::Window* pParentWindow,
74 : const ::rtl::Reference<ViewTabBar>& rpViewTabBar);
75 : virtual void Paint (const Rectangle& rRect);
76 : virtual void ActivatePage (void);
77 : private:
78 : ::rtl::Reference<ViewTabBar> mpViewTabBar;
79 : };
80 :
81 : } // end of anonymous namespace
82 :
83 :
84 :
85 :
86 :
87 0 : class ViewTabPage : public TabPage
88 : {
89 : public:
90 : ViewTabPage (Window* pParent) : TabPage(pParent) {}
91 0 : virtual void Resize (void)
92 0 : { SetPosSizePixel(Point(0,0),GetParent()->GetOutputSizePixel()); }
93 : };
94 :
95 :
96 :
97 :
98 : //===== ViewTabBar ============================================================
99 :
100 0 : ViewTabBar::ViewTabBar (
101 : const Reference<XResourceId>& rxViewTabBarId,
102 : const Reference<frame::XController>& rxController)
103 : : ViewTabBarInterfaceBase(maMutex),
104 0 : mpTabControl(new TabBarControl(GetAnchorWindow(rxViewTabBarId,rxController), this)),
105 : mxController(rxController),
106 : maTabBarButtons(),
107 : mpTabPage(NULL),
108 : mxViewTabBarId(rxViewTabBarId),
109 0 : mpViewShellBase(NULL)
110 : {
111 : // Set one new tab page for all tab entries. We need it only to
112 : // determine the height of the tab bar.
113 0 : mpTabPage.reset(new TabPage (mpTabControl.get()));
114 0 : mpTabPage->Hide();
115 :
116 : // add some space before the tabitems
117 0 : mpTabControl->SetItemsOffset(Point(5, 3));
118 :
119 : // Tunnel through the controller and use the ViewShellBase to obtain the
120 : // view frame.
121 : try
122 : {
123 0 : Reference<lang::XUnoTunnel> xTunnel (mxController, UNO_QUERY_THROW);
124 : DrawController* pController = reinterpret_cast<DrawController*>(
125 0 : xTunnel->getSomething(DrawController::getUnoTunnelId()));
126 0 : mpViewShellBase = pController->GetViewShellBase();
127 : }
128 0 : catch (const RuntimeException&)
129 : {
130 : }
131 :
132 : // Register as listener at XConfigurationController.
133 0 : Reference<XControllerManager> xControllerManager (mxController, UNO_QUERY);
134 0 : if (xControllerManager.is())
135 : {
136 0 : mxConfigurationController = xControllerManager->getConfigurationController();
137 0 : if (mxConfigurationController.is())
138 : {
139 0 : mxConfigurationController->addConfigurationChangeListener(
140 : this,
141 : FrameworkHelper::msResourceActivationEvent,
142 0 : Any());
143 : }
144 : }
145 :
146 0 : mpTabControl->Show();
147 :
148 0 : if (mpViewShellBase != NULL
149 0 : && rxViewTabBarId->isBoundToURL(
150 0 : FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT))
151 : {
152 0 : mpViewShellBase->SetViewTabBar(this);
153 0 : }
154 0 : }
155 :
156 :
157 :
158 :
159 0 : ViewTabBar::~ViewTabBar (void)
160 : {
161 0 : }
162 :
163 :
164 :
165 :
166 0 : void ViewTabBar::disposing (void)
167 : {
168 0 : if (mpViewShellBase != NULL
169 0 : && mxViewTabBarId->isBoundToURL(
170 0 : FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT))
171 : {
172 0 : mpViewShellBase->SetViewTabBar(NULL);
173 : }
174 :
175 0 : if (mxConfigurationController.is())
176 : {
177 : // Unregister listener from XConfigurationController.
178 : try
179 : {
180 0 : mxConfigurationController->removeConfigurationChangeListener(this);
181 : }
182 0 : catch (const lang::DisposedException&)
183 : {
184 : // Receiving a disposed exception is the normal case. Is there
185 : // a way to avoid it?
186 : }
187 0 : mxConfigurationController = NULL;
188 : }
189 :
190 : {
191 0 : const SolarMutexGuard aSolarGuard;
192 : // Set all references to the one tab page to NULL and delete the page.
193 0 : for (sal_uInt16 nIndex=0; nIndex<mpTabControl->GetPageCount(); ++nIndex)
194 0 : mpTabControl->SetTabPage(nIndex, NULL);
195 0 : mpTabPage.reset();
196 0 : mpTabControl.reset();
197 : }
198 :
199 0 : mxController = NULL;
200 0 : }
201 :
202 :
203 :
204 :
205 0 : ::boost::shared_ptr< ::TabControl> ViewTabBar::GetTabControl (void) const
206 : {
207 0 : return mpTabControl;
208 : }
209 :
210 :
211 :
212 :
213 0 : ::Window* ViewTabBar::GetAnchorWindow(
214 : const Reference<XResourceId>& rxViewTabBarId,
215 : const Reference<frame::XController>& rxController)
216 : {
217 0 : ::Window* pWindow = NULL;
218 0 : ViewShellBase* pBase = NULL;
219 :
220 : // Tunnel through the controller and use the ViewShellBase to obtain the
221 : // view frame.
222 : try
223 : {
224 0 : Reference<lang::XUnoTunnel> xTunnel (rxController, UNO_QUERY_THROW);
225 : DrawController* pController = reinterpret_cast<DrawController*>(
226 0 : xTunnel->getSomething(DrawController::getUnoTunnelId()));
227 0 : pBase = pController->GetViewShellBase();
228 : }
229 0 : catch (const RuntimeException&)
230 : {
231 : }
232 :
233 : // The ViewTabBar supports at the moment only the center pane.
234 0 : if (rxViewTabBarId.is()
235 0 : && rxViewTabBarId->isBoundToURL(
236 0 : FrameworkHelper::msCenterPaneURL, AnchorBindingMode_DIRECT))
237 : {
238 0 : if (pBase != NULL && pBase->GetViewFrame() != NULL)
239 0 : pWindow = &pBase->GetViewFrame()->GetWindow();
240 : }
241 :
242 : // The rest is (at the moment) just for the emergency case.
243 0 : if (pWindow == NULL)
244 : {
245 0 : Reference<XPane> xPane;
246 : try
247 : {
248 0 : Reference<XControllerManager> xControllerManager (rxController, UNO_QUERY_THROW);
249 : Reference<XConfigurationController> xCC (
250 0 : xControllerManager->getConfigurationController());
251 0 : if (xCC.is())
252 0 : xPane = Reference<XPane>(xCC->getResource(rxViewTabBarId->getAnchor()), UNO_QUERY);
253 : }
254 0 : catch (const RuntimeException&)
255 : {
256 : }
257 :
258 : // Tunnel through the XWindow to the VCL side.
259 : try
260 : {
261 0 : Reference<lang::XUnoTunnel> xTunnel (xPane, UNO_QUERY_THROW);
262 : framework::Pane* pPane = reinterpret_cast<framework::Pane*>(
263 0 : xTunnel->getSomething(framework::Pane::getUnoTunnelId()));
264 0 : if (pPane != NULL)
265 0 : pWindow = pPane->GetWindow()->GetParent();
266 : }
267 0 : catch (const RuntimeException&)
268 : {
269 0 : }
270 : }
271 :
272 0 : return pWindow;
273 : }
274 :
275 :
276 :
277 :
278 : //----- XConfigurationChangeListener ------------------------------------------
279 :
280 0 : void SAL_CALL ViewTabBar::notifyConfigurationChange (
281 : const ConfigurationChangeEvent& rEvent)
282 : throw (RuntimeException)
283 : {
284 0 : if (rEvent.Type.equals(FrameworkHelper::msResourceActivationEvent)
285 0 : && rEvent.ResourceId->getResourceURL().match(FrameworkHelper::msViewURLPrefix)
286 0 : && rEvent.ResourceId->isBoundTo(mxViewTabBarId->getAnchor(), AnchorBindingMode_DIRECT))
287 : {
288 0 : UpdateActiveButton();
289 : }
290 0 : }
291 :
292 :
293 :
294 :
295 : //----- XEventListener --------------------------------------------------------
296 :
297 0 : void SAL_CALL ViewTabBar::disposing(
298 : const lang::EventObject& rEvent)
299 : throw (RuntimeException)
300 : {
301 0 : if (rEvent.Source == mxConfigurationController)
302 : {
303 0 : mxConfigurationController = NULL;
304 0 : mxController = NULL;
305 : }
306 0 : }
307 :
308 :
309 :
310 :
311 : //----- XTabBar ---------------------------------------------------------------
312 :
313 0 : void SAL_CALL ViewTabBar::addTabBarButtonAfter (
314 : const TabBarButton& rButton,
315 : const TabBarButton& rAnchor)
316 : throw (::com::sun::star::uno::RuntimeException)
317 : {
318 0 : const SolarMutexGuard aSolarGuard;
319 0 : AddTabBarButton(rButton, rAnchor);
320 0 : }
321 :
322 :
323 :
324 :
325 0 : void SAL_CALL ViewTabBar::appendTabBarButton (const TabBarButton& rButton)
326 : throw (::com::sun::star::uno::RuntimeException)
327 : {
328 0 : const SolarMutexGuard aSolarGuard;
329 0 : AddTabBarButton(rButton);
330 0 : }
331 :
332 :
333 :
334 0 : void SAL_CALL ViewTabBar::removeTabBarButton (const TabBarButton& rButton)
335 : throw (::com::sun::star::uno::RuntimeException)
336 : {
337 0 : const SolarMutexGuard aSolarGuard;
338 0 : RemoveTabBarButton(rButton);
339 0 : }
340 :
341 :
342 :
343 :
344 0 : sal_Bool SAL_CALL ViewTabBar::hasTabBarButton (const TabBarButton& rButton)
345 : throw (::com::sun::star::uno::RuntimeException)
346 : {
347 0 : const SolarMutexGuard aSolarGuard;
348 0 : return HasTabBarButton(rButton);
349 : }
350 :
351 :
352 :
353 :
354 0 : Sequence<TabBarButton> SAL_CALL ViewTabBar::getTabBarButtons (void)
355 : throw (::com::sun::star::uno::RuntimeException)
356 : {
357 0 : const SolarMutexGuard aSolarGuard;
358 0 : return GetTabBarButtons();
359 : }
360 :
361 :
362 :
363 :
364 : //----- XResource -------------------------------------------------------------
365 :
366 0 : Reference<XResourceId> SAL_CALL ViewTabBar::getResourceId (void)
367 : throw (RuntimeException)
368 : {
369 0 : return mxViewTabBarId;
370 : }
371 :
372 :
373 :
374 :
375 0 : sal_Bool SAL_CALL ViewTabBar::isAnchorOnly (void)
376 : throw (RuntimeException)
377 : {
378 0 : return false;
379 : }
380 :
381 :
382 :
383 :
384 : //----- XUnoTunnel ------------------------------------------------------------
385 :
386 : namespace
387 : {
388 : class theViewTabBarUnoTunnelId : public rtl::Static< UnoTunnelIdInit, theViewTabBarUnoTunnelId > {};
389 : }
390 :
391 0 : const Sequence<sal_Int8>& ViewTabBar::getUnoTunnelId (void)
392 : {
393 0 : return theViewTabBarUnoTunnelId::get().getSeq();
394 : }
395 :
396 0 : sal_Int64 SAL_CALL ViewTabBar::getSomething (const Sequence<sal_Int8>& rId)
397 : throw (RuntimeException)
398 : {
399 0 : sal_Int64 nResult = 0;
400 :
401 0 : if (rId.getLength() == 16
402 0 : && memcmp(getUnoTunnelId().getConstArray(), rId.getConstArray(), 16) == 0)
403 : {
404 0 : nResult = reinterpret_cast<sal_Int64>(this);
405 : }
406 :
407 0 : return nResult;
408 : }
409 :
410 :
411 :
412 :
413 : //-----------------------------------------------------------------------------
414 :
415 0 : bool ViewTabBar::ActivatePage (void)
416 : {
417 : try
418 : {
419 0 : Reference<XControllerManager> xControllerManager (mxController,UNO_QUERY_THROW);
420 : Reference<XConfigurationController> xConfigurationController (
421 0 : xControllerManager->getConfigurationController());
422 0 : if ( ! xConfigurationController.is())
423 0 : throw RuntimeException();
424 0 : Reference<XView> xView;
425 : try
426 : {
427 0 : xView = Reference<XView>(xConfigurationController->getResource(
428 : ResourceId::create(
429 : ::comphelper::getProcessComponentContext(),
430 0 : FrameworkHelper::msCenterPaneURL)),
431 0 : UNO_QUERY);
432 : }
433 0 : catch (const DeploymentException&)
434 : {
435 : }
436 :
437 0 : Client* pIPClient = NULL;
438 0 : if (mpViewShellBase != NULL)
439 0 : pIPClient = dynamic_cast<Client*>(mpViewShellBase->GetIPClient());
440 0 : if (pIPClient==NULL || ! pIPClient->IsObjectInPlaceActive())
441 : {
442 0 : sal_uInt16 nIndex (mpTabControl->GetCurPageId() - 1);
443 0 : if (nIndex < maTabBarButtons.size())
444 : {
445 0 : xConfigurationController->requestResourceActivation(
446 0 : maTabBarButtons[nIndex].ResourceId,
447 0 : ResourceActivationMode_REPLACE);
448 : }
449 :
450 0 : return true;
451 : }
452 : else
453 : {
454 : // When we run into this else branch then we have an active OLE
455 : // object. We ignore the request to switch views. Additionally
456 : // we put the active tab back to the one for the current view.
457 0 : UpdateActiveButton();
458 0 : }
459 : }
460 0 : catch (const RuntimeException&)
461 : {
462 : DBG_UNHANDLED_EXCEPTION();
463 : }
464 :
465 0 : return false;
466 : }
467 :
468 :
469 :
470 :
471 0 : int ViewTabBar::GetHeight (void)
472 : {
473 0 : int nHeight (0);
474 :
475 0 : if (!maTabBarButtons.empty())
476 : {
477 : TabPage* pActivePage (mpTabControl->GetTabPage(
478 0 : mpTabControl->GetCurPageId()));
479 0 : if (pActivePage!=NULL && mpTabControl->IsReallyVisible())
480 0 : nHeight = pActivePage->GetPosPixel().Y();
481 :
482 0 : if (nHeight <= 0)
483 : // Using a default when the real height can not be determined.
484 : // To get correct height this method should be called when the
485 : // control is visible.
486 0 : nHeight = 21;
487 : }
488 :
489 0 : return nHeight;
490 : }
491 :
492 :
493 :
494 :
495 0 : void ViewTabBar::AddTabBarButton (
496 : const ::com::sun::star::drawing::framework::TabBarButton& rButton,
497 : const ::com::sun::star::drawing::framework::TabBarButton& rAnchor)
498 : {
499 : sal_uInt32 nIndex;
500 :
501 0 : if ( ! rAnchor.ResourceId.is()
502 0 : || (rAnchor.ResourceId->getResourceURL().isEmpty()
503 0 : && rAnchor.ButtonLabel.isEmpty()))
504 : {
505 0 : nIndex = 0;
506 : }
507 : else
508 : {
509 0 : for (nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex)
510 : {
511 0 : if (IsEqual(maTabBarButtons[nIndex], rAnchor))
512 : {
513 0 : ++nIndex;
514 0 : break;
515 : }
516 : }
517 : }
518 :
519 0 : AddTabBarButton(rButton,nIndex);
520 0 : }
521 :
522 :
523 :
524 :
525 0 : void ViewTabBar::AddTabBarButton (
526 : const ::com::sun::star::drawing::framework::TabBarButton& rButton)
527 : {
528 0 : AddTabBarButton(rButton, maTabBarButtons.size());
529 0 : }
530 :
531 :
532 :
533 :
534 0 : void ViewTabBar::AddTabBarButton (
535 : const ::com::sun::star::drawing::framework::TabBarButton& rButton,
536 : sal_Int32 nPosition)
537 : {
538 0 : if (nPosition>=0
539 0 : && nPosition<=mpTabControl->GetPageCount())
540 : {
541 0 : sal_uInt16 nIndex ((sal_uInt16)nPosition);
542 :
543 : // Insert the button into our local array.
544 0 : maTabBarButtons.insert(maTabBarButtons.begin()+nIndex, rButton);
545 0 : UpdateTabBarButtons();
546 0 : UpdateActiveButton();
547 : }
548 0 : }
549 :
550 :
551 :
552 :
553 0 : void ViewTabBar::RemoveTabBarButton (
554 : const ::com::sun::star::drawing::framework::TabBarButton& rButton)
555 : {
556 : sal_uInt16 nIndex;
557 0 : for (nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex)
558 : {
559 0 : if (IsEqual(maTabBarButtons[nIndex], rButton))
560 : {
561 0 : maTabBarButtons.erase(maTabBarButtons.begin()+nIndex);
562 0 : UpdateTabBarButtons();
563 0 : UpdateActiveButton();
564 0 : break;
565 : }
566 : }
567 0 : }
568 :
569 :
570 :
571 :
572 0 : bool ViewTabBar::HasTabBarButton (
573 : const ::com::sun::star::drawing::framework::TabBarButton& rButton)
574 : {
575 0 : bool bResult (false);
576 :
577 0 : for (sal_uInt32 nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex)
578 : {
579 0 : if (IsEqual(maTabBarButtons[nIndex], rButton))
580 : {
581 0 : bResult = true;
582 0 : break;
583 : }
584 : }
585 :
586 0 : return bResult;
587 : }
588 :
589 :
590 :
591 :
592 : ::com::sun::star::uno::Sequence<com::sun::star::drawing::framework::TabBarButton>
593 0 : ViewTabBar::GetTabBarButtons (void)
594 : {
595 0 : sal_uInt32 nCount (maTabBarButtons.size());
596 : ::com::sun::star::uno::Sequence<com::sun::star::drawing::framework::TabBarButton>
597 0 : aList (nCount);
598 :
599 0 : for (sal_uInt32 nIndex=0; nIndex<nCount; ++nIndex)
600 0 : aList[nIndex] = maTabBarButtons[nIndex];
601 :
602 0 : return aList;
603 : }
604 :
605 :
606 :
607 :
608 0 : void ViewTabBar::UpdateActiveButton (void)
609 : {
610 0 : Reference<XView> xView;
611 0 : if (mpViewShellBase != NULL)
612 : xView = FrameworkHelper::Instance(*mpViewShellBase)->GetView(
613 0 : mxViewTabBarId->getAnchor());
614 0 : if (xView.is())
615 : {
616 0 : Reference<XResourceId> xViewId (xView->getResourceId());
617 0 : for (sal_uInt16 nIndex=0; nIndex<maTabBarButtons.size(); ++nIndex)
618 : {
619 0 : if (maTabBarButtons[nIndex].ResourceId->compareTo(xViewId) == 0)
620 : {
621 0 : mpTabControl->SetCurPageId(nIndex+1);
622 0 : mpTabControl->::TabControl::ActivatePage();
623 0 : break;
624 : }
625 0 : }
626 0 : }
627 0 : }
628 :
629 :
630 :
631 :
632 0 : void ViewTabBar::UpdateTabBarButtons (void)
633 : {
634 0 : TabBarButtonList::const_iterator iTab;
635 0 : sal_uInt16 nPageCount (mpTabControl->GetPageCount());
636 : sal_uInt16 nIndex;
637 0 : for (iTab=maTabBarButtons.begin(),nIndex=1; iTab!=maTabBarButtons.end(); ++iTab,++nIndex)
638 : {
639 : // Create a new tab when there are not enough.
640 0 : if (nPageCount < nIndex)
641 0 : mpTabControl->InsertPage(nIndex, iTab->ButtonLabel);
642 :
643 : // Update the tab.
644 0 : mpTabControl->SetPageText(nIndex, iTab->ButtonLabel);
645 0 : mpTabControl->SetHelpText(nIndex, iTab->HelpText);
646 0 : mpTabControl->SetTabPage(nIndex, mpTabPage.get());
647 : }
648 :
649 : // Delete tabs that are no longer used.
650 0 : for (; nIndex<=nPageCount; ++nIndex)
651 0 : mpTabControl->RemovePage(nIndex);
652 :
653 0 : mpTabPage->Hide();
654 0 : }
655 :
656 :
657 :
658 :
659 : //===== TabBarControl =========================================================
660 :
661 0 : TabBarControl::TabBarControl (
662 : ::Window* pParentWindow,
663 : const ::rtl::Reference<ViewTabBar>& rpViewTabBar)
664 : : ::TabControl(pParentWindow),
665 0 : mpViewTabBar(rpViewTabBar)
666 : {
667 0 : }
668 :
669 :
670 :
671 :
672 0 : void TabBarControl::Paint (const Rectangle& rRect)
673 : {
674 0 : Color aOriginalFillColor (GetFillColor());
675 0 : Color aOriginalLineColor (GetLineColor());
676 :
677 : // Because the actual window background is transparent--to avoid
678 : // flickering due to multiple background paintings by this and by child
679 : // windows--we have to paint the background for this control explicitly:
680 : // the actual control is not painted over its whole bounding box.
681 0 : SetFillColor (GetSettings().GetStyleSettings().GetDialogColor());
682 0 : SetLineColor ();
683 0 : DrawRect (rRect);
684 0 : ::TabControl::Paint (rRect);
685 :
686 0 : SetFillColor (aOriginalFillColor);
687 0 : SetLineColor (aOriginalLineColor);
688 0 : }
689 :
690 :
691 :
692 :
693 0 : void TabBarControl::ActivatePage (void)
694 : {
695 0 : if (mpViewTabBar->ActivatePage())
696 : {
697 : // Call the parent so that the correct tab is highlighted.
698 0 : this->::TabControl::ActivatePage();
699 : }
700 0 : }
701 :
702 : } // end of namespace sd
703 :
704 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|