1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/cshelp.cpp 
   3 // Purpose:     Context sensitive help class implementation 
   4 // Author:      Julian Smart, Vadim Zeitlin 
   8 // Copyright:   (c) 2000 Julian Smart, Vadim Zeitlin 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  17     #pragma implementation "cshelp.h" 
  20 // ---------------------------------------------------------------------------- 
  22 // ---------------------------------------------------------------------------- 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  36 #include "wx/tipwin.h" 
  38 #include "wx/module.h" 
  39 #include "wx/cshelp.h" 
  41 // ---------------------------------------------------------------------------- 
  42 // wxContextHelpEvtHandler private class 
  43 // ---------------------------------------------------------------------------- 
  45 // This class exists in order to eat events until the left mouse button is 
  47 class wxContextHelpEvtHandler
: public wxEvtHandler
 
  50     wxContextHelpEvtHandler(wxContextHelp
* contextHelp
) 
  52         m_contextHelp 
= contextHelp
; 
  55     virtual bool ProcessEvent(wxEvent
& event
); 
  58     wxContextHelp
* m_contextHelp
; 
  60     DECLARE_NO_COPY_CLASS(wxContextHelpEvtHandler
) 
  63 // ============================================================================ 
  65 // ============================================================================ 
  67 // ---------------------------------------------------------------------------- 
  69 // ---------------------------------------------------------------------------- 
  72  * Invokes context-sensitive help 
  76 IMPLEMENT_DYNAMIC_CLASS(wxContextHelp
, wxObject
) 
  78 wxContextHelp::wxContextHelp(wxWindow
* win
, bool beginHelp
) 
  83         BeginContextHelp(win
); 
  86 wxContextHelp::~wxContextHelp() 
  92 // Not currently needed, but on some systems capture may not work as 
  93 // expected so we'll leave it here for now. 
  95 static void wxPushOrPopEventHandlers(wxContextHelp
* help
, wxWindow
* win
, bool push
) 
  98         win
->PushEventHandler(new wxContextHelpEvtHandler(help
)); 
 100         win
->PopEventHandler(true); 
 102     wxWindowList::compatibility_iterator node 
= win
->GetChildren().GetFirst(); 
 105         wxWindow
* child 
= node
->GetData(); 
 106         wxPushOrPopEventHandlers(help
, child
, push
); 
 108         node 
= node
->GetNext(); 
 113 // Begin 'context help mode' 
 114 bool wxContextHelp::BeginContextHelp(wxWindow
* win
) 
 117         win 
= wxTheApp
->GetTopWindow(); 
 121     wxCursor 
cursor(wxCURSOR_QUESTION_ARROW
); 
 122     wxCursor oldCursor 
= win
->GetCursor(); 
 123     win
->SetCursor(cursor
); 
 126     //    wxSetCursor(cursor); 
 132     wxPushOrPopEventHandlers(this, win
, true); 
 134     win
->PushEventHandler(new wxContextHelpEvtHandler(this)); 
 144     wxPushOrPopEventHandlers(this, win
, false); 
 146     win
->PopEventHandler(true); 
 149     win
->SetCursor(oldCursor
); 
 154         wxWindow
* winAtPtr 
= wxFindWindowAtPointer(pt
); 
 159             printf("Picked %s (%d)\n", winAtPtr
->GetName().c_str(), 
 165             DispatchEvent(winAtPtr
, pt
); 
 171 bool wxContextHelp::EndContextHelp() 
 178 bool wxContextHelp::EventLoop() 
 184         if (wxTheApp
->Pending()) 
 186             wxTheApp
->Dispatch(); 
 190             wxTheApp
->ProcessIdle(); 
 197 bool wxContextHelpEvtHandler::ProcessEvent(wxEvent
& event
) 
 199     if (event
.GetEventType() == wxEVT_LEFT_DOWN
) 
 201         m_contextHelp
->SetStatus(true); 
 202         m_contextHelp
->EndContextHelp(); 
 206     if ((event
.GetEventType() == wxEVT_CHAR
) || 
 207         (event
.GetEventType() == wxEVT_KEY_DOWN
) || 
 208         (event
.GetEventType() == wxEVT_ACTIVATE
) || 
 209         (event
.GetEventType() == wxEVT_MOUSE_CAPTURE_CHANGED
)) 
 211         // May have already been set to true by a left-click 
 212         //m_contextHelp->SetStatus(false); 
 213         m_contextHelp
->EndContextHelp(); 
 217     if ((event
.GetEventType() == wxEVT_PAINT
) || 
 218         (event
.GetEventType() == wxEVT_ERASE_BACKGROUND
)) 
 227 // Dispatch the help event to the relevant window 
 228 bool wxContextHelp::DispatchEvent(wxWindow
* win
, const wxPoint
& pt
) 
 230     wxWindow
* subjectOfHelp 
= win
; 
 231     bool eventProcessed 
= false; 
 232     while (subjectOfHelp 
&& !eventProcessed
) 
 234         wxHelpEvent 
helpEvent(wxEVT_HELP
, subjectOfHelp
->GetId(), pt
) ; 
 235         helpEvent
.SetEventObject(subjectOfHelp
); 
 237         eventProcessed 
= win
->GetEventHandler()->ProcessEvent(helpEvent
); 
 239         // Go up the window hierarchy until the event is handled (or not). 
 240         // I.e. keep submitting ancestor windows until one is recognised 
 241         // by the app code that processes the ids and displays help. 
 242         subjectOfHelp 
= subjectOfHelp
->GetParent(); 
 244     return eventProcessed
; 
 247 // ---------------------------------------------------------------------------- 
 248 // wxContextHelpButton 
 249 // ---------------------------------------------------------------------------- 
 252  * wxContextHelpButton 
 253  * You can add this to your dialogs (especially on non-Windows platforms) 
 254  * to put the application into context help mode. 
 257 static const char * csquery_xpm
[] = { 
 273 IMPLEMENT_CLASS(wxContextHelpButton
, wxBitmapButton
) 
 275 BEGIN_EVENT_TABLE(wxContextHelpButton
, wxBitmapButton
) 
 276     EVT_BUTTON(wxID_CONTEXT_HELP
, wxContextHelpButton::OnContextHelp
) 
 279 wxContextHelpButton::wxContextHelpButton(wxWindow
* parent
, 
 284 #if defined(__WXPM__) 
 285                    : wxBitmapButton(parent
, id
, wxBitmap(wxCSQUERY_BITMAP
 
 286                                                          ,wxBITMAP_TYPE_RESOURCE
 
 290                    : wxBitmapButton(parent
, id
, wxBitmap(csquery_xpm
), 
 296 void wxContextHelpButton::OnContextHelp(wxCommandEvent
& WXUNUSED(event
)) 
 298     wxContextHelp 
contextHelp(GetParent()); 
 301 // ---------------------------------------------------------------------------- 
 303 // ---------------------------------------------------------------------------- 
 305 wxHelpProvider 
*wxHelpProvider::ms_helpProvider 
= (wxHelpProvider 
*)NULL
; 
 307 // trivial implementation of some methods which we don't want to make pure 
 308 // virtual for convenience 
 310 void wxHelpProvider::AddHelp(wxWindowBase 
* WXUNUSED(window
), 
 311                              const wxString
& WXUNUSED(text
)) 
 315 void wxHelpProvider::AddHelp(wxWindowID 
WXUNUSED(id
), 
 316                              const wxString
& WXUNUSED(text
)) 
 320 // removes the association 
 321 void wxHelpProvider::RemoveHelp(wxWindowBase
* WXUNUSED(window
)) 
 325 wxHelpProvider::~wxHelpProvider() 
 329 // ---------------------------------------------------------------------------- 
 330 // wxSimpleHelpProvider 
 331 // ---------------------------------------------------------------------------- 
 333 wxString 
wxSimpleHelpProvider::GetHelp(const wxWindowBase 
*window
) 
 335     wxLongToStringHashMap::iterator it 
= m_hashWindows
.find((long)window
); 
 337     if ( it 
== m_hashWindows
.end() ) 
 339         it 
= m_hashIds
.find(window
->GetId()); 
 340         if ( it 
== m_hashIds
.end() ) 
 341             return wxEmptyString
; 
 347 void wxSimpleHelpProvider::AddHelp(wxWindowBase 
*window
, const wxString
& text
) 
 349     m_hashWindows
.erase((long)window
); 
 350     m_hashWindows
[(long)window
] = text
; 
 353 void wxSimpleHelpProvider::AddHelp(wxWindowID id
, const wxString
& text
) 
 355     wxLongToStringHashMap::key_type key 
= (wxLongToStringHashMap::key_type
)id
; 
 356     m_hashIds
.erase(key
); 
 357     m_hashIds
[key
] = text
; 
 360 // removes the association 
 361 void wxSimpleHelpProvider::RemoveHelp(wxWindowBase
* window
) 
 363     m_hashWindows
.erase((long)window
); 
 366 bool wxSimpleHelpProvider::ShowHelp(wxWindowBase 
*window
) 
 369     static wxTipWindow
* s_tipWindow 
= NULL
; 
 373         // Prevent s_tipWindow being nulled in OnIdle, 
 374         // thereby removing the chance for the window to be closed by ShowHelp 
 375         s_tipWindow
->SetTipWindowPtr(NULL
); 
 376         s_tipWindow
->Close(); 
 380     wxString text 
= GetHelp(window
); 
 383         s_tipWindow 
= new wxTipWindow((wxWindow 
*)window
, text
, 100, & s_tipWindow
); 
 389 #endif // wxUSE_TIPWINDOW 
 394 // ---------------------------------------------------------------------------- 
 395 // wxHelpControllerHelpProvider 
 396 // ---------------------------------------------------------------------------- 
 398 wxHelpControllerHelpProvider::wxHelpControllerHelpProvider(wxHelpControllerBase
* hc
) 
 400     m_helpController 
= hc
; 
 403 bool wxHelpControllerHelpProvider::ShowHelp(wxWindowBase 
*window
) 
 405     wxString text 
= GetHelp(window
); 
 408         if (m_helpController
) 
 411                 return m_helpController
->DisplayContextPopup(wxAtoi(text
)); 
 413             // If the help controller is capable of popping up the text... 
 414             else if (m_helpController
->DisplayTextPopup(text
, wxGetMousePosition())) 
 419             // ...else use the default method. 
 420                 return wxSimpleHelpProvider::ShowHelp(window
); 
 423             return wxSimpleHelpProvider::ShowHelp(window
); 
 430 // Convenience function for turning context id into wxString 
 431 wxString 
wxContextId(int id
) 
 433     return wxString::Format(_T("%d"), id
); 
 436 // ---------------------------------------------------------------------------- 
 437 // wxHelpProviderModule: module responsible for cleaning up help provider. 
 438 // ---------------------------------------------------------------------------- 
 440 class wxHelpProviderModule 
: public wxModule
 
 447     DECLARE_DYNAMIC_CLASS(wxHelpProviderModule
) 
 450 IMPLEMENT_DYNAMIC_CLASS(wxHelpProviderModule
, wxModule
) 
 452 bool wxHelpProviderModule::OnInit() 
 454     // Probably we don't want to do anything by default, 
 455     // since it could pull in extra code 
 456     // wxHelpProvider::Set(new wxSimpleHelpProvider); 
 461 void wxHelpProviderModule::OnExit() 
 463     if (wxHelpProvider::Get()) 
 465         delete wxHelpProvider::Get(); 
 466         wxHelpProvider::Set(NULL
);