/* AbiWord * Copyright (C) 1998 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. */ #include "ap_Features.h" #include "ut_types.h" #include "ut_string.h" #include "ut_assert.h" #include "ut_debugmsg.h" #include "ev_Menu_Labels.h" #include "ev_Menu.h" #include "xap_App.h" #include "xap_Menu_Layouts.h" #include "xap_Menu_ActionSet.h" #include "xap_EncodingManager.h" #include "ap_Menu_Id.h" #include "ap_Menu_Id.h" #include "ap_Strings.h" struct _lt { EV_Menu_LayoutFlags m_flags; XAP_Menu_Id m_id; }; struct _tt { const char * m_name; UT_uint32 m_nrEntries; struct _lt * m_lt; EV_EditMouseContext m_emc; }; class _vectt { public: _vectt(_tt * orig) { m_name = orig->m_name; m_emc = orig->m_emc; m_Vec_lt.clear(); UT_uint32 k = 0; for(k = 0; k < orig->m_nrEntries; k++) { _lt * plt = new _lt; *plt = orig->m_lt[k]; m_Vec_lt.addItem((void *) plt); } }; ~_vectt() { UT_VECTOR_PURGEALL(_lt *,m_Vec_lt); }; UT_uint32 getNrEntries(void) { return m_Vec_lt.getItemCount(); }; _lt * getNth_lt(UT_uint32 n) { return (_lt *) m_Vec_lt.getNthItem(n); }; void insertItemAt(void * p, XAP_Menu_Id id) { UT_uint32 i = 0; bool bFound = false; for(i=0; i< m_Vec_lt.getItemCount(); i++) { _lt * plt = (_lt *) m_Vec_lt.getNthItem(i); if(plt->m_id == id) { if((i+1) == m_Vec_lt.getItemCount()) { m_Vec_lt.addItem(p); } else { m_Vec_lt.insertItemAt(p,i+1); } bFound = true; } } UT_ASSERT_HARMLESS(bFound); }; void removeItem(XAP_Menu_Id id) { UT_uint32 i = 0; bool bFound = false; for(i=0; i< m_Vec_lt.getItemCount(); i++) { _lt * plt = (_lt *) m_Vec_lt.getNthItem(i); if(plt->m_id == id) { m_Vec_lt.deleteNthItem(i); delete plt; bFound = true; } } UT_ASSERT_HARMLESS(bFound); }; void setNthIDFlags(UT_uint32 n, XAP_Menu_Id id,EV_Menu_LayoutFlags flags) { _lt * plt = (_lt *) m_Vec_lt.getNthItem(n); plt->m_flags = flags; plt->m_id = id; }; const char * m_name; EV_EditMouseContext m_emc; private: UT_Vector m_Vec_lt; }; /***************************************************************** ****************************************************************** ** Here we begin a little CPP magic to load the layout for each ** menu layout in the application. It is important that all of ** the ...Layout_*.h files allow themselves to be included more ** than one time. ****************************************************************** *****************************************************************/ #define BeginLayout(Name,Cxt) static struct _lt s_ltTable_##Name[] = { #define MenuItem(id) { EV_MLF_Normal, (id) }, #define BeginSubMenu(id) { EV_MLF_BeginSubMenu, (id) }, #define BeginPopupMenu() { EV_MLF_BeginPopupMenu,AP_MENU_ID__BOGUS1__ }, #define Separator() { EV_MLF_Separator, AP_MENU_ID__BOGUS1__ }, #define EndSubMenu() { EV_MLF_EndSubMenu, AP_MENU_ID__BOGUS1__ }, #define EndPopupMenu() { EV_MLF_EndPopupMenu, AP_MENU_ID__BOGUS1__ }, #define EndLayout() }; #include "ap_Menu_Layouts_All.h" #undef BeginLayout #undef MenuItem #undef BeginSubMenu #undef BeginPopupMenu #undef Separator #undef EndSubMenu #undef EndPopupMenu #undef EndLayout /***************************************************************** ****************************************************************** ** Here we begin a little CPP magic to construct a table containing ** the names and addresses of all the tables we constructed in the ** previous section. ****************************************************************** *****************************************************************/ #define BeginLayout(Name,Cxt) { #Name, NrElements(s_ltTable_##Name), s_ltTable_##Name, Cxt }, #define MenuItem(id) /*nothing*/ #define BeginSubMenu(id) /*nothing*/ #define BeginPopupMenu() /*nothing*/ #define Separator() /*nothing*/ #define EndSubMenu() /*nothing*/ #define EndPopupMenu() /*nothing*/ #define EndLayout() /*nothing*/ static struct _tt s_ttTable[] = { #include "ap_Menu_Layouts_All.h" }; #undef BeginLayout #undef MenuItem #undef BeginSubMenu #undef BeginPopupMenu #undef Separator #undef EndSubMenu #undef EndPopupMenu #undef EndLayout /***************************************************************** ****************************************************************** ** Put it all together and have a "load Layout by Name" ****************************************************************** *****************************************************************/ /*! * Load these cleverly constructed static tables into vectors so they can be * manipulated dynamically. */ XAP_Menu_Factory::XAP_Menu_Factory(XAP_App * pApp) : m_pApp(pApp), m_pLabelSet(NULL), m_maxID(0) { UT_uint32 k = 0; m_vecTT.clear(); for (k=0; k 0) { m_maxID++; return m_maxID; } UT_uint32 i,j =0; for(i=0; i < m_vecTT.getItemCount(); i++) { _vectt * pTT = (_vectt *) m_vecTT.getNthItem(i); for(j=0; j < pTT->getNrEntries(); j++) { _lt * plt = (_lt *) pTT->getNth_lt(j); if(plt->m_id > m_maxID) { m_maxID = plt->m_id; } } } m_maxID++; return m_maxID; } EV_Menu_Layout * XAP_Menu_Factory::CreateMenuLayout(const char * szName) { UT_return_val_if_fail (szName && *szName, NULL); // no defaults for (UT_uint32 k=0; k< m_vecTT.getItemCount(); k++) { _vectt * pVectt = (_vectt *)m_vecTT.getNthItem(k); if (UT_stricmp(szName,pVectt->m_name)==0) { UT_uint32 NrEntries = pVectt->getNrEntries(); EV_Menu_Layout * pLayout = new EV_Menu_Layout(pVectt->m_name,NrEntries); UT_return_val_if_fail (pLayout, NULL); for (UT_uint32 j=0; (j < NrEntries); j++) { _lt * plt = pVectt->getNth_lt(j); bool bResult = pLayout->setLayoutItem(j, plt->m_id, plt->m_flags); UT_ASSERT_HARMLESS(bResult); } return pLayout; } } UT_ASSERT_HARMLESS(0); // no defaults return NULL; } const char * XAP_Menu_Factory::FindContextMenu(EV_EditMouseContext emc) { for (UT_uint32 k=0; k< m_vecTT.getItemCount(); k++) { _vectt * pVectt = (_vectt *) m_vecTT.getNthItem(k); UT_DEBUGMSG(("Look menu %s id %x requested %x \n",pVectt->m_name,pVectt->m_emc,emc)); if (emc == pVectt->m_emc) { return pVectt->m_name; } } UT_ASSERT_HARMLESS(UT_NOT_IMPLEMENTED); return NULL; } XAP_Menu_Id XAP_Menu_Factory::addNewMenuAfter(const char * szMenu, const char * szLanguage, const char * szAfter, EV_Menu_LayoutFlags flags, XAP_Menu_Id newID) { UT_return_val_if_fail (szMenu && *szMenu, 0); // no defaults UT_uint32 k = 0; bool bFoundMenu = false; _vectt * pVectt = NULL; for (k=0; (k< m_vecTT.getItemCount()) && !bFoundMenu; k++) { pVectt = (_vectt *)m_vecTT.getNthItem(k); bFoundMenu = (UT_stricmp(szMenu,pVectt->m_name)==0); } if(!bFoundMenu) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return 0; } // next we need to find the id of the label name // // OK now search for this label // UT_String After = szAfter; XAP_Menu_Id afterID = EV_searchMenuLabel( m_pLabelSet, After); if(afterID == 0) { if(m_pEnglishLabelSet == NULL) { buildBuiltInMenuLabelSet( m_pEnglishLabelSet); } afterID = EV_searchMenuLabel( m_pEnglishLabelSet, After); if(afterID == 0) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return 0; } } // // OK got the menu ID at last, insert the new id here. // if(newID == 0) { newID = getNewID(); } // // Now insert our new ID // _lt * plt = new _lt; plt->m_id = newID; plt->m_flags = flags; pVectt->insertItemAt((void *) plt, afterID); return (XAP_Menu_Id) newID; } XAP_Menu_Id XAP_Menu_Factory::addNewMenuBefore(const char * szMenu, const char * szLanguage, const char * szBefore, EV_Menu_LayoutFlags flags, XAP_Menu_Id newID) { UT_return_val_if_fail (szMenu && *szMenu, 0); // no defaults UT_uint32 k = 0; bool bFoundMenu = false; _vectt * pVectt = NULL; for (k=0; (k< m_vecTT.getItemCount()) && !bFoundMenu; k++) { pVectt = (_vectt *)m_vecTT.getNthItem(k); bFoundMenu = (UT_stricmp(szMenu,pVectt->m_name)==0); } if(!bFoundMenu) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return 0; } // next we need to find the id of the label name // // OK now search for this label // UT_String Before = szBefore; XAP_Menu_Id beforeID = EV_searchMenuLabel( m_pLabelSet, Before); if(beforeID == 0) { if(m_pEnglishLabelSet == NULL) { buildBuiltInMenuLabelSet( m_pEnglishLabelSet); } beforeID = EV_searchMenuLabel( m_pEnglishLabelSet, Before); if(beforeID == 0) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return 0; } } // // OK got the menu ID at last, insert the new id here. // if(newID == 0) { newID = getNewID(); } // // Now insert our new ID // _lt * plt = new _lt; plt->m_id = newID; plt->m_flags = flags; if(beforeID > 0) { pVectt->insertItemAt((void *) plt, beforeID-1); } else { pVectt->insertItemAt((void *) plt, beforeID); } return (XAP_Menu_Id) newID; } /*! * Remove the menu item named szNuke from the menu labelled szMenu in the * Language set szLanguage */ XAP_Menu_Id XAP_Menu_Factory::removeMenuItem(const char * szMenu, const char * szLanguage, const char * szNuke) { UT_return_val_if_fail (szMenu && *szMenu, 0); // no defaults UT_uint32 k = 0; bool bFoundMenu = false; _vectt * pVectt = NULL; for (k=0; (k< m_vecTT.getItemCount()) && !bFoundMenu; k++) { pVectt = (_vectt *)m_vecTT.getNthItem(k); bFoundMenu = (UT_stricmp(szMenu,pVectt->m_name)==0); } if(!bFoundMenu) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return 0; } // next we need to find the id of the label name // // OK now search for this label // UT_String Nuke = szNuke; XAP_Menu_Id nukeID = EV_searchMenuLabel( m_pLabelSet, Nuke); if(nukeID == 0) { if(m_pEnglishLabelSet == NULL) { buildBuiltInMenuLabelSet( m_pEnglishLabelSet); } nukeID = EV_searchMenuLabel( m_pEnglishLabelSet, Nuke); if(nukeID == 0) { UT_ASSERT_HARMLESS(UT_SHOULD_NOT_HAPPEN); return 0; } } // // Got the id. Now remove it. // pVectt->removeItem(nukeID); return nukeID; } void XAP_Menu_Factory::resetMenusToDefault(void) { UT_VECTOR_PURGEALL(_vectt *,m_vecTT); m_vecTT.clear(); UT_uint32 k = 0; for (k=0; kgetStringSet(); if( !m_pLabelSet ) { m_pLabelSet = new EV_Menu_LabelSet(szLanguage,AP_MENU_ID__BOGUS1__,AP_MENU_ID__BOGUS2__); UT_UTF8String s1, s2; #define menuitem(id) \ { \ pSS->getValueUTF8(AP_STRING_ID_MENU_LABEL_##id, s1); \ pSS->getValueUTF8(AP_STRING_ID_MENU_STATUSLINE_##id, s2); \ m_pLabelSet->setLabel((AP_MENU_ID_##id), s1.utf8_str(), s2.utf8_str() ); \ } #include "ap_Menu_Id_List.h" #undef menuitem return true; } return false; } /*! * Build a builtin label set in memory that can be used by plugins to find where to put menu items in non-Enmgish locales. * Do tricky reference to pointer here.... */ bool XAP_Menu_Factory::buildBuiltInMenuLabelSet(EV_Menu_LabelSet *& pLabelSet) { if(m_pBSS == NULL) { AP_BuiltinStringSet * pBSS = new AP_BuiltinStringSet(m_pApp,"en-US"); m_pBSS = (XAP_StringSet *) pBSS; } pLabelSet = new EV_Menu_LabelSet("en-US",AP_MENU_ID__BOGUS1__,AP_MENU_ID__BOGUS2__); UT_UTF8String s1, s2; #define menuitem(id) \ { \ m_pBSS->getValueUTF8(AP_STRING_ID_MENU_LABEL_##id, s1); \ m_pBSS->getValueUTF8(AP_STRING_ID_MENU_STATUSLINE_##id, s2); \ pLabelSet->setLabel( (AP_MENU_ID_##id), s1.utf8_str(), s2.utf8_str() );\ } #include "ap_Menu_Id_List.h" #undef menuitem return true; } EV_Menu_LabelSet * XAP_Menu_Factory::CreateMenuLabelSet(const char * szLanguage_) { char buf[300]; strncpy(buf,szLanguage_ ? szLanguage_ : "", sizeof(buf)-1); char* szLanguage = buf; char* dot = strrchr(szLanguage,'.'); if (dot) *dot = '\0'; /* remove encoding part from locale name */ UT_DEBUGMSG(("CreateMenuLabelSet: szLanguage_ %s, szLanguage %s\n" ,szLanguage_,szLanguage)); const XAP_StringSet * pSS = XAP_App::getApp()->getStringSet(); if( !m_pLabelSet ) { m_pLabelSet = new EV_Menu_LabelSet(szLanguage,AP_MENU_ID__BOGUS1__,AP_MENU_ID__BOGUS2__); UT_UTF8String s1, s2; #define menuitem(id) \ { \ pSS->getValueUTF8(AP_STRING_ID_MENU_LABEL_##id, s1); \ pSS->getValueUTF8(AP_STRING_ID_MENU_STATUSLINE_##id, s2); \ m_pLabelSet->setLabel( (AP_MENU_ID_##id), s1.utf8_str(), s2.utf8_str() ); \ } #include "ap_Menu_Id_List.h" #undef menuitem } return new EV_Menu_LabelSet(m_pLabelSet); } UT_uint32 XAP_Menu_Factory::GetMenuLabelSetLanguageCount(void) { return 1; } const char * XAP_Menu_Factory::GetNthMenuLabelLanguageName(UT_uint32 ndx) { return m_pLabelSet->getLanguage(); } /*! * afterID has just been created. All labels of value equal or greater * must be incremented */ bool XAP_Menu_Factory::addNewLabel(const char * szLanguage, XAP_Menu_Id newID, const char * szNewName, const char * szNewTooltip) { // TODO incrementing label IDs?? EV_Menu_Label * newLab = new EV_Menu_Label(newID,szNewName,szNewTooltip); return m_pLabelSet->addLabel(newLab); } bool XAP_Menu_Factory::removeLabel(const char * szLanguage, XAP_Menu_Id nukeID) { return false; } void XAP_Menu_Factory::resetLabelsToDefault(void) { // TODO - do nothing as extra labels in set don't hurt operation }