src/hbcore/gui/hbwidget.cpp
branchGCC_SURGE
changeset 15 f378acbc9cfb
parent 7 923ff622b8b9
child 21 4633027730f5
child 34 ed14f46c0e55
equal deleted inserted replaced
9:730c025d4b77 15:f378acbc9cfb
    31 #include "hbstyleoption_p.h"
    31 #include "hbstyleoption_p.h"
    32 #include "hbstyleparameters.h"
    32 #include "hbstyleparameters.h"
    33 #include "hbgraphicsscene.h"
    33 #include "hbgraphicsscene.h"
    34 #include "hbgraphicsscene_p.h"
    34 #include "hbgraphicsscene_p.h"
    35 #include "hbdeviceprofile.h"
    35 #include "hbdeviceprofile.h"
    36 #include "hbspaceritem_p.h"
    36 #include "hbtapgesture.h"
       
    37 #include "hbnamespace_p.h"
    37 #include <QCoreApplication>
    38 #include <QCoreApplication>
    38 #include <QMetaType>
    39 #include <QMetaType>
    39 #include <QAction>
    40 #include <QAction>
    40 #include <QDebug>
    41 #include <QDebug>
    41 #include <QDynamicPropertyChangeEvent>
    42 #include <QDynamicPropertyChangeEvent>
    65     backgroundPrimitiveType(HbStyle::P_None),
    66     backgroundPrimitiveType(HbStyle::P_None),
    66     polished(false),
    67     polished(false),
    67     polishPending(false),
    68     polishPending(false),
    68     themingPending(true),
    69     themingPending(true),
    69     repolishOutstanding(false),
    70     repolishOutstanding(false),
       
    71     mHandlingRepolishSynchronously(false),
    70     notifyScene(false),
    72     notifyScene(false),
    71     pluginBaseId(0),
       
    72     focusGroup(0),
    73     focusGroup(0),
    73     focusActiveType(HbStyle::P_None),
    74     focusActiveType(HbStyle::P_None),
    74     focusResidualType(HbStyle::P_None),
    75     focusResidualType(HbStyle::P_None),
    75     highlightExpired(false),
    76     highlightExpired(false),
    76     backgroundItem(0),
    77     backgroundItem(0),
    87 {
    88 {
    88     if (backgroundItem) {
    89     if (backgroundItem) {
    89         delete backgroundItem;
    90         delete backgroundItem;
    90         backgroundItem = 0;
    91         backgroundItem = 0;
    91     }
    92     }
    92 
    93 }
    93     // need to delete the hash values
    94 
    94     QList<QGraphicsLayoutItem *> deletedSpacers = mSpacers.values();
    95 void HbWidgetPrivate::setBackgroundItem(HbStyle::Primitive type, int zValue)
    95     mSpacers.clear();
    96 {
    96     qDeleteAll( deletedSpacers );            
    97     Q_Q(HbWidget);
    97 }
    98     if(type!=backgroundPrimitiveType || type == HbStyle::P_None) {
    98 
    99         if (backgroundItem) {
    99 /*!
   100             delete backgroundItem;
   100     Creates a new spacer item with the given \a name.
   101             backgroundItem = 0;
   101 
   102         }
   102     Returns a pointer to the newly created spacer item. Widget keeps the ownership.
   103         backgroundPrimitiveType = type;
   103 */
   104         backgroundItem = q->style()->createPrimitive(backgroundPrimitiveType, const_cast<HbWidget*>(q));
   104 QGraphicsLayoutItem *HbWidgetPrivate::createSpacerItem( const QString &name )
   105         if(backgroundItem) {
   105 {
   106             backgroundItem->setParentItem(q);
   106 #ifndef Q_OS_SYMBIAN
   107         }
   107     Q_ASSERT( !name.isEmpty() );
   108     q->updatePrimitives();
   108     // check in desktop environments that the item does not exist already
   109 }
   109     Q_ASSERT( !mSpacers.contains( name ) );
   110     if(backgroundItem && zValue != backgroundItem->zValue()) {
   110 #endif
   111         backgroundItem->setZValue(zValue);
   111 
   112     }
   112     HbSpacerItem *newSpacer = new HbSpacerItem();
   113 } 
   113     mSpacers.insert( name, newSpacer );
   114 
   114     return newSpacer;
   115 
   115 }
   116 /*!
   116 
       
   117 /*
       
   118     
       
   119     \deprecated HbWidget::setBackgroundItem(HbStyle::Primitive, int)
       
   120     is deprecated. Use HbWidget::setBackgroundItem(QGraphicsItem *item, int zValue) instead.
       
   121     
       
   122     Creates background item to the widget from style primitive.
       
   123     
       
   124     Creates a new background item to the widget and the reparents it to
       
   125     be child of the widget. Also Z-value of the background item will be 
       
   126     changed to zValue. By default the zValue is -1, which should be 
       
   127     behind other widget content. Background item will be always resized 
       
   128     to be same size as the widgets bounding rect. 
       
   129     
       
   130     If type is HbStyle::P_None, the background item will be removed from
       
   131     widget.
       
   132     
       
   133     Previously set background item will be deleted.
       
   134  */
       
   135 void HbWidget::setBackgroundItem(HbStyle::Primitive type, int zValue)
       
   136 {
       
   137     Q_D(HbWidget);
       
   138     if(type!=d->backgroundPrimitiveType || type == HbStyle::P_None) {
       
   139         if (d->backgroundItem) {
       
   140             delete d->backgroundItem;
       
   141             d->backgroundItem = 0;
       
   142         }
       
   143         d->backgroundPrimitiveType = type;
       
   144         d->backgroundItem = style()->createPrimitive(d->backgroundPrimitiveType, const_cast<HbWidget*>(this));
       
   145         if(d->backgroundItem) {
       
   146             d->backgroundItem->setParentItem(this);
       
   147         }
       
   148         updatePrimitives();
       
   149     }
       
   150     if(d->backgroundItem && zValue != d->backgroundItem->zValue()) {
       
   151         d->backgroundItem->setZValue(zValue);
       
   152     }
       
   153 }
       
   154 
       
   155 /*
       
   156     Sets background item to the widget.
   117     Sets background item to the widget.
   157     
   118     
   158     The item will be reparented to be child of the widget. Also Z-value 
   119     The item will be reparented to be child of the widget. Also Z-value 
   159     of the background item will be changed to be zValue. By default the 
   120     of the background item will be changed to be zValue. By default the 
   160     zValue is -1, which should be  behind other widget content. 
   121     zValue is -1, which should be  behind other widget content. 
   180     if(d->backgroundItem) {
   141     if(d->backgroundItem) {
   181         d->backgroundItem->setParentItem(this);
   142         d->backgroundItem->setParentItem(this);
   182         d->backgroundItem->setZValue(zValue);
   143         d->backgroundItem->setZValue(zValue);
   183         d->updateBackgroundItemSize();
   144         d->updateBackgroundItemSize();
   184     }
   145     }
   185 }
   146     updatePrimitives();
   186 
   147 }
   187 /*
   148 
       
   149 /*!
   188     Returns background item. 0 is returned if there isn't background
   150     Returns background item. 0 is returned if there isn't background
   189     item in the widget.
   151     item in the widget.
   190  */
   152  */
   191 QGraphicsItem *HbWidget::backgroundItem() const
   153 QGraphicsItem *HbWidget::backgroundItem() const
   192 {
   154 {
   193     Q_D( const HbWidget );
   155     Q_D( const HbWidget );
   194     return d->backgroundItem;
   156     return d->backgroundItem;
   195 }
   157 }
   196 
   158 
   197 /*
   159 /*!
   198     Returns focusItem primitive items.
   160     Returns focusItem primitive items.
   199     Focus primitive is created if has not been created already.
   161     Focus primitive is created if has not been created already.
   200  */
   162  */
   201 QGraphicsItem *HbWidgetPrivate::focusPrimitive(HbWidget::FocusHighlight highlightType) const
   163 QGraphicsItem *HbWidgetPrivate::focusPrimitive(HbWidget::FocusHighlight highlightType) const
   202 {
   164 {
   219     }
   181     }
   220 
   182 
   221     return 0;
   183     return 0;
   222 }
   184 }
   223 
   185 
   224 /*
   186 /*!
   225     Hides or shows focus primitive depending on the focus state of the widget.
   187     Hides or shows focus primitive depending on the focus state of the widget.
   226 */
   188 */
   227 void HbWidgetPrivate::focusChangeEvent(HbWidget::FocusHighlight focusHighlight)
   189 void HbWidgetPrivate::focusChangeEvent(HbWidget::FocusHighlight focusHighlight)
   228 {
   190 {
   229     Q_Q(HbWidget);
   191     Q_Q(HbWidget);
   263         focusResidualItem->update();
   225         focusResidualItem->update();
   264     }
   226     }
   265 
   227 
   266 }
   228 }
   267 
   229 
   268 /*
   230 /*!
   269     Find closest parent with focus group and update the focused child.
   231     Find closest parent with focus group and update the focused child.
   270 */
   232 */
   271 void HbWidgetPrivate::updateCurrentFocusChild()
   233 void HbWidgetPrivate::updateCurrentFocusChild()
   272 {
   234 {
   273     Q_Q(HbWidget);
   235     Q_Q(HbWidget);
   277     if (group) {
   239     if (group) {
   278         group->updateCurrentFocusChild(q);
   240         group->updateCurrentFocusChild(q);
   279     }
   241     }
   280 }
   242 }
   281 
   243 
   282 /*
   244 /*!
   283     Find and return the closest parent with focus group if any. If propagate
   245     Find and return the closest parent with focus group if any. If propagate
   284     is true then the closest parent with focus group and children is accepted as
   246     is true then the closest parent with focus group and children is accepted as
   285     valid focus group e.g. used for a widget which has the key for changing the 
   247     valid focus group e.g. used for a widget which has the key for changing the 
   286     focus mode set only.
   248     focus mode set only.
   287 */
   249 */
   314         };
   276         };
   315     }
   277     }
   316     return (group) ? group : 0;
   278     return (group) ? group : 0;
   317 }
   279 }
   318 
   280 
   319 /*
   281 /*!
   320     Set focus to child widget depending on the set focus delegation
   282     Set focus to child widget depending on the set focus delegation
   321     policy. 
   283     policy. 
   322 */
   284 */
   323 bool HbWidgetPrivate::delegateFocus(QFocusEvent *event) const
   285 bool HbWidgetPrivate::delegateFocus(QFocusEvent *event) const
   324 {
   286 {
   341             }
   303             }
   342         }
   304         }
   343     }
   305     }
   344 }
   306 }
   345 
   307 
   346 /*
       
   347     Test if some item in our parent hierarchy has
       
   348     the Hb::InputMethodNeutral flag set.
       
   349 */
       
   350 bool HbWidgetPrivate::isInputMethodNeutral()
       
   351 {
       
   352     Q_Q(HbWidget);
       
   353 
       
   354     // Test the widget itself...
       
   355     if(q->testAttribute(Hb::InputMethodNeutral)) {
       
   356         return true;
       
   357     }
       
   358 
       
   359     // and then all parents
       
   360     QGraphicsWidget* i = q->parentWidget();
       
   361     while(i) {
       
   362         HbWidget* w = qobject_cast<HbWidget *>(i);
       
   363         if(w && w->testAttribute(Hb::InputMethodNeutral)) {
       
   364             return true;
       
   365         }
       
   366         i = i->parentWidget();
       
   367     }
       
   368     return false;
       
   369 }
       
   370 
   308 
   371 /*!
   309 /*!
   372     \class HbWidget
   310     \class HbWidget
   373 
   311 
   374     \brief HbWidget is a base for all Hb widgets. It contains common functionality
   312     \brief HbWidget is a base for all Hb widgets. It contains common functionality
   462 HbWidget::HbWidget( QGraphicsItem *parent, Qt::WindowFlags wFlags ):
   400 HbWidget::HbWidget( QGraphicsItem *parent, Qt::WindowFlags wFlags ):
   463     HbWidgetBase( *new HbWidgetPrivate, parent, wFlags )
   401     HbWidgetBase( *new HbWidgetPrivate, parent, wFlags )
   464 {
   402 {
   465     Q_D( HbWidget );
   403     Q_D( HbWidget );
   466     d->q_ptr = this;
   404     d->q_ptr = this;
       
   405     setAttribute(Hb::Widget, true);
   467 
   406 
   468 #ifdef HB_TESTABILITY 
   407 #ifdef HB_TESTABILITY 
   469 	if(d->testabilitySignal) {
   408 	if(d->testabilitySignal) {
   470 		d->testabilitySignal->setParent(this);
   409 		d->testabilitySignal->setParent(this);
   471 		d->testabilitySignal->notifySignalEnabled(testabilitySignalEnabledChange, this);
   410 		d->testabilitySignal->notifySignalEnabled(testabilitySignalEnabledChange, this);
   480 HbWidget::HbWidget(HbWidgetPrivate &dd, QGraphicsItem *parent, Qt::WindowFlags wFlags):
   419 HbWidget::HbWidget(HbWidgetPrivate &dd, QGraphicsItem *parent, Qt::WindowFlags wFlags):
   481     HbWidgetBase( dd, parent, wFlags )
   420     HbWidgetBase( dd, parent, wFlags )
   482 {
   421 {
   483     Q_D( HbWidget );
   422     Q_D( HbWidget );
   484     d->q_ptr = this;
   423     d->q_ptr = this;
       
   424     setAttribute(Hb::Widget, true);
   485 
   425 
   486 #ifdef HB_TESTABILITY 
   426 #ifdef HB_TESTABILITY 
   487 	if (d->testabilitySignal) {
   427 	if (d->testabilitySignal) {
   488 		d->testabilitySignal->setParent(this);
   428 		d->testabilitySignal->setParent(this);
   489 		d->testabilitySignal->notifySignalEnabled(testabilitySignalEnabledChange, this);
   429 		d->testabilitySignal->notifySignalEnabled(testabilitySignalEnabledChange, this);
   521         //Start of snippet 1
   461         //Start of snippet 1
   522         HbEvent event1( HbEvent::ChildFocusIn);
   462         HbEvent event1( HbEvent::ChildFocusIn);
   523         QCoreApplication::sendEvent( parentWidget(), &event1 );        
   463         QCoreApplication::sendEvent( parentWidget(), &event1 );        
   524         //End of snippet 1
   464         //End of snippet 1
   525     }
   465     }
   526 
       
   527 #if QT_VERSION >= 0x040600
       
   528     if(!d->isInputMethodNeutral() && !(flags() & QGraphicsItem::ItemAcceptsInputMethod)) {
       
   529         // Make sure the input panel is closed if this widget is not input method neutral or
       
   530         // it does not accept input method
       
   531         // Send close input panel event.
       
   532         QInputContext *ic = qApp->inputContext();
       
   533         if (ic) {
       
   534             QEvent *closeEvent = new QEvent(QEvent::CloseSoftwareInputPanel);
       
   535             ic->filterEvent(closeEvent);
       
   536             delete closeEvent;
       
   537         }
       
   538     }
       
   539 #endif
       
   540 
       
   541 }
   466 }
   542 
   467 
   543 /*! 
   468 /*! 
   544     FocusOut event hadler.
   469     FocusOut event hadler.
   545  */
   470  */
   617             delete acts[i];
   542             delete acts[i];
   618     }
   543     }
   619 }
   544 }
   620 
   545 
   621 /*!
   546 /*!
   622     This function returns the base id of the style plugin of the widget or
   547     \deprecated HbWidget::pluginBaseId()
   623 	\c 0 if the widget is not plugin based.
   548         is deprecated. Style plugins are deprecated.
   624 */
   549 */
   625 int HbWidget::pluginBaseId() const
   550 int HbWidget::pluginBaseId() const
   626 {
   551 {
   627     Q_D(const HbWidget);
   552     return 0; // deprecated
   628     return d->pluginBaseId;
   553 }
   629 }
   554 
   630 
   555 /*!
   631 /*!
   556     \deprecated HbWidget::setPluginBaseId(int)
   632     Sets the base id of the style plugin of the widget.
   557         is deprecated. Style plugins are deprecated.
   633 */
   558 */
   634 void HbWidget::setPluginBaseId( int baseId )
   559 void HbWidget::setPluginBaseId( int baseId )
   635 {
   560 {
   636     Q_D(HbWidget);
   561     Q_UNUSED(baseId); // deprecated
   637     d->pluginBaseId = baseId;
       
   638 }
   562 }
   639 
   563 
   640 /*!
   564 /*!
   641   This function returns the HbMainWindow of the widget or \c 0 if
   565   This function returns the HbMainWindow of the widget or \c 0 if
   642   it doesn't exist.  Note that if the widget is not added to the scene
   566   it doesn't exist.  Note that if the widget is not added to the scene
   695         if(d->backgroundPrimitiveType != HbStyle::P_None) {
   619         if(d->backgroundPrimitiveType != HbStyle::P_None) {
   696             style()->updatePrimitive(backgroundItem(), d->backgroundPrimitiveType, &option);
   620             style()->updatePrimitive(backgroundItem(), d->backgroundPrimitiveType, &option);
   697         }
   621         }
   698         d->updateBackgroundItemSize();
   622         d->updateBackgroundItemSize();
   699     }
   623     }
   700 
       
   701     if (d->focusPrimitive(HbWidget::FocusHighlightResidual)) {
   624     if (d->focusPrimitive(HbWidget::FocusHighlightResidual)) {
   702         style()->updatePrimitive(d->focusPrimitive(HbWidget::FocusHighlightResidual),
   625         style()->updatePrimitive(d->focusPrimitive(HbWidget::FocusHighlightResidual),
   703             d->focusResidualType, &option);        
   626             d->focusResidualType, &option);        
   704     }
   627     }
   705 
   628 
   858         }
   781         }
   859         if (d->polishPending && value.toBool()) {
   782         if (d->polishPending && value.toBool()) {
   860             d->polishPending = false;
   783             d->polishPending = false;
   861             HbStyleParameters params;
   784             HbStyleParameters params;
   862             polish( params );
   785             polish( params );
       
   786             if (scene()) {
       
   787              // The widget is polished again or is beign polished now. As we set layout to widget in polishEvent,
       
   788              // inform scene to polish any new child items and handle any resulting layoutrequest
       
   789              // events before drawing.
       
   790                 HbGraphicsScene *hbscene = qobject_cast<HbGraphicsScene*>(scene());
       
   791                 if (hbscene) {
       
   792                     HbGraphicsScenePrivate::d_ptr(hbscene)->mPolishWidgets = true;
       
   793                 }
       
   794             }
   863         }
   795         }
   864     }
   796     }
   865     else if (change == QGraphicsItem::ItemChildAddedChange) {
   797     else if (change == QGraphicsItem::ItemChildAddedChange) {
   866         if (d->focusGroup) {
   798         if (d->focusGroup) {
   867             QGraphicsItem *child = qVariantValue<QGraphicsItem *>(value);
   799             QGraphicsItem *child = qVariantValue<QGraphicsItem *>(value);
   930     Override this method if you want to do something special in polish
   862     Override this method if you want to do something special in polish
   931     (e.g. fetch some custom style parameters to your widget from style
   863     (e.g. fetch some custom style parameters to your widget from style
   932     backend). When overriding, always call the base classes 
   864     backend). When overriding, always call the base classes 
   933     impelentation.
   865     impelentation.
   934 
   866 
   935     If you are implementing a custom widget that has a style plugin
       
   936     you must make sure that pluginBaseId is set.
       
   937 
       
   938     \param params, For querying (custom) style parameters from HbStyle.
   867     \param params, For querying (custom) style parameters from HbStyle.
   939 
   868 
   940     \sa polish(), pluginBaseId()
   869     \sa polish()
   941 */
   870 */
   942 void HbWidget::polish( HbStyleParameters& params )
   871 void HbWidget::polish( HbStyleParameters& params )
   943 {
   872 {
   944     Q_D(HbWidget);
   873     Q_D(HbWidget);
   945     if (isVisible()) {
   874     if (isVisible()) {
   973 	if (d->polished && !d->repolishOutstanding) {
   902 	if (d->polished && !d->repolishOutstanding) {
   974 #endif
   903 #endif
   975         d->repolishOutstanding = true;
   904         d->repolishOutstanding = true;
   976         QEvent* polishEvent = new QEvent( QEvent::Polish );
   905         QEvent* polishEvent = new QEvent( QEvent::Polish );
   977         QCoreApplication::postEvent(this, polishEvent);
   906         QCoreApplication::postEvent(this, polishEvent);
       
   907         // If no one is handling repolish synchronously, lets make sure they are handled
       
   908         // before painting. For example view items handle them synchronously.
       
   909         if (scene() && !d->mHandlingRepolishSynchronously) {
       
   910             // The widget needs to be polished again. As we set layout to widget in polishEvent,
       
   911             // inform scene to handle polish and any resulting layoutrequest
       
   912             // events before drawing.
       
   913             HbGraphicsScene *hbscene = qobject_cast<HbGraphicsScene*>(scene());
       
   914             if (hbscene) {
       
   915                 HbGraphicsScenePrivate::d_ptr(hbscene)->mRepolishWidgets = true;
       
   916             }
       
   917         }
   978     }
   918     }
   979 }
   919 }
   980 
   920 
   981 /*!
   921 /*!
   982 
   922 
  1032 }
   972 }
  1033 
   973 
  1034 /*!
   974 /*!
  1035     Returns primitive which HbStyle::itemName equals to \a itemName.
   975     Returns primitive which HbStyle::itemName equals to \a itemName.
  1036     
   976     
  1037     If the \a itemName is empty, the layout is returned. 
   977     If the \a itemName is empty, the layout is returned, otherwise
       
   978     returns the primitive that matches the \a itemName.
  1038     
   979     
  1039     If the \a itemName does not match any primitive, \a itemName is mapped to owned spacers.
       
  1040 
       
  1041     If the \a itemName cannot be mapped to any of the above, returns 0.
   980     If the \a itemName cannot be mapped to any of the above, returns 0.
  1042 
   981 
  1043     \sa HbStyle::itemName()
   982     \sa HbStyle::itemName()
  1044 */
   983 */
  1045 QGraphicsLayoutItem *HbWidget::layoutPrimitive(const QString &itemName) const
   984 QGraphicsLayoutItem *HbWidget::layoutPrimitive(const QString &itemName) const
  1046 {
   985 {
  1047     Q_D(const HbWidget);
       
  1048     if ( itemName == "" ) {
   986     if ( itemName == "" ) {
  1049         return layout();
   987         return layout();
  1050     } else {
   988     } else {
  1051         QList<QGraphicsItem*> list = childItems();
   989         QList<QGraphicsItem*> list = childItems();
  1052         for ( int i = 0 ; i < list.count() ; i++  ) {
   990         for ( int i = 0 ; i < list.count() ; i++  ) {
  1054             if ( HbStyle::itemName(item) == itemName ) {
   992             if ( HbStyle::itemName(item) == itemName ) {
  1055                 return item->isWidget() ? static_cast<QGraphicsWidget*>(item) : 0;
   993                 return item->isWidget() ? static_cast<QGraphicsWidget*>(item) : 0;
  1056             }
   994             }
  1057         }
   995         }
  1058     }
   996     }
  1059 
       
  1060     if ( d->mSpacers.contains(itemName) ) {
       
  1061         return d->mSpacers.value(itemName); 
       
  1062     }
       
  1063  
       
  1064     return 0;
   997     return 0;
  1065 }
   998 }
  1066 
   999 
  1067 /*!
  1000 /*!
  1068     Defines the orientation to move focus from focused widget to previous or next
  1001     Defines the orientation to move focus from focused widget to previous or next
  1365     } else if (highlightType == HbWidget::FocusHighlightResidual) {
  1298     } else if (highlightType == HbWidget::FocusHighlightResidual) {
  1366         primitive = d->focusResidualType;
  1299         primitive = d->focusResidualType;
  1367     }
  1300     }
  1368     return primitive;
  1301     return primitive;
  1369 }
  1302 }
       
  1303 
       
  1304 bool HbWidget::sceneEventFilter (QGraphicsItem *watched, QEvent *event)
       
  1305 {
       
  1306     if(event->type() == QEvent::Gesture && watched->type() == Hb::ItemType_TouchArea) {
       
  1307         QGestureEvent* ge = static_cast<QGestureEvent*>(event);
       
  1308         HbTapGesture* tap = qobject_cast<HbTapGesture*>(ge->gesture(Qt::TapGesture));
       
  1309 
       
  1310         if (tap && tap->state() == Qt::GestureStarted) {
       
  1311             tap->setProperty(HbPrivate::ThresholdRect.latin1(), watched->mapRectToScene(watched->boundingRect()).toRect());
       
  1312         }
       
  1313         sceneEvent(event);
       
  1314         return true;
       
  1315     }
       
  1316     return HbWidgetBase::sceneEventFilter(watched, event);
       
  1317 }