1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     an external help controller for wxWidgets 
   4 // Author:      Karsten Ballueder 
   8 // Copyright:   (c) Karsten Ballueder 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  13 #   pragma implementation "wxexthlp.h" 
  16 #include "wx/wxprec.h" 
  22 #if wxUSE_HELP && !defined(__WXWINCE__) && (!defined(__WXMAC__) || defined(__WXMAC_OSX__)) 
  26     #include "wx/string.h" 
  30     #include "wx/msgdlg.h" 
  31     #include "wx/choicdlg.h" 
  35 #include "wx/helpbase.h" 
  36 #include "wx/generic/helpext.h" 
  42 #if !defined(__WINDOWS__) && !defined(__OS2__) 
  48 #include "wx/msw/winundef.h" 
  51 // ---------------------------------------------------------------------------- 
  53 // ---------------------------------------------------------------------------- 
  55 /// Name for map file. 
  56 #define WXEXTHELP_MAPFILE   _T("wxhelp.map") 
  57 /// Maximum line length in map file. 
  58 #define WXEXTHELP_BUFLEN 512 
  59 /// Character introducing comments/documentation field in map file. 
  60 #define WXEXTHELP_COMMENTCHAR   ';' 
  64 IMPLEMENT_CLASS(wxExtHelpController
, wxHelpControllerBase
) 
  66 /// Name of environment variable to set help browser. 
  67 #define   WXEXTHELP_ENVVAR_BROWSER   wxT("WX_HELPBROWSER") 
  68 /// Is browser a netscape browser? 
  69 #define   WXEXTHELP_ENVVAR_BROWSERISNETSCAPE wxT("WX_HELPBROWSER_NS") 
  72    This class implements help via an external browser. 
  73    It requires the name of a directory containing the documentation 
  74    and a file mapping numerical Section numbers to relative URLS. 
  77 wxExtHelpController::wxExtHelpController() 
  79    m_MapList 
= (wxList
*) NULL
; 
  81    m_BrowserName 
= WXEXTHELP_DEFAULTBROWSER
; 
  82    m_BrowserIsNetscape 
= WXEXTHELP_DEFAULTBROWSER_IS_NETSCAPE
; 
  84    wxChar 
*browser 
= wxGetenv(WXEXTHELP_ENVVAR_BROWSER
); 
  87       m_BrowserName 
= browser
; 
  88       browser 
= wxGetenv(WXEXTHELP_ENVVAR_BROWSERISNETSCAPE
); 
  89       m_BrowserIsNetscape 
= browser 
&& (wxAtoi(browser
) != 0); 
  93 wxExtHelpController::~wxExtHelpController() 
  98 void wxExtHelpController::SetBrowser(const wxString
& browsername
, bool isNetscape
) 
 100    m_BrowserName 
= browsername
; 
 101    m_BrowserIsNetscape 
= isNetscape
; 
 104 // Set viewer: new, generic name for SetBrowser 
 105 void wxExtHelpController::SetViewer(const wxString
& viewer
, long flags
) 
 107     SetBrowser(viewer
, ((flags 
& wxHELP_NETSCAPE
) == wxHELP_NETSCAPE
)); 
 111 wxExtHelpController::DisplayHelp(const wxString 
&relativeURL
) 
 113    wxBusyCursor b
; // display a busy cursor 
 116 #if defined(__WXMSW__) 
 118    url 
<< m_MapFile 
<< '\\' << relativeURL
.BeforeFirst('#'); 
 119    bool bOk 
= (int)ShellExecute(NULL
, wxT("open"), url
.c_str(), 
 120                                 NULL
, NULL
, SW_SHOWNORMAL 
) > 32; 
 123       wxLogSysError(_("Cannot open URL '%s'"), relativeURL
.c_str()); 
 128 #elif  defined(__WXPM__) 
 131    url 
<< m_MapFile 
<< '\\' << relativeURL
.BeforeFirst('#'); 
 132 //   will have to fix for OS/2, later.....DW 
 133 //   bool bOk = (int)ShellExecute(NULL, "open", url, 
 134 //                                NULL, NULL, SW_SHOWNORMAL ) > 32; 
 137 //      wxLogSysError(_("Cannot open URL '%s'"), relativeURL.c_str()); 
 143 #elif defined(__DOS__) 
 146    command 
= m_BrowserName
; 
 147    command 
<< wxT(" file://") 
 148            << m_MapFile 
<< WXEXTHELP_SEPARATOR 
<< relativeURL
; 
 149    return wxExecute(command
) != 0; 
 155    if(m_BrowserIsNetscape
) // try re-loading first 
 158       wxGetHomeDir(&lockfile
); 
 160       lockfile 
<< WXEXTHELP_SEPARATOR 
<< wxT(".netscape]lock."); 
 162       if(stat(lockfile
.fn_str(), &statbuf
) == 0) 
 164       lockfile 
<< WXEXTHELP_SEPARATOR 
<< wxT(".netscape/lock"); 
 166       if(lstat(lockfile
.fn_str(), &statbuf
) == 0) 
 167       // cannot use wxFileExists, because it's a link pointing to a 
 168       // non-existing location      if(wxFileExists(lockfile)) 
 172          command 
<< m_BrowserName 
<< wxT(" -remote openURL(") 
 173                  << wxT("file://") << m_MapFile
 
 174                  << WXEXTHELP_SEPARATOR 
<< relativeURL 
<< wxT(")"); 
 175          success 
= wxExecute(command
); 
 176          if(success 
!= 0 ) // returns PID on success 
 181    command 
= m_BrowserName
; 
 182    command 
<< wxT(" file://") 
 183            << m_MapFile 
<< WXEXTHELP_SEPARATOR 
<< relativeURL
; 
 184    return wxExecute(command
) != 0; 
 188 class wxExtHelpMapEntry 
: public wxObject
 
 194    wxExtHelpMapEntry(int iid
, wxString 
const &iurl
, wxString 
const &idoc
) 
 195       { id 
= iid
; url 
= iurl
; doc 
= idoc
; } 
 198 void wxExtHelpController::DeleteList() 
 202       wxList::compatibility_iterator node 
= m_MapList
->GetFirst(); 
 205          delete (wxExtHelpMapEntry 
*)node
->GetData(); 
 206          m_MapList
->Erase(node
); 
 207          node 
= m_MapList
->GetFirst(); 
 210       m_MapList 
= (wxList
*) NULL
; 
 214 /** This must be called to tell the controller where to find the 
 216     @param file - NOT a filename, but a directory name. 
 217     @return true on success 
 220 wxExtHelpController::Initialize(const wxString
& file
) 
 222    return LoadFile(file
); 
 226 // ifile is the name of the base help directory 
 227 bool wxExtHelpController::LoadFile(const wxString
& ifile
) 
 229    wxString mapFile
, file
, url
, doc
; 
 231    char buffer
[WXEXTHELP_BUFLEN
]; 
 233    wxBusyCursor b
; // display a busy cursor 
 238       if(! wxIsAbsolutePath(file
)) 
 240          wxChar
* f 
= wxGetWorkingDirectory(); 
 242          delete[] f
; // wxGetWorkingDirectory returns new memory 
 246          file 
<< WXEXTHELP_SEPARATOR 
<< ifile
; 
 253       // If a locale is set, look in file/localename, i.e. 
 254       // If passed "/usr/local/myapp/help" and the current wxLocale is 
 255       // set to be "de", then look in "/usr/local/myapp/help/de/" 
 256       // first and fall back to "/usr/local/myapp/help" if that 
 258       if(wxGetLocale() && !wxGetLocale()->GetName().empty()) 
 261          newfile 
<< WXEXTHELP_SEPARATOR 
<< wxGetLocale()->GetName(); 
 262          if(wxDirExists(newfile
)) 
 266             newfile 
= WXEXTHELP_SEPARATOR
; 
 267             const wxChar 
*cptr 
= wxGetLocale()->GetName().c_str(); 
 268             while(*cptr 
&& *cptr 
!= wxT('_')) 
 269                newfile 
<< *(cptr
++); 
 270             if(wxDirExists(newfile
)) 
 276       if(! wxDirExists(file
)) 
 279       mapFile 
<< file 
<< WXEXTHELP_SEPARATOR 
<< WXEXTHELP_MAPFILE
; 
 281    else // try to reload old file 
 284    if(! wxFileExists(mapFile
)) 
 288    m_MapList 
= new wxList
; 
 291    FILE *input 
= wxFopen(mapFile
,wxT("rt")); 
 296       if(fgets(buffer
,WXEXTHELP_BUFLEN
,input
) && *buffer 
!= WXEXTHELP_COMMENTCHAR
) 
 298          len 
= strlen(buffer
); 
 299          if(buffer
[len
-1] == '\n') 
 300             buffer
[len
-1] = '\0'; // cut of trailing newline 
 301          if(sscanf(buffer
,"%d", &id
) != 1) 
 303          for(i
=0; isdigit(buffer
[i
])||isspace(buffer
[i
])||buffer
[i
]=='-'; i
++) 
 304             ; // find begin of URL 
 306          while(buffer
[i
] && ! isspace(buffer
[i
]) && buffer
[i
] != 
 307                WXEXTHELP_COMMENTCHAR
) 
 308             url 
<< (wxChar
) buffer
[i
++]; 
 309          while(buffer
[i
] && buffer
[i
] != WXEXTHELP_COMMENTCHAR
) 
 313             doc 
= wxString::FromAscii( (buffer 
+ i 
+ 1) ); // skip the comment character 
 314          m_MapList
->Append(new wxExtHelpMapEntry(id
,url
,doc
)); 
 317    }while(! feof(input
)); 
 320    m_MapFile 
= file
; // now it's valid 
 326 wxExtHelpController::DisplayContents() 
 332    wxList::compatibility_iterator node 
= m_MapList
->GetFirst(); 
 333    wxExtHelpMapEntry 
*entry
; 
 336       entry 
= (wxExtHelpMapEntry 
*)node
->GetData(); 
 337       if(entry
->id 
== CONTENTS_ID
) 
 339          contents 
= entry
->url
; 
 342       node 
= node
->GetNext(); 
 347    file 
<< m_MapFile 
<< WXEXTHELP_SEPARATOR 
<< contents
; 
 348    if(file
.Contains(wxT('#'))) 
 349       file 
= file
.BeforeLast(wxT('#')); 
 350    if(contents
.Length() && wxFileExists(file
)) 
 351       rc 
= DisplaySection(CONTENTS_ID
); 
 353    // if not found, open homemade toc: 
 354    return rc 
? true : KeywordSearch(wxEmptyString
); 
 358 wxExtHelpController::DisplaySection(int sectionNo
) 
 363    wxBusyCursor b
; // display a busy cursor 
 364    wxList::compatibility_iterator node 
= m_MapList
->GetFirst(); 
 365    wxExtHelpMapEntry 
*entry
; 
 368       entry 
= (wxExtHelpMapEntry 
*)node
->GetData(); 
 369       if(entry
->id 
== sectionNo
) 
 370          return DisplayHelp(entry
->url
); 
 371       node 
= node
->GetNext(); 
 376 bool wxExtHelpController::DisplaySection(const wxString
& section
) 
 378     bool isFilename 
= (section
.Find(wxT(".htm")) != -1); 
 381         return DisplayHelp(section
); 
 383         return KeywordSearch(section
); 
 387 wxExtHelpController::DisplayBlock(long blockNo
) 
 389    return DisplaySection((int)blockNo
); 
 393 wxExtHelpController::KeywordSearch(const wxString
& k
, 
 394                                    wxHelpSearchMode 
WXUNUSED(mode
)) 
 399    wxString     
*choices 
= new wxString
[m_NumOfEntries
]; 
 400    wxString     
*urls 
= new wxString
[m_NumOfEntries
]; 
 401    wxString compA
, compB
; 
 405    bool         showAll 
= k
.empty(); 
 406    wxList::compatibility_iterator node 
= m_MapList
->GetFirst(); 
 407    wxExtHelpMapEntry 
*entry
; 
 410       wxBusyCursor b
; // display a busy cursor 
 411       compA 
= k
; compA
.LowerCase(); // we compare case insensitive 
 414          entry 
= (wxExtHelpMapEntry 
*)node
->GetData(); 
 415          compB 
= entry
->doc
; compB
.LowerCase(); 
 416          if((showAll 
|| compB
.Contains(k
)) && ! compB
.empty()) 
 418             urls
[idx
] = entry
->url
; 
 420             // choices[idx] = (**i).doc.Contains((**i).doc.Before(WXEXTHELP_COMMENTCHAR)); 
 421             //if(choices[idx].empty()) // didn't contain the ';' 
 422             //   choices[idx] = (**i).doc; 
 423             choices
[idx
] = wxEmptyString
; 
 424             for(j
=0;entry
->doc
.c_str()[j
] 
 425                    && entry
->doc
.c_str()[j
] != WXEXTHELP_COMMENTCHAR
; j
++) 
 426                choices
[idx
] << entry
->doc
.c_str()[j
]; 
 429          node 
= node
->GetNext(); 
 434       rc 
= DisplayHelp(urls
[0]); 
 437       wxMessageBox(_("No entries found.")); 
 442       idx 
= wxGetSingleChoiceIndex(showAll 
? _("Help Index") : _("Relevant entries:"), 
 443                                    showAll 
? _("Help Index") : _("Entries found"), 
 446          rc 
= DisplayHelp(urls
[idx
]); 
 457 bool wxExtHelpController::Quit() 
 462 void wxExtHelpController::OnQuit()