/* AbiWord * Copyright (C) 1998-2000 AbiSource, Inc. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA * 02111-1307, USA. */ /***************************************************************** ** Only one of these is created by the application. *****************************************************************/ #ifdef ABI_OPT_JS #include #endif /* ABI_OPT_JS */ #include #include #include #include #include "ut_debugmsg.h" #include "ut_bytebuf.h" #include "ut_string.h" #include "xap_Args.h" #include "ap_BeOSFrame.h" #include "ap_BeOSApp.h" #include "fv_View.h" #include "fp_Run.h" // HACK: set this symbol to 0 to turn off command-line conversion // TODO: if it works, remove it entirely #ifndef CONVERT #define CONVERT 1 #endif #if CONVERT #include "ap_Convert.h" #endif #include "sp_spell.h" #include "ap_Strings.h" #include "xap_EditMethods.h" #include "ap_LoadBindings.h" #include "xap_Menu_ActionSet.h" #include "xap_Toolbar_ActionSet.h" #include "ie_imp.h" #include "ie_types.h" #include "ie_exp_Text.h" #include "ie_exp_RTF.h" #include "ie_exp_AbiWord_1.h" #include "ie_exp_HTML.h" #include "ie_imp_Text.h" #include "ie_imp_RTF.h" #include "ap_Clipboard.h" // Signal handling #include void signalWrapper(int sig_num); /*****************************************************************/ /* Splash Window Related Stuff */ #include "ut_Rehydrate.h" #include #include #define SPLASH_UP_TIME 5 // seconds extern unsigned char g_pngSplash[]; // see ap_wp_Splash.cpp extern unsigned long g_pngSplash_sizeof; // see ap_wp_Splash.cpp class SplashWin:public BWindow { public: SplashWin(BMessage *data); virtual void DispatchMessage(BMessage *msg, BHandler *handler); private: int ignore; }; SplashWin::SplashWin(BMessage *data) :BWindow(data) { BView *view = FindView("splashView"); if (view) { BMemoryIO memio(g_pngSplash, g_pngSplash_sizeof); BBitmap *bitmap = BTranslationUtils::GetBitmap(&memio); if (bitmap) view->SetViewBitmap(bitmap, B_FOLLOW_ALL, 0); else UT_DEBUGMSG(("Could not interpret splash image...\n")); view->Sync(); } Show(); ignore = SPLASH_UP_TIME; // keep on screen n seconds SetPulseRate(1000000); // a 1 second pulse } void SplashWin::DispatchMessage(BMessage *msg, BHandler *handler) { switch (msg->what) { case B_PULSE: if (ignore-- > 0) break; case B_KEY_DOWN: case B_MOUSE_DOWN: BWindow::DispatchMessage(msg, handler); this->Close(); break; default: BWindow::DispatchMessage(msg, handler); } } void _showSplash(XAP_Args * pArgs, const char * /*szAppName*/) { // Unix does put the program name in argv[0], // unlike Win32, so [1] is the first argument int nFirstArg = 1; int k; UT_Bool bShowSplash = UT_TRUE; // scan args for splash-related stuff #if CONVERT for (k=nFirstArg; (km_argc); k++) { if (*pArgs->m_argv[k] == '-') { if (UT_stricmp(pArgs->m_argv[k],"-to") == 0) { bShowSplash = UT_FALSE; } } } for (k=nFirstArg; (km_argc); k++) { if (*pArgs->m_argv[k] == '-') { if (UT_stricmp(pArgs->m_argv[k],"-show") == 0) { bShowSplash = UT_TRUE; } } } #endif for (k=nFirstArg; (km_argc); k++) { if (*pArgs->m_argv[k] == '-') { if (UT_stricmp(pArgs->m_argv[k],"-nosplash") == 0) bShowSplash = UT_FALSE; } } if (bShowSplash == UT_FALSE) return; BMessage *msg = new BMessage(); if (RehydrateWindow("SplashWindow", msg)) { //Automatically shows and hides itself SplashWin *nwin = new SplashWin(msg); } } /*****************************************************************/ AP_BeOSApp::AP_BeOSApp(XAP_Args * pArgs, const char * szAppName) : XAP_BeOSApp(pArgs,szAppName) { m_pStringSet = NULL; m_pClipboard = NULL; } AP_BeOSApp::~AP_BeOSApp(void) { // SpellCheckCleanup(); DELETEP(m_pStringSet); DELETEP(m_pClipboard); } static UT_Bool s_createDirectoryIfNecessary(const char * szDir) { struct stat statbuf; if (stat(szDir,&statbuf) == 0) // if it exists { if (S_ISDIR(statbuf.st_mode)) // and is a directory return UT_TRUE; UT_DEBUGMSG(("Pathname [%s] is not a directory.\n",szDir)); return UT_FALSE; } if (mkdir(szDir,0700) == 0) return UT_TRUE; UT_DEBUGMSG(("Could not create Directory [%s].\n",szDir)); return UT_FALSE; } UT_Bool AP_BeOSApp::initialize(void) { const char * szUserPrivateDirectory = getUserPrivateDirectory(); UT_Bool bVerified = s_createDirectoryIfNecessary(szUserPrivateDirectory); UT_ASSERT(bVerified); // load the preferences. m_prefs = new AP_BeOSPrefs(this); m_prefs->fullInit(); // now that preferences are established, let the xap init m_pClipboard = new AP_BeOSClipboard(); UT_ASSERT(m_pClipboard); m_pEMC = AP_GetEditMethods(); UT_ASSERT(m_pEMC); m_pBindingSet = new AP_BindingSet(m_pEMC); UT_ASSERT(m_pBindingSet); m_pMenuActionSet = AP_CreateMenuActionSet(); UT_ASSERT(m_pMenuActionSet); m_pToolbarActionSet = AP_CreateToolbarActionSet(); UT_ASSERT(m_pToolbarActionSet); if (! XAP_BeOSApp::initialize()) return UT_FALSE; ////////////////////////////////////////////////////////////////// // initializes the spell checker. ////////////////////////////////////////////////////////////////// #if 1 { const char * szISpellDirectory = NULL; getPrefsValueDirectory(UT_FALSE,AP_PREF_KEY_SpellDirectory,&szISpellDirectory); UT_ASSERT((szISpellDirectory) && (*szISpellDirectory)); const char * szSpellCheckWordList = NULL; getPrefsValue(AP_PREF_KEY_SpellCheckWordList,&szSpellCheckWordList); UT_ASSERT((szSpellCheckWordList) && (*szSpellCheckWordList)); char * szPathname = (char *)calloc(sizeof(char),strlen(szISpellDirectory)+strlen(szSpellCheckWordList)+2); UT_ASSERT(szPathname); sprintf(szPathname,"%s%s%s", szISpellDirectory, ((szISpellDirectory[strlen(szISpellDirectory)-1]=='/') ? "" : "/"), szSpellCheckWordList); UT_DEBUGMSG(("Loading SpellCheckWordList [%s]\n",szPathname)); SpellCheckInit(szPathname); free(szPathname); // we silently go on if we cannot load it.... } #endif ////////////////////////////////////////////////////////////////// // load the dialog and message box strings ////////////////////////////////////////////////////////////////// { // assume we will be using the builtin set (either as the main // set or as the fallback set). AP_BuiltinStringSet * pBuiltinStringSet = new AP_BuiltinStringSet(this,AP_PREF_DEFAULT_StringSet); UT_ASSERT(pBuiltinStringSet); m_pStringSet = pBuiltinStringSet; // see if we should load an alternate set from the disk const char * szDirectory = NULL; const char * szStringSet = NULL; if ( (getPrefsValue(AP_PREF_KEY_StringSet,&szStringSet)) && (szStringSet) && (*szStringSet) && (UT_stricmp(szStringSet,AP_PREF_DEFAULT_StringSet) != 0)) { getPrefsValueDirectory(UT_TRUE,AP_PREF_KEY_StringSetDirectory,&szDirectory); UT_ASSERT((szDirectory) && (*szDirectory)); char * szPathname = (char *)calloc(sizeof(char),strlen(szDirectory)+strlen(szStringSet)+100); UT_ASSERT(szPathname); sprintf(szPathname,"%s%s%s.strings", szDirectory, ((szDirectory[strlen(szDirectory)-1]=='/') ? "" : "/"), szStringSet); AP_DiskStringSet * pDiskStringSet = new AP_DiskStringSet(this); UT_ASSERT(pDiskStringSet); if (pDiskStringSet->loadStringsFromDisk(szPathname)) { pDiskStringSet->setFallbackStringSet(m_pStringSet); m_pStringSet = pDiskStringSet; UT_DEBUGMSG(("Using StringSet [%s]\n",szPathname)); } else { DELETEP(pDiskStringSet); UT_DEBUGMSG(("Unable to load StringSet [%s] -- using builtin strings instead.\n",szPathname)); } free(szPathname); } } // Now we have the strings loaded we can populate the field names correctly // CHECK THIS: the following was added by a Linux developer who can't test // on BeOS, or even compile, so someone with a BeOS box needs to check it int i; for (i = 0; fp_FieldTypes[i].m_Type != FPFIELDTYPE_END; i++) { (&fp_FieldTypes[i])->m_Desc = m_pStringSet->getValue(fp_FieldTypes[i].m_DescId); UT_DEBUGMSG(("Setting field type desc for type %d, desc=%s\n", fp_FieldTypes[i].m_Type, fp_FieldTypes[i].m_Desc)); } for (i = 0; fp_FieldFmts[i].m_Tag != NULL; i++) { (&fp_FieldFmts[i])->m_Desc = m_pStringSet->getValue(fp_FieldFmts[i].m_DescId); UT_DEBUGMSG(("Setting field desc for field %s, desc=%s\n", fp_FieldFmts[i].m_Tag, fp_FieldFmts[i].m_Desc)); } ////////////////////////////////////////////////////////////////// return UT_TRUE; } XAP_Frame * AP_BeOSApp::newFrame(const char *path) { AP_BeOSFrame * pBeOSFrame = new AP_BeOSFrame(this); if (pBeOSFrame) pBeOSFrame->initialize(); pBeOSFrame->loadDocument(path, IEFT_Unknown); return pBeOSFrame; } XAP_Frame * AP_BeOSApp::newFrame(void) { AP_BeOSFrame * pBeOSFrame = new AP_BeOSFrame(this); if (pBeOSFrame) pBeOSFrame->initialize(); return pBeOSFrame; } UT_Bool AP_BeOSApp::shutdown(void) { if (m_prefs->getAutoSavePrefs()) m_prefs->savePrefsFile(); return UT_TRUE; } UT_Bool AP_BeOSApp::getPrefsValueDirectory(UT_Bool bAppSpecific, const XML_Char * szKey, const XML_Char ** pszValue) const { if (!m_prefs) return UT_FALSE; const XML_Char * psz = NULL; if (!m_prefs->getPrefsValue(szKey,&psz)) return UT_FALSE; if (*psz == '/') { *pszValue = psz; return UT_TRUE; } const XML_Char * dir = ((bAppSpecific) ? getAbiSuiteAppDir() : getAbiSuiteLibDir()); UT_ASSERT((dir && *dir)); static XML_Char buf[1024]; UT_ASSERT((strlen(dir) + strlen(psz) + 2) < sizeof(buf)); sprintf(buf,"%s/%s",dir,psz); *pszValue = buf; return UT_TRUE; } const char * AP_BeOSApp::getAbiSuiteAppDir(void) const { // we return a static string, use it quickly. static XML_Char buf[1024]; UT_ASSERT((strlen(getAbiSuiteLibDir()) + strlen(ABIWORD_APP_LIBDIR) + 2) < sizeof(buf)); sprintf(buf,"%s/%s",getAbiSuiteLibDir(),ABIWORD_APP_LIBDIR); return buf; } const XAP_StringSet * AP_BeOSApp::getStringSet(void) const { return m_pStringSet; } ////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////// void AP_BeOSApp::copyToClipboard(PD_DocumentRange * pDocRange) { // copy the given subset of the given document to the // system clipboard in a variety of formats. m_pClipboard->clearClipboard(); // also put RTF on the clipboard IE_Exp_RTF * pExpRtf = new IE_Exp_RTF(pDocRange->m_pDoc); if (pExpRtf) { UT_ByteBuf buf; UT_Error status = pExpRtf->copyToBuffer(pDocRange,&buf); UT_Byte b = 0; buf.append(&b,1); // null terminate string m_pClipboard->addData(AP_CLIPBOARD_RTF,(UT_Byte *)buf.getPointer(0),buf.getLength()); DELETEP(pExpRtf); UT_DEBUGMSG(("CopyToClipboard: copying %d bytes in RTF format.\n",buf.getLength())); } // put raw 8bit text on the clipboard IE_Exp_Text * pExpText = new IE_Exp_Text(pDocRange->m_pDoc); if (pExpText) { UT_ByteBuf buf; UT_Error status = pExpText->copyToBuffer(pDocRange,&buf); UT_Byte b = 0; buf.append(&b,1); // null terminate string m_pClipboard->addData(AP_CLIPBOARD_TEXTPLAIN_8BIT,(UT_Byte *)buf.getPointer(0),buf.getLength()); DELETEP(pExpText); UT_DEBUGMSG(("CopyToClipboard: copying %d bytes in TEXTPLAIN format.\n",buf.getLength())); } } void AP_BeOSApp::pasteFromClipboard(PD_DocumentRange * pDocRange, UT_Bool) { // paste from the system clipboard using the best-for-us format // that is present. if (m_pClipboard->hasFormat(AP_CLIPBOARD_RTF)) { unsigned char * pData = NULL; UT_uint32 iLen = 0; UT_Bool bResult = m_pClipboard->getClipboardData(AP_CLIPBOARD_RTF,(void**)&pData,&iLen); UT_ASSERT(bResult); iLen = MyMin(iLen,strlen((const char *) pData)); UT_DEBUGMSG(("PasteFromClipboard: pasting %d bytes in RTF format.\n",iLen)); IE_Imp_RTF * pImpRTF = new IE_Imp_RTF(pDocRange->m_pDoc); pImpRTF->pasteFromBuffer(pDocRange,pData,iLen); DELETEP(pImpRTF); goto MyEnd; } if (m_pClipboard->hasFormat(AP_CLIPBOARD_TEXTPLAIN_8BIT)) { unsigned char * pData = NULL; UT_uint32 iLen = 0; UT_Bool bResult = m_pClipboard->getClipboardData(AP_CLIPBOARD_TEXTPLAIN_8BIT,(void**)&pData,&iLen); UT_ASSERT(bResult); iLen = MyMin(iLen,strlen((const char *) pData)); UT_DEBUGMSG(("PasteFromClipboard: pasting %d bytes in TEXTPLAIN format.\n",iLen)); IE_Imp_Text * pImpText = new IE_Imp_Text(pDocRange->m_pDoc); pImpText->pasteFromBuffer(pDocRange,pData,iLen); DELETEP(pImpText); goto MyEnd; } // TODO figure out what to do with an image.... UT_DEBUGMSG(("PasteFromClipboard: TODO support this format...")); MyEnd: return; } UT_Bool AP_BeOSApp::canPasteFromClipboard(void) { if (m_pClipboard->hasFormat(AP_CLIPBOARD_RTF)) goto ReturnTrue; if (m_pClipboard->hasFormat(AP_CLIPBOARD_TEXTPLAIN_8BIT)) goto ReturnTrue; return UT_FALSE; ReturnTrue: return UT_TRUE; } /*****************************************************************/ int AP_BeOSApp::local_main(const char * szAppName, int argc, char ** argv) { // This is a static function. #if CONVERT UT_Bool bShowApp = UT_TRUE; #endif UT_DEBUGMSG(("Build ID:\t%s\n", XAP_App::s_szBuild_ID)); UT_DEBUGMSG(("Version:\t%s\n", XAP_App::s_szBuild_Version)); UT_DEBUGMSG(("Build Options: \t%s\n", XAP_App::s_szBuild_Options)); UT_DEBUGMSG(("Build Target: \t%s\n", XAP_App::s_szBuild_Target)); UT_DEBUGMSG(("Compile Date:\t%s\n", XAP_App::s_szBuild_CompileDate)); UT_DEBUGMSG(("Compile Time:\t%s\n", XAP_App::s_szBuild_CompileTime)); // initialize our application. XAP_Args Args = XAP_Args(argc,argv); AP_BeOSApp * pMyBeOSApp = new AP_BeOSApp(&Args, szAppName); //Show the splash screen perhaps _showSplash(&Args, szAppName); // Setup signal handlers, primarily for segfault // If we segfaulted before here, we *really* blew it struct sigaction sa; sa.sa_handler = signalWrapper; sigfillset(&sa.sa_mask); // We don't want to hear about other signals sigdelset(&sa.sa_mask, SIGABRT); // But we will call abort(), so we can't ignore that sa.sa_flags = 0; // Unsupported under BeOS -- SA_NODEFER | SA_RESETHAND; // Don't handle nested signals sigaction(SIGSEGV, &sa, NULL); sigaction(SIGBUS, &sa, NULL); sigaction(SIGILL, &sa, NULL); sigaction(SIGQUIT, &sa, NULL); sigaction(SIGFPE, &sa, NULL); // TODO: handle SIGABRT // if the initialize fails, we don't have icons, fonts, etc. if (!pMyBeOSApp->initialize()) { delete pMyBeOSApp; return -1; // make this something standard? } pMyBeOSApp->ParseCommandLine(); #if CONVERT int nFirstArg = 1; int k; for (k=nFirstArg; (km_BApp.Run(); // destroy the App. It should take care of deleting all frames. pMyBeOSApp->shutdown(); sleep(1); #if CONVERT } #endif delete pMyBeOSApp; return 0; } void AP_BeOSApp::ParseCommandLine(void) { // parse the command line // [-script ]* [-dumpstrings] [-lib ] []* // TODO when we refactor the App classes, consider moving // TODO this to app-specific, cross-platform. // TODO replace this with getopt or something similar. // BeOS puts the program name in argv[0], so [1] is the first argument. int nFirstArg = 1; int k; int kWindowsOpened = 0; #if CONVERT char *to = NULL; int verbose = 1; UT_Bool show = UT_FALSE; #endif for (k=nFirstArg; (km_argc); k++) { if (*m_pArgs->m_argv[k] == '-') { if (UT_stricmp(m_pArgs->m_argv[k],"-script") == 0) { // [-script scriptname] k++; } else if (UT_stricmp(m_pArgs->m_argv[k],"-lib") == 0) { // [-lib ] // we've already processed this when we initialized the App class k++; } else if (UT_stricmp(m_pArgs->m_argv[k],"-dumpstrings") == 0) { // [-dumpstrings] #ifdef DEBUG // dump the string table in english as a template for translators. // see abi/docs/AbiSource_Localization.abw for details. AP_BuiltinStringSet * pBuiltinStringSet = new AP_BuiltinStringSet(this,AP_PREF_DEFAULT_StringSet); pBuiltinStringSet->dumpBuiltinSet("en-US.strings"); delete pBuiltinStringSet; #endif } #if CONVERT else if (UT_stricmp (m_pArgs->m_argv[k],"-to") == 0) { k++; to = m_pArgs->m_argv[k]; } else if (UT_stricmp (m_pArgs->m_argv[k], "-show") == 0) { show = UT_TRUE; } else if (UT_stricmp (m_pArgs->m_argv[k], "-verbose") == 0) { k++; verbose = atoi (m_pArgs->m_argv[k]); } #endif else { UT_DEBUGMSG(("Unknown command line option [%s]\n",m_pArgs->m_argv[k])); // TODO don't know if it has a following argument or not -- assume not } } else { // [filename] #if CONVERT if (to) { AP_Convert * conv = new AP_Convert(); conv->setVerbose(verbose); conv->convertTo(m_pArgs->m_argv[k], to); delete conv; } else { #endif AP_BeOSFrame * pFirstBeOSFrame = new AP_BeOSFrame(this); pFirstBeOSFrame->initialize(); UT_Error error = pFirstBeOSFrame->loadDocument(m_pArgs->m_argv[k], IEFT_Unknown); if (!error) { kWindowsOpened++; } else { // TODO: warn user that we couldn't open that file #if 1 // TODO we crash if we just delete this without putting something // TODO in it, so let's go ahead and open an untitled document // TODO for now. this would cause us to get 2 untitled documents // TODO if the user gave us 2 bogus pathnames.... kWindowsOpened++; pFirstBeOSFrame->loadDocument(NULL, IEFT_Unknown); #else delete pFirstBeOSFrame; #endif } #if CONVERT } #endif } } // command-line conversion may not open any windows at all if (to && !show) return; if (kWindowsOpened == 0) { // no documents specified or were able to be opened, open an untitled one AP_BeOSFrame * pFirstBeOSFrame = new AP_BeOSFrame(this); pFirstBeOSFrame->initialize(); pFirstBeOSFrame->loadDocument(NULL, IEFT_Unknown); } return; } void signalWrapper(int sig_num) { AP_BeOSApp *pApp = (AP_BeOSApp *) XAP_App::getApp(); pApp->catchSignals(sig_num); } static int s_signal_count = 0; void AP_BeOSApp::catchSignals(int sig_num) { // Reset the signal handler (not that it matters - this is mostly for race conditions) signal(SIGSEGV, signalWrapper); s_signal_count = s_signal_count + 1; if(s_signal_count > 1) { UT_DEBUGMSG(("Segfault during filesave - no file saved \n")); fflush(stdout); abort(); } UT_DEBUGMSG(("Oh no - we just segfaulted!\n")); UT_uint32 i = 0; for(;ibackup(); } fflush(stdout); // Abort and dump core abort(); }