BCMenu.cpp 81 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052
  1. #include "stdafx.h" // Standard windows header file
  2. #include "BCMenu.h" // BCMenu class declaration
  3. #include <afxpriv.h> //SK: makes A2W and other spiffy AFX macros work
  4. #include <VersionHelpers.h>
  5. #ifdef _DEBUG
  6. #define new DEBUG_NEW
  7. #undef THIS_FILE
  8. static char THIS_FILE[] = __FILE__;
  9. #endif
  10. #define BCMENU_GAP 1
  11. #ifndef OBM_CHECK
  12. #define OBM_CHECK 32760 // from winuser.h
  13. #endif
  14. #if _MFC_VER <0x400
  15. #error This code does not work on Versions of MFC prior to 4.0
  16. #endif
  17. static CPINFO CPInfo;
  18. // how the menu's are drawn in win9x/NT/2000
  19. UINT BCMenu::original_drawmode=BCMENU_DRAWMODE_ORIGINAL;
  20. BOOL BCMenu::original_select_disabled=TRUE;
  21. // how the menu's are drawn in winXP
  22. UINT BCMenu::xp_drawmode=BCMENU_DRAWMODE_XP;
  23. BOOL BCMenu::xp_select_disabled=FALSE;
  24. BOOL BCMenu::xp_draw_3D_bitmaps=TRUE;
  25. CImageList BCMenu::m_AllImages;
  26. CArray<int,int&> BCMenu::m_AllImagesID;
  27. enum Win32Type{
  28. Win32s,
  29. WinNT3,
  30. Win95,
  31. Win98,
  32. WinME,
  33. WinNT4,
  34. Win2000,
  35. WinXP
  36. };
  37. Win32Type IsShellType()
  38. {
  39. Win32Type ShellType;
  40. if (IsWindowsXPOrGreater())
  41. {
  42. ShellType = WinXP;
  43. }
  44. else if (IsWindowsXPSP3OrGreater())
  45. {
  46. ShellType = WinXP;
  47. }
  48. else
  49. {
  50. ShellType = WinNT4;
  51. }
  52. return ShellType;
  53. }
  54. static Win32Type g_Shell=IsShellType();
  55. void BCMenuData::SetAnsiString(LPCSTR szAnsiString)
  56. {
  57. USES_CONVERSION;
  58. SetWideString(A2W(szAnsiString)); //SK: see MFC Tech Note 059
  59. }
  60. CString BCMenuData::GetString(void)//returns the menu text in ANSI or UNICODE
  61. //depending on the MFC-Version we are using
  62. {
  63. CString strText;
  64. if (m_szMenuText)
  65. {
  66. #ifdef UNICODE
  67. strText = m_szMenuText;
  68. #else
  69. USES_CONVERSION;
  70. strText=W2A(m_szMenuText); //SK: see MFC Tech Note 059
  71. #endif
  72. }
  73. return strText;
  74. }
  75. CTypedPtrArray<CPtrArray, HMENU> BCMenu::m_AllSubMenus; // Stores list of all sub-menus
  76. IMPLEMENT_DYNAMIC( BCMenu, CMenu )
  77. /*
  78. ===============================================================================
  79. BCMenu::BCMenu()
  80. BCMenu::~BCMenu()
  81. -----------------
  82. Constructor and Destructor.
  83. ===============================================================================
  84. */
  85. BCMenu::BCMenu()
  86. {
  87. m_bDynIcons = FALSE; // O.S. - no dynamic icons by default
  88. disable_old_style=FALSE;
  89. m_iconX = 16; // Icon sizes default to 16 x 16
  90. m_iconY = 15; // ...
  91. m_selectcheck = -1;
  92. m_unselectcheck = -1;
  93. checkmaps=NULL;
  94. checkmapsshare=FALSE;
  95. // set the color used for the transparent background in all bitmaps
  96. m_bitmapBackground=RGB(192,192,192); //gray
  97. m_bitmapBackgroundFlag=FALSE;
  98. GetCPInfo(CP_ACP,&CPInfo);
  99. m_loadmenu=FALSE;
  100. }
  101. BCMenu::~BCMenu()
  102. {
  103. DestroyMenu();
  104. }
  105. BOOL BCMenu::IsNewShell ()
  106. {
  107. return (g_Shell>=Win95);
  108. }
  109. BOOL BCMenu::IsWinXPLuna()
  110. {
  111. if(g_Shell==WinXP){
  112. if(IsWindowsClassicTheme())return(FALSE);
  113. else return(TRUE);
  114. }
  115. return(FALSE);
  116. }
  117. BOOL BCMenu::IsLunaMenuStyle()
  118. {
  119. if(IsWinXPLuna()){
  120. if(xp_drawmode==BCMENU_DRAWMODE_XP)return(TRUE);
  121. }
  122. else{
  123. if(original_drawmode==BCMENU_DRAWMODE_XP)return(TRUE);
  124. }
  125. return(FALSE);
  126. }
  127. BCMenuData::~BCMenuData()
  128. {
  129. if(bitmap)
  130. delete(bitmap);
  131. delete[] m_szMenuText; //Need not check for NULL because ANSI X3J16 allows "delete NULL"
  132. }
  133. void BCMenuData::SetWideString(const wchar_t *szWideString)
  134. {
  135. delete[] m_szMenuText;//Need not check for NULL because ANSI X3J16 allows "delete NULL"
  136. if (szWideString)
  137. {
  138. m_szMenuText = new wchar_t[sizeof(wchar_t)*(wcslen(szWideString)+1)];
  139. if (m_szMenuText)
  140. wcscpy_s(m_szMenuText, sizeof(wchar_t) * (wcslen(szWideString) + 1), szWideString);
  141. }
  142. else
  143. m_szMenuText=NULL;//set to NULL so we need not bother about dangling non-NULL Ptrs
  144. }
  145. BOOL BCMenu::IsMenu(CMenu *submenu)
  146. {
  147. int m;
  148. int numSubMenus = m_AllSubMenus.GetUpperBound();
  149. for(m=0;m<=numSubMenus;++m){
  150. if(submenu->m_hMenu==m_AllSubMenus[m])return(TRUE);
  151. }
  152. return(FALSE);
  153. }
  154. BOOL BCMenu::IsMenu(HMENU submenu)
  155. {
  156. int m;
  157. int numSubMenus = m_AllSubMenus.GetUpperBound();
  158. for(m=0;m<=numSubMenus;++m){
  159. if(submenu==m_AllSubMenus[m])return(TRUE);
  160. }
  161. return(FALSE);
  162. }
  163. BOOL BCMenu::DestroyMenu()
  164. {
  165. // Destroy Sub menus:
  166. int m,n;
  167. int numAllSubMenus = m_AllSubMenus.GetUpperBound();
  168. for(n = numAllSubMenus; n>= 0; n--){
  169. if(m_AllSubMenus[n]==this->m_hMenu)m_AllSubMenus.RemoveAt(n);
  170. }
  171. int numSubMenus = m_SubMenus.GetUpperBound();
  172. for(m = numSubMenus; m >= 0; m--){
  173. numAllSubMenus = m_AllSubMenus.GetUpperBound();
  174. for(n = numAllSubMenus; n>= 0; n--){
  175. if(m_AllSubMenus[n]==m_SubMenus[m])m_AllSubMenus.RemoveAt(n);
  176. }
  177. CMenu *ptr=FromHandle(m_SubMenus[m]);
  178. BOOL flag=ptr->IsKindOf(RUNTIME_CLASS( BCMenu ));
  179. if(flag)delete((BCMenu *)ptr);
  180. }
  181. m_SubMenus.RemoveAll();
  182. // Destroy menu data
  183. int numItems = m_MenuList.GetUpperBound();
  184. for(m = 0; m <= numItems; m++)delete(m_MenuList[m]);
  185. m_MenuList.RemoveAll();
  186. if(checkmaps&&!checkmapsshare){
  187. delete checkmaps;
  188. checkmaps=NULL;
  189. }
  190. // Call base-class implementation last:
  191. return(CMenu::DestroyMenu());
  192. };
  193. int BCMenu::GetMenuDrawMode(void)
  194. {
  195. if(IsWinXPLuna())return(xp_drawmode);
  196. return(original_drawmode);
  197. }
  198. BOOL BCMenu::GetSelectDisableMode(void)
  199. {
  200. if(IsLunaMenuStyle())return(xp_select_disabled);
  201. return(original_select_disabled);
  202. }
  203. /*
  204. ==========================================================================
  205. void BCMenu::DrawItem(LPDRAWITEMSTRUCT)
  206. ---------------------------------------
  207. Called by the framework when a particular item needs to be drawn. We
  208. overide this to draw the menu item in a custom-fashion, including icons
  209. and the 3D rectangle bar.
  210. ==========================================================================
  211. */
  212. void BCMenu::DrawItem (LPDRAWITEMSTRUCT lpDIS)
  213. {
  214. ASSERT(lpDIS != NULL);
  215. CDC* pDC = CDC::FromHandle(lpDIS->hDC);
  216. if(pDC->GetDeviceCaps(RASTERCAPS) & RC_PALETTE)DrawItem_Win9xNT2000(lpDIS);
  217. else{
  218. if(IsWinXPLuna()){
  219. if(xp_drawmode==BCMENU_DRAWMODE_XP) DrawItem_WinXP(lpDIS);
  220. else DrawItem_Win9xNT2000(lpDIS);
  221. }
  222. else{
  223. if(original_drawmode==BCMENU_DRAWMODE_XP) DrawItem_WinXP(lpDIS);
  224. else DrawItem_Win9xNT2000(lpDIS);
  225. }
  226. }
  227. }
  228. void BCMenu::DrawItem_Win9xNT2000 (LPDRAWITEMSTRUCT lpDIS)
  229. {
  230. ASSERT(lpDIS != NULL);
  231. CDC* pDC = CDC::FromHandle(lpDIS->hDC);
  232. CRect rect;
  233. UINT state = (((BCMenuData*)(lpDIS->itemData))->nFlags);
  234. CBrush m_brBackground;
  235. COLORREF m_clrBack;
  236. if(IsWinXPLuna())m_clrBack=GetSysColor(COLOR_3DFACE);
  237. else m_clrBack=GetSysColor(COLOR_MENU);
  238. m_brBackground.CreateSolidBrush(m_clrBack);
  239. // remove the selected bit if it's grayed out
  240. if(lpDIS->itemState & ODS_GRAYED&&!original_select_disabled){
  241. if(lpDIS->itemState & ODS_SELECTED)lpDIS->itemState=lpDIS->itemState & ~ODS_SELECTED;
  242. }
  243. if(state & MF_SEPARATOR){
  244. rect.CopyRect(&lpDIS->rcItem);
  245. pDC->FillRect (rect,&m_brBackground);
  246. rect.top += (rect.Height()>>1);
  247. pDC->DrawEdge(&rect,EDGE_ETCHED,BF_TOP);
  248. }
  249. else{
  250. CRect rect2;
  251. BOOL standardflag=FALSE,selectedflag=FALSE,disableflag=FALSE;
  252. BOOL checkflag=FALSE;
  253. COLORREF crText = GetSysColor(COLOR_MENUTEXT);
  254. CBrush m_brSelect;
  255. CPen m_penBack;
  256. int x0,y0,dy;
  257. int nIconNormal=-1,xoffset=-1,global_offset=-1;
  258. CImageList *bitmap=NULL;
  259. // set some colors
  260. m_penBack.CreatePen (PS_SOLID,0,m_clrBack);
  261. m_brSelect.CreateSolidBrush(GetSysColor(COLOR_HIGHLIGHT));
  262. // draw the colored rectangle portion
  263. rect.CopyRect(&lpDIS->rcItem);
  264. rect2=rect;
  265. // draw the up/down/focused/disabled state
  266. UINT state = lpDIS->itemState;
  267. CString strText;
  268. if(lpDIS->itemData != NULL){
  269. nIconNormal = (((BCMenuData*)(lpDIS->itemData))->menuIconNormal);
  270. xoffset = (((BCMenuData*)(lpDIS->itemData))->xoffset);
  271. global_offset = (((BCMenuData*)(lpDIS->itemData))->global_offset);
  272. bitmap = (((BCMenuData*)(lpDIS->itemData))->bitmap);
  273. strText = ((BCMenuData*) (lpDIS->itemData))->GetString();
  274. if(nIconNormal<0&&global_offset>=0){
  275. xoffset=global_offset;
  276. nIconNormal=0;
  277. bitmap = &m_AllImages;
  278. }
  279. if(state&ODS_CHECKED && nIconNormal<0){
  280. if(state&ODS_SELECTED && m_selectcheck>0)checkflag=TRUE;
  281. else if(m_unselectcheck>0) checkflag=TRUE;
  282. }
  283. else if(nIconNormal != -1){
  284. standardflag=TRUE;
  285. if(state&ODS_SELECTED && !(state&ODS_GRAYED))selectedflag=TRUE;
  286. else if(state&ODS_GRAYED) disableflag=TRUE;
  287. }
  288. }
  289. else{
  290. strText.Empty();
  291. }
  292. if(state&ODS_SELECTED){ // draw the down edges
  293. CPen *pOldPen = pDC->SelectObject (&m_penBack);
  294. // You need only Text highlight and thats what you get
  295. if(checkflag||standardflag||selectedflag||disableflag||state&ODS_CHECKED)
  296. rect2.SetRect(rect.left+m_iconX+4+BCMENU_GAP,rect.top,rect.right,rect.bottom);
  297. pDC->FillRect (rect2,&m_brSelect);
  298. pDC->SelectObject (pOldPen);
  299. crText = GetSysColor(COLOR_HIGHLIGHTTEXT);
  300. }
  301. else {
  302. CPen *pOldPen = pDC->SelectObject (&m_penBack);
  303. pDC->FillRect (rect,&m_brBackground);
  304. pDC->SelectObject (pOldPen);
  305. // draw the up edges
  306. pDC->Draw3dRect (rect,m_clrBack,m_clrBack);
  307. }
  308. // draw the text if there is any
  309. //We have to paint the text only if the image is nonexistant
  310. dy = (rect.Height()-4-m_iconY)/2;
  311. dy = dy<0 ? 0 : dy;
  312. if(checkflag||standardflag||selectedflag||disableflag){
  313. rect2.SetRect(rect.left+1,rect.top+1+dy,rect.left+m_iconX+3,
  314. rect.top+m_iconY+3+dy);
  315. pDC->Draw3dRect (rect2,m_clrBack,m_clrBack);
  316. if(checkflag && checkmaps){
  317. pDC->FillRect (rect2,&m_brBackground);
  318. rect2.SetRect(rect.left,rect.top+dy,rect.left+m_iconX+4,
  319. rect.top+m_iconY+4+dy);
  320. pDC->Draw3dRect (rect2,m_clrBack,m_clrBack);
  321. CPoint ptImage(rect.left+2,rect.top+2+dy);
  322. if(state&ODS_SELECTED)checkmaps->Draw(pDC,1,ptImage,ILD_TRANSPARENT);
  323. else checkmaps->Draw(pDC,0,ptImage,ILD_TRANSPARENT);
  324. }
  325. else if(disableflag){
  326. if(!selectedflag){
  327. CBitmap bitmapstandard;
  328. GetBitmapFromImageList(pDC,bitmap,xoffset,bitmapstandard);
  329. rect2.SetRect(rect.left,rect.top+dy,rect.left+m_iconX+4,
  330. rect.top+m_iconY+4+dy);
  331. pDC->Draw3dRect (rect2,m_clrBack,m_clrBack);
  332. if(disable_old_style)
  333. DitherBlt(lpDIS->hDC,rect.left+2,rect.top+2+dy,m_iconX,m_iconY,
  334. (HBITMAP)(bitmapstandard),0,0,m_clrBack);
  335. else
  336. DitherBlt2(pDC,rect.left+2,rect.top+2+dy,m_iconX,m_iconY,
  337. bitmapstandard,0,0,m_clrBack);
  338. bitmapstandard.DeleteObject();
  339. }
  340. }
  341. else if(selectedflag){
  342. pDC->FillRect (rect2,&m_brBackground);
  343. rect2.SetRect(rect.left,rect.top+dy,rect.left+m_iconX+4,
  344. rect.top+m_iconY+4+dy);
  345. if (IsNewShell()){
  346. if(state&ODS_CHECKED)
  347. pDC->Draw3dRect(rect2,GetSysColor(COLOR_3DSHADOW),
  348. GetSysColor(COLOR_3DHILIGHT));
  349. else
  350. pDC->Draw3dRect(rect2,GetSysColor(COLOR_3DHILIGHT),
  351. GetSysColor(COLOR_3DSHADOW));
  352. }
  353. CPoint ptImage(rect.left+2,rect.top+2+dy);
  354. if(bitmap)bitmap->Draw(pDC,xoffset,ptImage,ILD_TRANSPARENT);
  355. }
  356. else{
  357. if(state&ODS_CHECKED){
  358. CBrush brush;
  359. COLORREF col = m_clrBack;
  360. col = LightenColor(col,0.6);
  361. brush.CreateSolidBrush(col);
  362. pDC->FillRect(rect2,&brush);
  363. brush.DeleteObject();
  364. rect2.SetRect(rect.left,rect.top+dy,rect.left+m_iconX+4,
  365. rect.top+m_iconY+4+dy);
  366. if (IsNewShell())
  367. pDC->Draw3dRect(rect2,GetSysColor(COLOR_3DSHADOW),
  368. GetSysColor(COLOR_3DHILIGHT));
  369. }
  370. else{
  371. pDC->FillRect (rect2,&m_brBackground);
  372. rect2.SetRect(rect.left,rect.top+dy,rect.left+m_iconX+4,
  373. rect.top+m_iconY+4+dy);
  374. pDC->Draw3dRect (rect2,m_clrBack,m_clrBack);
  375. }
  376. CPoint ptImage(rect.left+2,rect.top+2+dy);
  377. if(bitmap)bitmap->Draw(pDC,xoffset,ptImage,ILD_TRANSPARENT);
  378. }
  379. }
  380. if(nIconNormal<0 && state&ODS_CHECKED && !checkflag){
  381. rect2.SetRect(rect.left+1,rect.top+2+dy,rect.left+m_iconX+1,
  382. rect.top+m_iconY+2+dy);
  383. CMenuItemInfo info;
  384. info.fMask = MIIM_CHECKMARKS;
  385. ::GetMenuItemInfo((HMENU)lpDIS->hwndItem,lpDIS->itemID,
  386. MF_BYCOMMAND, &info);
  387. if(state&ODS_CHECKED || info.hbmpUnchecked) {
  388. Draw3DCheckmark(pDC, rect2, state&ODS_SELECTED,
  389. state&ODS_CHECKED ? info.hbmpChecked :
  390. info.hbmpUnchecked);
  391. }
  392. }
  393. //This is needed always so that we can have the space for check marks
  394. x0=rect.left;y0=rect.top;
  395. rect.left = rect.left + m_iconX + 8 + BCMENU_GAP;
  396. if(!strText.IsEmpty()){
  397. CRect rectt(rect.left,rect.top-1,rect.right,rect.bottom-1);
  398. // Find tabs
  399. CString leftStr,rightStr;
  400. leftStr.Empty();rightStr.Empty();
  401. int tablocr=strText.ReverseFind(_T('\t'));
  402. if(tablocr!=-1){
  403. rightStr=strText.Mid(tablocr+1);
  404. leftStr=strText.Left(strText.Find(_T('\t')));
  405. rectt.right-=m_iconX;
  406. }
  407. else leftStr=strText;
  408. int iOldMode = pDC->GetBkMode();
  409. pDC->SetBkMode( TRANSPARENT);
  410. // Draw the text in the correct colour:
  411. UINT nFormat = DT_LEFT|DT_SINGLELINE|DT_VCENTER;
  412. UINT nFormatr = DT_RIGHT|DT_SINGLELINE|DT_VCENTER;
  413. if(!(lpDIS->itemState & ODS_GRAYED)){
  414. pDC->SetTextColor(crText);
  415. pDC->DrawText (leftStr,rectt,nFormat);
  416. if(tablocr!=-1) pDC->DrawText (rightStr,rectt,nFormatr);
  417. }
  418. else{
  419. // Draw the disabled text
  420. if(!(state & ODS_SELECTED)){
  421. RECT offset = *rectt;
  422. offset.left+=1;
  423. offset.right+=1;
  424. offset.top+=1;
  425. offset.bottom+=1;
  426. pDC->SetTextColor(GetSysColor(COLOR_BTNHILIGHT));
  427. pDC->DrawText(leftStr,&offset, nFormat);
  428. if(tablocr!=-1) pDC->DrawText (rightStr,&offset,nFormatr);
  429. pDC->SetTextColor(GetSysColor(COLOR_GRAYTEXT));
  430. pDC->DrawText(leftStr,rectt, nFormat);
  431. if(tablocr!=-1) pDC->DrawText (rightStr,rectt,nFormatr);
  432. }
  433. else{
  434. // And the standard Grey text:
  435. pDC->SetTextColor(m_clrBack);
  436. pDC->DrawText(leftStr,rectt, nFormat);
  437. if(tablocr!=-1) pDC->DrawText (rightStr,rectt,nFormatr);
  438. }
  439. }
  440. pDC->SetBkMode( iOldMode );
  441. }
  442. m_penBack.DeleteObject();
  443. m_brSelect.DeleteObject();
  444. }
  445. m_brBackground.DeleteObject();
  446. }
  447. COLORREF BCMenu::LightenColor(COLORREF col,double factor)
  448. {
  449. if(factor>0.0&&factor<=1.0){
  450. BYTE red,green,blue,lightred,lightgreen,lightblue;
  451. red = GetRValue(col);
  452. green = GetGValue(col);
  453. blue = GetBValue(col);
  454. lightred = (BYTE)((factor*(255-red)) + red);
  455. lightgreen = (BYTE)((factor*(255-green)) + green);
  456. lightblue = (BYTE)((factor*(255-blue)) + blue);
  457. col = RGB(lightred,lightgreen,lightblue);
  458. }
  459. return(col);
  460. }
  461. COLORREF BCMenu::DarkenColor(COLORREF col,double factor)
  462. {
  463. if(factor>0.0&&factor<=1.0){
  464. BYTE red,green,blue,lightred,lightgreen,lightblue;
  465. red = GetRValue(col);
  466. green = GetGValue(col);
  467. blue = GetBValue(col);
  468. lightred = (BYTE)(red-(factor*red));
  469. lightgreen = (BYTE)(green-(factor*green));
  470. lightblue = (BYTE)(blue-(factor*blue));
  471. col = RGB(lightred,lightgreen,lightblue);
  472. }
  473. return(col);
  474. }
  475. void BCMenu::DrawItem_WinXP (LPDRAWITEMSTRUCT lpDIS)
  476. {
  477. ASSERT(lpDIS != NULL);
  478. CDC* pDC = CDC::FromHandle(lpDIS->hDC);
  479. #ifdef BCMENU_USE_MEMDC
  480. BCMenuMemDC *pMemDC=NULL;
  481. #endif
  482. CRect rect,rect2;
  483. UINT state = (((BCMenuData*)(lpDIS->itemData))->nFlags);
  484. COLORREF m_newclrBack=GetSysColor(COLOR_3DFACE);
  485. COLORREF m_clrBack=GetSysColor(COLOR_WINDOW);
  486. CFont m_fontMenu,*pFont=NULL;
  487. LOGFONT m_lf;
  488. if(!IsWinXPLuna())m_newclrBack=LightenColor(m_newclrBack,0.25);
  489. CBrush m_newbrBackground,m_brBackground;
  490. m_brBackground.CreateSolidBrush(m_clrBack);
  491. m_newbrBackground.CreateSolidBrush(m_newclrBack);
  492. int BCMENU_PAD=4;
  493. if(xp_draw_3D_bitmaps)BCMENU_PAD=7;
  494. int barwidth=m_iconX+BCMENU_PAD;
  495. // remove the selected bit if it's grayed out
  496. if(lpDIS->itemState & ODS_GRAYED&&!xp_select_disabled){
  497. if(lpDIS->itemState & ODS_SELECTED)lpDIS->itemState=lpDIS->itemState & ~ODS_SELECTED;
  498. #ifdef BCMENU_USE_MEMDC
  499. pMemDC=new BCMenuMemDC(pDC,&lpDIS->rcItem);
  500. pDC = pMemDC;
  501. ZeroMemory ((PVOID) &m_lf,sizeof (LOGFONT));
  502. NONCLIENTMETRICS nm;
  503. nm.cbSize = sizeof (NONCLIENTMETRICS);
  504. VERIFY (SystemParametersInfo(SPI_GETNONCLIENTMETRICS,nm.cbSize,&nm,0));
  505. m_lf = nm.lfMenuFont;
  506. m_fontMenu.CreateFontIndirect (&m_lf);
  507. pFont = pDC->SelectObject (&m_fontMenu);
  508. #endif
  509. }
  510. if(state & MF_SEPARATOR){
  511. rect.CopyRect(&lpDIS->rcItem);
  512. pDC->FillRect (rect,&m_brBackground);
  513. rect2.SetRect(rect.left,rect.top,rect.left+barwidth,rect.bottom);
  514. rect.top+=rect.Height()>>1;
  515. rect.left = rect2.right+BCMENU_PAD;
  516. pDC->DrawEdge(&rect,EDGE_ETCHED,BF_TOP);
  517. pDC->FillRect (rect2,&m_newbrBackground);
  518. pDC->Draw3dRect (rect2,m_newclrBack,m_newclrBack);
  519. }
  520. else{
  521. BOOL standardflag=FALSE,selectedflag=FALSE,disableflag=FALSE;
  522. BOOL checkflag=FALSE;
  523. COLORREF crText = GetSysColor(COLOR_MENUTEXT);
  524. COLORREF crSelect = GetSysColor(COLOR_HIGHLIGHT);
  525. COLORREF crSelectFill;
  526. if(!IsWinXPLuna())crSelectFill=LightenColor(crSelect,0.85);
  527. else crSelectFill=LightenColor(crSelect,0.7);
  528. CBrush m_brSelect;
  529. CPen m_penBack;
  530. int x0,y0,dx,dy;
  531. int nIconNormal=-1,xoffset=-1,global_offset=-1;
  532. int faded_offset=1,shadow_offset=2;
  533. CImageList *bitmap=NULL;
  534. BOOL CanDraw3D=FALSE;
  535. // set some colors
  536. m_penBack.CreatePen (PS_SOLID,0,m_clrBack);
  537. m_brSelect.CreateSolidBrush(crSelectFill);
  538. // draw the colored rectangle portion
  539. rect.CopyRect(&lpDIS->rcItem);
  540. rect2=rect;
  541. // draw the up/down/focused/disabled state
  542. UINT state = lpDIS->itemState;
  543. CString strText;
  544. if(lpDIS->itemData != NULL){
  545. nIconNormal = (((BCMenuData*)(lpDIS->itemData))->menuIconNormal);
  546. xoffset = (((BCMenuData*)(lpDIS->itemData))->xoffset);
  547. bitmap = (((BCMenuData*)(lpDIS->itemData))->bitmap);
  548. strText = ((BCMenuData*) (lpDIS->itemData))->GetString();
  549. global_offset = (((BCMenuData*)(lpDIS->itemData))->global_offset);
  550. if(xoffset==0&&xp_draw_3D_bitmaps&&bitmap&&bitmap->GetImageCount()>2)CanDraw3D=TRUE;
  551. if(nIconNormal<0&&xoffset<0&&global_offset>=0){
  552. xoffset=global_offset;
  553. nIconNormal=0;
  554. bitmap = &m_AllImages;
  555. if(xp_draw_3D_bitmaps&&CanDraw3DImageList(global_offset)){
  556. CanDraw3D=TRUE;
  557. faded_offset=global_offset+1;
  558. shadow_offset=global_offset+2;
  559. }
  560. }
  561. if(state&ODS_CHECKED && nIconNormal<0){
  562. if(state&ODS_SELECTED && m_selectcheck>0)checkflag=TRUE;
  563. else if(m_unselectcheck>0) checkflag=TRUE;
  564. }
  565. else if(nIconNormal != -1){
  566. standardflag=TRUE;
  567. if(state&ODS_SELECTED && !(state&ODS_GRAYED))selectedflag=TRUE;
  568. else if(state&ODS_GRAYED) disableflag=TRUE;
  569. }
  570. }
  571. else{
  572. strText.Empty();
  573. }
  574. if(state&ODS_SELECTED){ // draw the down edges
  575. CPen *pOldPen = pDC->SelectObject (&m_penBack);
  576. pDC->FillRect (rect,&m_brSelect);
  577. pDC->Draw3dRect (rect,crSelect,crSelect);
  578. pDC->SelectObject (pOldPen);
  579. }
  580. else {
  581. rect2.SetRect(rect.left,rect.top,rect.left+barwidth,rect.bottom);
  582. CPen *pOldPen = pDC->SelectObject (&m_penBack);
  583. pDC->FillRect (rect,&m_brBackground);
  584. pDC->FillRect (rect2,&m_newbrBackground);
  585. pDC->SelectObject (pOldPen);
  586. // draw the up edges
  587. pDC->Draw3dRect (rect,m_clrBack,m_clrBack);
  588. pDC->Draw3dRect (rect2,m_newclrBack,m_newclrBack);
  589. }
  590. // draw the text if there is any
  591. //We have to paint the text only if the image is nonexistant
  592. dy = (int)(0.5+(rect.Height()-m_iconY)/2.0);
  593. dy = dy<0 ? 0 : dy;
  594. dx = (int)(0.5+(barwidth-m_iconX)/2.0);
  595. dx = dx<0 ? 0 : dx;
  596. if(checkflag||standardflag||selectedflag||disableflag){
  597. rect2.SetRect(rect.left+1,rect.top+1,rect.left+barwidth-1,
  598. rect.bottom-1);
  599. if(checkflag && checkmaps){
  600. pDC->FillRect (rect2,&m_newbrBackground);
  601. CPoint ptImage(rect.left+dx,rect.top+dy);
  602. if(state&ODS_SELECTED)checkmaps->Draw(pDC,1,ptImage,ILD_TRANSPARENT);
  603. else checkmaps->Draw(pDC,0,ptImage,ILD_TRANSPARENT);
  604. }
  605. else if(disableflag){
  606. if(!selectedflag){
  607. CBitmap bitmapstandard;
  608. GetBitmapFromImageList(pDC,bitmap,xoffset,bitmapstandard);
  609. COLORREF transparentcol=m_newclrBack;
  610. if(state&ODS_SELECTED)transparentcol=crSelectFill;
  611. if(disable_old_style)
  612. DitherBlt(lpDIS->hDC,rect.left+dx,rect.top+dy,m_iconX,m_iconY,
  613. (HBITMAP)(bitmapstandard),0,0,transparentcol);
  614. else
  615. DitherBlt2(pDC,rect.left+dx,rect.top+dy,m_iconX,m_iconY,
  616. bitmapstandard,0,0,transparentcol);
  617. if(state&ODS_SELECTED)pDC->Draw3dRect (rect,crSelect,crSelect);
  618. bitmapstandard.DeleteObject();
  619. }
  620. }
  621. else if(selectedflag){
  622. pDC->FillRect (rect2,&m_brSelect);
  623. CPoint ptImage(rect.left+dx,rect.top+dy);
  624. if(state&ODS_CHECKED){
  625. pDC->Draw3dRect(rect2,crSelect,crSelect);
  626. ptImage.x-=1;ptImage.y-=1;
  627. }
  628. if(bitmap){
  629. if(CanDraw3D&&!(state&ODS_CHECKED)){
  630. CPoint ptImage1(ptImage.x+1,ptImage.y+1);
  631. CPoint ptImage2(ptImage.x-1,ptImage.y-1);
  632. bitmap->Draw(pDC,shadow_offset,ptImage1,ILD_TRANSPARENT);
  633. bitmap->Draw(pDC,xoffset,ptImage2,ILD_TRANSPARENT);
  634. }
  635. else bitmap->Draw(pDC,xoffset,ptImage,ILD_TRANSPARENT);
  636. }
  637. }
  638. else{
  639. if(state&ODS_CHECKED){
  640. CBrush brushin;
  641. brushin.CreateSolidBrush(LightenColor(crSelect,0.85));
  642. pDC->FillRect(rect2,&brushin);
  643. brushin.DeleteObject();
  644. pDC->Draw3dRect(rect2,crSelect,crSelect);
  645. CPoint ptImage(rect.left+dx-1,rect.top+dy-1);
  646. if(bitmap)bitmap->Draw(pDC,xoffset,ptImage,ILD_TRANSPARENT);
  647. }
  648. else{
  649. pDC->FillRect (rect2,&m_newbrBackground);
  650. pDC->Draw3dRect (rect2,m_newclrBack,m_newclrBack);
  651. CPoint ptImage(rect.left+dx,rect.top+dy);
  652. if(bitmap){
  653. if(CanDraw3D)
  654. bitmap->Draw(pDC,faded_offset,ptImage,ILD_TRANSPARENT);
  655. else
  656. bitmap->Draw(pDC,xoffset,ptImage,ILD_TRANSPARENT);
  657. }
  658. }
  659. }
  660. }
  661. if(nIconNormal<0 && state&ODS_CHECKED && !checkflag){
  662. dy = (int)(0.5+(rect.Height()-16)/2.0);
  663. dy = dy<0 ? 0 : dy;
  664. dx = (int)(0.5+(barwidth-16)/2.0);
  665. dx = dx<0 ? 0 : dx;
  666. CMenuItemInfo info;
  667. info.fMask = MIIM_CHECKMARKS;
  668. ::GetMenuItemInfo((HMENU)lpDIS->hwndItem,lpDIS->itemID,
  669. MF_BYCOMMAND, &info);
  670. if(state&ODS_CHECKED || info.hbmpUnchecked) {
  671. rect2.SetRect(rect.left+dx,rect.top+dy,rect.left+dx+16,rect.top+dy+16);
  672. DrawXPCheckmark(pDC, rect2,state&ODS_CHECKED ? info.hbmpChecked :
  673. info.hbmpUnchecked,crSelect);
  674. }
  675. }
  676. //This is needed always so that we can have the space for check marks
  677. x0=rect.left;y0=rect.top;
  678. rect.left = rect.left + barwidth + 8;
  679. if(!strText.IsEmpty()){
  680. CRect rectt(rect.left,rect.top-1,rect.right,rect.bottom-1);
  681. // Find tabs
  682. CString leftStr,rightStr;
  683. leftStr.Empty();rightStr.Empty();
  684. int tablocr=strText.ReverseFind(_T('\t'));
  685. if(tablocr!=-1){
  686. rightStr=strText.Mid(tablocr+1);
  687. leftStr=strText.Left(strText.Find(_T('\t')));
  688. rectt.right-=m_iconX;
  689. }
  690. else leftStr=strText;
  691. int iOldMode = pDC->GetBkMode();
  692. pDC->SetBkMode( TRANSPARENT);
  693. // Draw the text in the correct colour:
  694. UINT nFormat = DT_LEFT|DT_SINGLELINE|DT_VCENTER;
  695. UINT nFormatr = DT_RIGHT|DT_SINGLELINE|DT_VCENTER;
  696. if(!(lpDIS->itemState & ODS_GRAYED)){
  697. pDC->SetTextColor(crText);
  698. pDC->DrawText (leftStr,rectt,nFormat);
  699. if(tablocr!=-1) pDC->DrawText (rightStr,rectt,nFormatr);
  700. }
  701. else{
  702. RECT offset = *rectt;
  703. offset.left+=1;
  704. offset.right+=1;
  705. offset.top+=1;
  706. offset.bottom+=1;
  707. if(!IsWinXPLuna()){
  708. COLORREF graycol=GetSysColor(COLOR_GRAYTEXT);
  709. if(!(state&ODS_SELECTED))graycol = LightenColor(graycol,0.4);
  710. pDC->SetTextColor(graycol);
  711. }
  712. else pDC->SetTextColor(GetSysColor(COLOR_GRAYTEXT));
  713. pDC->DrawText(leftStr,rectt, nFormat);
  714. if(tablocr!=-1) pDC->DrawText (rightStr,rectt,nFormatr);
  715. }
  716. pDC->SetBkMode( iOldMode );
  717. }
  718. m_penBack.DeleteObject();
  719. m_brSelect.DeleteObject();
  720. }
  721. m_brBackground.DeleteObject();
  722. m_newbrBackground.DeleteObject();
  723. #ifdef BCMENU_USE_MEMDC
  724. if(pFont)pDC->SelectObject (pFont); //set it to the old font
  725. m_fontMenu.DeleteObject();
  726. if(pMemDC)delete pMemDC;
  727. #endif
  728. }
  729. BOOL BCMenu::GetBitmapFromImageList(CDC* pDC,CImageList *imglist,int nIndex,CBitmap &bmp)
  730. {
  731. HICON hIcon = imglist->ExtractIcon(nIndex);
  732. CDC dc;
  733. dc.CreateCompatibleDC(pDC);
  734. bmp.CreateCompatibleBitmap(pDC,m_iconX,m_iconY);
  735. CBitmap* pOldBmp = dc.SelectObject(&bmp);
  736. CBrush brush ;
  737. COLORREF m_newclrBack;
  738. m_newclrBack=GetSysColor(COLOR_3DFACE);
  739. brush.CreateSolidBrush(m_newclrBack);
  740. ::DrawIconEx(
  741. dc.GetSafeHdc(),
  742. 0,
  743. 0,
  744. hIcon,
  745. m_iconX,
  746. m_iconY,
  747. 0,
  748. (HBRUSH)brush,
  749. DI_NORMAL
  750. );
  751. dc.SelectObject( pOldBmp );
  752. dc.DeleteDC();
  753. // the icon is not longer needed
  754. ::DestroyIcon(hIcon);
  755. return(TRUE);
  756. }
  757. /*
  758. ==========================================================================
  759. void BCMenu::MeasureItem(LPMEASUREITEMSTRUCT)
  760. ---------------------------------------------
  761. Called by the framework when it wants to know what the width and height
  762. of our item will be. To accomplish this we provide the width of the
  763. icon plus the width of the menu text, and then the height of the icon.
  764. ==========================================================================
  765. */
  766. void BCMenu::MeasureItem( LPMEASUREITEMSTRUCT lpMIS )
  767. {
  768. UINT state = (((BCMenuData*)(lpMIS->itemData))->nFlags);
  769. int BCMENU_PAD=4;
  770. if(IsLunaMenuStyle()&&xp_draw_3D_bitmaps)BCMENU_PAD=7;
  771. if(state & MF_SEPARATOR){
  772. lpMIS->itemWidth = 0;
  773. int temp = GetSystemMetrics(SM_CYMENU)>>1;
  774. lpMIS->itemHeight = temp>(m_iconY+BCMENU_PAD)/2 ? temp : (m_iconY+BCMENU_PAD)/2;
  775. }
  776. else{
  777. CFont m_fontMenu;
  778. LOGFONT m_lf;
  779. ZeroMemory ((PVOID) &m_lf,sizeof (LOGFONT));
  780. NONCLIENTMETRICS nm;
  781. nm.cbSize = sizeof (NONCLIENTMETRICS);
  782. VERIFY(SystemParametersInfo(SPI_GETNONCLIENTMETRICS,
  783. nm.cbSize,&nm,0));
  784. m_lf = nm.lfMenuFont;
  785. m_fontMenu.CreateFontIndirect (&m_lf);
  786. // Obtain the width of the text:
  787. CWnd *pWnd = AfxGetMainWnd(); // Get main window
  788. if (pWnd == NULL) pWnd = CWnd::GetDesktopWindow();
  789. CDC *pDC = pWnd->GetDC(); // Get device context
  790. CFont* pFont=NULL; // Select menu font in...
  791. if (IsNewShell())
  792. pFont = pDC->SelectObject (&m_fontMenu);// Select menu font in...
  793. //Get pointer to text SK
  794. const wchar_t *lpstrText = ((BCMenuData*)(lpMIS->itemData))->GetWideString();//SK: we use const to prevent misuse
  795. SIZE size;
  796. size.cx=size.cy=0;
  797. if (Win32s!=g_Shell)
  798. VERIFY(::GetTextExtentPoint32W(pDC->m_hDC,lpstrText,
  799. wcslen(lpstrText),&size)); //SK should also work on 95
  800. #ifndef UNICODE //can't be UNICODE for Win32s
  801. else{//it's Win32suckx
  802. RECT rect;
  803. rect.left=rect.top=0;
  804. size.cy=DrawText(pDC->m_hDC,(LPCTSTR)lpstrText,
  805. wcslen(lpstrText),&rect,
  806. DT_SINGLELINE|DT_LEFT|DT_VCENTER|DT_CALCRECT);
  807. //+3 makes at least three pixels space to the menu border
  808. size.cx=rect.right-rect.left+3;
  809. size.cx += 3*(size.cx/wcslen(lpstrText));
  810. }
  811. #endif
  812. CSize t = CSize(size);
  813. if(IsNewShell())
  814. pDC->SelectObject (pFont); // Select old font in
  815. pWnd->ReleaseDC(pDC); // Release the DC
  816. // Set width and height:
  817. if(IsLunaMenuStyle())lpMIS->itemWidth = m_iconX+BCMENU_PAD+8+t.cx;
  818. else lpMIS->itemWidth = m_iconX + t.cx + m_iconX + BCMENU_GAP;
  819. int temp = GetSystemMetrics(SM_CYMENU);
  820. lpMIS->itemHeight = temp>m_iconY+BCMENU_PAD ? temp : m_iconY+BCMENU_PAD;
  821. m_fontMenu.DeleteObject();
  822. }
  823. }
  824. void BCMenu::SetIconSize (int width, int height)
  825. {
  826. m_iconX = width;
  827. m_iconY = height;
  828. }
  829. BOOL BCMenu::AppendODMenuA(LPCSTR lpstrText,UINT nFlags,UINT nID,
  830. int nIconNormal)
  831. {
  832. USES_CONVERSION;
  833. return AppendODMenuW(A2W(lpstrText),nFlags,nID,nIconNormal);//SK: See MFC Tech Note 059
  834. }
  835. BOOL BCMenu::AppendODMenuW(wchar_t *lpstrText,UINT nFlags,UINT nID,
  836. int nIconNormal)
  837. {
  838. // Add the MF_OWNERDRAW flag if not specified:
  839. if(!nID){
  840. if(nFlags&MF_BYPOSITION)nFlags=MF_SEPARATOR|MF_OWNERDRAW|MF_BYPOSITION;
  841. else nFlags=MF_SEPARATOR|MF_OWNERDRAW;
  842. }
  843. else if(!(nFlags & MF_OWNERDRAW))nFlags |= MF_OWNERDRAW;
  844. if(nFlags & MF_POPUP){
  845. m_AllSubMenus.Add((HMENU)nID);
  846. m_SubMenus.Add((HMENU)nID);
  847. }
  848. BCMenuData *mdata = new BCMenuData;
  849. m_MenuList.Add(mdata);
  850. mdata->SetWideString(lpstrText); //SK: modified for dynamic allocation
  851. mdata->menuIconNormal = -1;
  852. mdata->xoffset = -1;
  853. if(nIconNormal>=0){
  854. CImageList bitmap;
  855. int xoffset=0;
  856. LoadFromToolBar(nID,nIconNormal,xoffset);
  857. if(mdata->bitmap){
  858. mdata->bitmap->DeleteImageList();
  859. mdata->bitmap=NULL;
  860. }
  861. bitmap.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  862. if(AddBitmapToImageList(&bitmap,nIconNormal)){
  863. mdata->global_offset = AddToGlobalImageList(&bitmap,xoffset,nID);
  864. }
  865. }
  866. else mdata->global_offset = GlobalImageListOffset(nID);
  867. mdata->nFlags = nFlags;
  868. mdata->nID = nID;
  869. BOOL returnflag=CMenu::AppendMenu(nFlags, nID, (LPCTSTR)mdata);
  870. if(m_loadmenu)RemoveTopLevelOwnerDraw();
  871. return(returnflag);
  872. }
  873. BOOL BCMenu::AppendODMenuA(LPCSTR lpstrText,UINT nFlags,UINT nID,
  874. CImageList *il,int xoffset)
  875. {
  876. USES_CONVERSION;
  877. return AppendODMenuW(A2W(lpstrText),nFlags,nID,il,xoffset);
  878. }
  879. BOOL BCMenu::AppendODMenuW(wchar_t *lpstrText,UINT nFlags,UINT nID,
  880. CImageList *il,int xoffset)
  881. {
  882. // Add the MF_OWNERDRAW flag if not specified:
  883. if(!nID){
  884. if(nFlags&MF_BYPOSITION)nFlags=MF_SEPARATOR|MF_OWNERDRAW|MF_BYPOSITION;
  885. else nFlags=MF_SEPARATOR|MF_OWNERDRAW;
  886. }
  887. else if(!(nFlags & MF_OWNERDRAW))nFlags |= MF_OWNERDRAW;
  888. if(nFlags & MF_POPUP){
  889. m_AllSubMenus.Add((HMENU)nID);
  890. m_SubMenus.Add((HMENU)nID);
  891. }
  892. BCMenuData *mdata = new BCMenuData;
  893. m_MenuList.Add(mdata);
  894. mdata->SetWideString(lpstrText); //SK: modified for dynamic allocation
  895. if(il){
  896. mdata->menuIconNormal = 0;
  897. mdata->xoffset=0;
  898. if(mdata->bitmap)mdata->bitmap->DeleteImageList();
  899. else mdata->bitmap=new(CImageList);
  900. ImageListDuplicate(il,xoffset,mdata->bitmap);
  901. }
  902. else{
  903. mdata->menuIconNormal = -1;
  904. mdata->xoffset = -1;
  905. }
  906. mdata->nFlags = nFlags;
  907. mdata->nID = nID;
  908. return(CMenu::AppendMenu(nFlags, nID, (LPCTSTR)mdata));
  909. }
  910. BOOL BCMenu::InsertODMenuA(UINT nPosition,LPCSTR lpstrText,UINT nFlags,UINT nID,
  911. int nIconNormal)
  912. {
  913. USES_CONVERSION;
  914. return InsertODMenuW(nPosition,A2W(lpstrText),nFlags,nID,nIconNormal);
  915. }
  916. BOOL BCMenu::InsertODMenuW(UINT nPosition,wchar_t *lpstrText,UINT nFlags,UINT nID,
  917. int nIconNormal)
  918. {
  919. if(!(nFlags & MF_BYPOSITION)){
  920. int iPosition =0;
  921. BCMenu* pMenu = FindMenuOption(nPosition,iPosition);
  922. if(pMenu){
  923. return(pMenu->InsertODMenuW(iPosition,lpstrText,nFlags|MF_BYPOSITION,nID,nIconNormal));
  924. }
  925. else return(FALSE);
  926. }
  927. if(!nID)nFlags=MF_SEPARATOR|MF_OWNERDRAW|MF_BYPOSITION;
  928. else if(!(nFlags & MF_OWNERDRAW))nFlags |= MF_OWNERDRAW;
  929. int menustart=0;
  930. if(nFlags & MF_POPUP){
  931. if(m_loadmenu){
  932. menustart=GetMenuStart();
  933. if(nPosition<(UINT)menustart)menustart=0;
  934. }
  935. m_AllSubMenus.Add((HMENU)nID);
  936. m_SubMenus.InsertAt(nPosition,(HMENU)nID);
  937. }
  938. //Stephane Clog suggested adding this, believe it or not it's in the help
  939. if(nPosition==(UINT)-1)nPosition=GetMenuItemCount();
  940. BCMenuData *mdata = new BCMenuData;
  941. m_MenuList.InsertAt(nPosition-menustart,mdata);
  942. mdata->SetWideString(lpstrText); //SK: modified for dynamic allocation
  943. mdata->menuIconNormal = nIconNormal;
  944. mdata->xoffset=-1;
  945. if(nIconNormal>=0){
  946. CImageList bitmap;
  947. int xoffset=0;
  948. LoadFromToolBar(nID,nIconNormal,xoffset);
  949. if(mdata->bitmap){
  950. mdata->bitmap->DeleteImageList();
  951. mdata->bitmap=NULL;
  952. }
  953. bitmap.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  954. if(AddBitmapToImageList(&bitmap,nIconNormal)){
  955. mdata->global_offset = AddToGlobalImageList(&bitmap,xoffset,nID);
  956. }
  957. }
  958. else mdata->global_offset = GlobalImageListOffset(nID);
  959. mdata->nFlags = nFlags;
  960. mdata->nID = nID;
  961. BOOL returnflag=CMenu::InsertMenu(nPosition,nFlags,nID,(LPCTSTR)mdata);
  962. if(m_loadmenu)RemoveTopLevelOwnerDraw();
  963. return(returnflag);
  964. }
  965. BOOL BCMenu::InsertODMenuA(UINT nPosition,LPCSTR lpstrText,UINT nFlags,UINT nID,
  966. CImageList *il,int xoffset)
  967. {
  968. USES_CONVERSION;
  969. return InsertODMenuW(nPosition,A2W(lpstrText),nFlags,nID,il,xoffset);
  970. }
  971. BOOL BCMenu::InsertODMenuW(UINT nPosition,wchar_t *lpstrText,UINT nFlags,UINT nID,
  972. CImageList *il,int xoffset)
  973. {
  974. if(!(nFlags & MF_BYPOSITION)){
  975. int iPosition =0;
  976. BCMenu* pMenu = FindMenuOption(nPosition,iPosition);
  977. if(pMenu){
  978. return(pMenu->InsertODMenuW(iPosition,lpstrText,nFlags|MF_BYPOSITION,nID,il,xoffset));
  979. }
  980. else return(FALSE);
  981. }
  982. if(!nID)nFlags=MF_SEPARATOR|MF_OWNERDRAW|MF_BYPOSITION;
  983. else if(!(nFlags & MF_OWNERDRAW))nFlags |= MF_OWNERDRAW;
  984. if(nFlags & MF_POPUP){
  985. m_AllSubMenus.Add((HMENU)nID);
  986. m_SubMenus.InsertAt(nPosition,(HMENU)nID);
  987. }
  988. //Stephane Clog suggested adding this, believe it or not it's in the help
  989. if(nPosition==(UINT)-1)nPosition=GetMenuItemCount();
  990. BCMenuData *mdata = new BCMenuData;
  991. m_MenuList.InsertAt(nPosition,mdata);
  992. mdata->SetWideString(lpstrText); //SK: modified for dynamic allocation
  993. mdata->menuIconNormal = -1;
  994. mdata->xoffset = -1;
  995. if(il){
  996. if(mdata->bitmap){
  997. mdata->bitmap->DeleteImageList();
  998. mdata->bitmap=NULL;
  999. }
  1000. mdata->global_offset = AddToGlobalImageList(il,xoffset,nID);
  1001. }
  1002. mdata->nFlags = nFlags;
  1003. mdata->nID = nID;
  1004. return(CMenu::InsertMenu(nPosition,nFlags,nID,(LPCTSTR)mdata));
  1005. }
  1006. BOOL BCMenu::ModifyODMenuA(const char * lpstrText,UINT nID,int nIconNormal)
  1007. {
  1008. USES_CONVERSION;
  1009. return ModifyODMenuW(A2W(lpstrText),nID,nIconNormal);//SK: see MFC Tech Note 059
  1010. }
  1011. BOOL BCMenu::ModifyODMenuW(wchar_t *lpstrText,UINT nID,int nIconNormal)
  1012. {
  1013. int nLoc;
  1014. BCMenuData *mdata;
  1015. CArray<BCMenu*,BCMenu*>bcsubs;
  1016. CArray<int,int&>bclocs;
  1017. // Find the old BCMenuData structure:
  1018. BCMenu *psubmenu = FindMenuOption(nID,nLoc);
  1019. do{
  1020. if(psubmenu && nLoc>=0)mdata = psubmenu->m_MenuList[nLoc];
  1021. else{
  1022. // Create a new BCMenuData structure:
  1023. mdata = new BCMenuData;
  1024. m_MenuList.Add(mdata);
  1025. }
  1026. ASSERT(mdata);
  1027. if(lpstrText)
  1028. mdata->SetWideString(lpstrText); //SK: modified for dynamic allocation
  1029. mdata->menuIconNormal = -1;
  1030. mdata->xoffset = -1;
  1031. if(nIconNormal>=0){
  1032. CImageList bitmap;
  1033. int xoffset=0;
  1034. LoadFromToolBar(nID,nIconNormal,xoffset);
  1035. if(mdata->bitmap){
  1036. mdata->bitmap->DeleteImageList();
  1037. mdata->bitmap=NULL;
  1038. }
  1039. bitmap.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  1040. if(AddBitmapToImageList(&bitmap,nIconNormal)){
  1041. mdata->global_offset = AddToGlobalImageList(&bitmap,xoffset,nID);
  1042. }
  1043. }
  1044. else mdata->global_offset = GlobalImageListOffset(nID);
  1045. mdata->nFlags &= ~(MF_BYPOSITION);
  1046. mdata->nFlags |= MF_OWNERDRAW;
  1047. mdata->nID = nID;
  1048. bcsubs.Add(psubmenu);
  1049. bclocs.Add(nLoc);
  1050. if(psubmenu && nLoc>=0)psubmenu = FindAnotherMenuOption(nID,nLoc,bcsubs,bclocs);
  1051. else psubmenu=NULL;
  1052. }while(psubmenu);
  1053. return (CMenu::ModifyMenu(nID,mdata->nFlags,nID,(LPCTSTR)mdata));
  1054. }
  1055. BOOL BCMenu::ModifyODMenuA(const char * lpstrText,UINT nID,CImageList *il,int xoffset)
  1056. {
  1057. USES_CONVERSION;
  1058. return ModifyODMenuW(A2W(lpstrText),nID,il,xoffset);
  1059. }
  1060. BOOL BCMenu::ModifyODMenuW(wchar_t *lpstrText,UINT nID,CImageList *il,int xoffset)
  1061. {
  1062. int nLoc;
  1063. BCMenuData *mdata;
  1064. CArray<BCMenu*,BCMenu*>bcsubs;
  1065. CArray<int,int&>bclocs;
  1066. // Find the old BCMenuData structure:
  1067. BCMenu *psubmenu = FindMenuOption(nID,nLoc);
  1068. do{
  1069. if(psubmenu && nLoc>=0)mdata = psubmenu->m_MenuList[nLoc];
  1070. else{
  1071. // Create a new BCMenuData structure:
  1072. mdata = new BCMenuData;
  1073. m_MenuList.Add(mdata);
  1074. }
  1075. ASSERT(mdata);
  1076. if(lpstrText)
  1077. mdata->SetWideString(lpstrText); //SK: modified for dynamic allocation
  1078. mdata->menuIconNormal = -1;
  1079. mdata->xoffset = -1;
  1080. if(il){
  1081. if(mdata->bitmap){
  1082. mdata->bitmap->DeleteImageList();
  1083. mdata->bitmap=NULL;
  1084. }
  1085. mdata->global_offset = AddToGlobalImageList(il,xoffset,nID);
  1086. }
  1087. mdata->nFlags &= ~(MF_BYPOSITION);
  1088. mdata->nFlags |= MF_OWNERDRAW;
  1089. mdata->nID = nID;
  1090. bcsubs.Add(psubmenu);
  1091. bclocs.Add(nLoc);
  1092. if(psubmenu && nLoc>=0)psubmenu = FindAnotherMenuOption(nID,nLoc,bcsubs,bclocs);
  1093. else psubmenu=NULL;
  1094. }while(psubmenu);
  1095. return (CMenu::ModifyMenu(nID,mdata->nFlags,nID,(LPCTSTR)mdata));
  1096. }
  1097. BOOL BCMenu::ModifyODMenuA(const char * lpstrText,UINT nID,CBitmap *bmp)
  1098. {
  1099. USES_CONVERSION;
  1100. return ModifyODMenuW(A2W(lpstrText),nID,bmp);
  1101. }
  1102. BOOL BCMenu::ModifyODMenuW(wchar_t *lpstrText,UINT nID,CBitmap *bmp)
  1103. {
  1104. if(bmp){
  1105. CImageList temp;
  1106. temp.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  1107. if(m_bitmapBackgroundFlag)temp.Add(bmp,m_bitmapBackground);
  1108. else temp.Add(bmp,GetSysColor(COLOR_3DFACE));
  1109. return ModifyODMenuW(lpstrText,nID,&temp,0);
  1110. }
  1111. return ModifyODMenuW(lpstrText,nID,NULL,0);
  1112. }
  1113. // courtesy of Warren Stevens
  1114. BOOL BCMenu::ModifyODMenuA(const char * lpstrText,UINT nID,COLORREF fill,COLORREF border,int hatchstyle,CSize *pSize)
  1115. {
  1116. USES_CONVERSION;
  1117. return ModifyODMenuW(A2W(lpstrText),nID,fill,border,hatchstyle,pSize);
  1118. }
  1119. BOOL BCMenu::ModifyODMenuW(wchar_t *lpstrText,UINT nID,COLORREF fill,COLORREF border,int hatchstyle,CSize *pSize)
  1120. {
  1121. CWnd *pWnd = AfxGetMainWnd(); // Get main window
  1122. CDC *pDC = pWnd->GetDC(); // Get device context
  1123. SIZE sz;
  1124. if(!pSize){
  1125. sz.cx = m_iconX;
  1126. sz.cy = m_iconY;
  1127. }
  1128. else{
  1129. sz.cx = pSize->cx;
  1130. sz.cy = pSize->cy;
  1131. }
  1132. CSize bitmap_size(sz);
  1133. CSize icon_size(m_iconX,m_iconY);
  1134. CBitmap bmp;
  1135. ColorBitmap(pDC,bmp,bitmap_size,icon_size,fill,border,hatchstyle);
  1136. pWnd->ReleaseDC(pDC);
  1137. return ModifyODMenuW(lpstrText,nID,&bmp);
  1138. }
  1139. BOOL BCMenu::ModifyODMenuA(const char *lpstrText,const char *OptionText,int nIconNormal)
  1140. {
  1141. USES_CONVERSION;
  1142. return ModifyODMenuW(A2W(lpstrText),A2W(OptionText),nIconNormal);//SK: see MFC Tech Note 059
  1143. }
  1144. BOOL BCMenu::ModifyODMenuW(wchar_t *lpstrText,wchar_t *OptionText,int nIconNormal)
  1145. {
  1146. BCMenuData *mdata;
  1147. // Find the old BCMenuData structure:
  1148. CStringW junk=OptionText;
  1149. mdata=FindMenuOption(OptionText);
  1150. if(mdata){
  1151. if(lpstrText)
  1152. mdata->SetWideString(lpstrText);//SK: modified for dynamic allocation
  1153. mdata->menuIconNormal = nIconNormal;
  1154. mdata->xoffset=-1;
  1155. if(nIconNormal>=0){
  1156. mdata->xoffset=0;
  1157. if(mdata->bitmap)mdata->bitmap->DeleteImageList();
  1158. else mdata->bitmap=new(CImageList);
  1159. mdata->bitmap->Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  1160. if(!AddBitmapToImageList(mdata->bitmap,nIconNormal)){
  1161. mdata->bitmap->DeleteImageList();
  1162. delete mdata->bitmap;
  1163. mdata->bitmap=NULL;
  1164. mdata->menuIconNormal = nIconNormal = -1;
  1165. mdata->xoffset = -1;
  1166. }
  1167. }
  1168. return(TRUE);
  1169. }
  1170. return(FALSE);
  1171. }
  1172. BCMenuData *BCMenu::NewODMenu(UINT pos,UINT nFlags,UINT nID,CString string)
  1173. {
  1174. BCMenuData *mdata;
  1175. mdata = new BCMenuData;
  1176. mdata->menuIconNormal = -1;
  1177. mdata->xoffset=-1;
  1178. #ifdef UNICODE
  1179. mdata->SetWideString((LPCTSTR)string);//SK: modified for dynamic allocation
  1180. #else
  1181. mdata->SetAnsiString(string);
  1182. #endif
  1183. mdata->nFlags = nFlags;
  1184. mdata->nID = nID;
  1185. // if(nFlags & MF_POPUP)m_AllSubMenus.Add((HMENU)nID);
  1186. if (nFlags&MF_OWNERDRAW){
  1187. ASSERT(!(nFlags&MF_STRING));
  1188. ModifyMenu(pos,nFlags,nID,(LPCTSTR)mdata);
  1189. }
  1190. else if (nFlags&MF_STRING){
  1191. ASSERT(!(nFlags&MF_OWNERDRAW));
  1192. ModifyMenu(pos,nFlags,nID,mdata->GetString());
  1193. }
  1194. else{
  1195. ASSERT(nFlags&MF_SEPARATOR);
  1196. ModifyMenu(pos,nFlags,nID);
  1197. }
  1198. return(mdata);
  1199. };
  1200. BOOL BCMenu::LoadToolbars(const UINT *arID,int n)
  1201. {
  1202. ASSERT(arID);
  1203. BOOL returnflag=TRUE;
  1204. for(int i=0;i<n;++i){
  1205. if(!LoadToolbar(arID[i]))returnflag=FALSE;
  1206. }
  1207. return(returnflag);
  1208. }
  1209. BOOL BCMenu::LoadToolbar(UINT nToolBar)
  1210. {
  1211. UINT nID,nStyle;
  1212. BOOL returnflag=FALSE;
  1213. BCMenuToolBar bar;
  1214. int xoffset=-1,xset;
  1215. int iconx,icony;
  1216. CWnd* pWnd = AfxGetMainWnd();
  1217. if (pWnd == NULL)pWnd = CWnd::GetDesktopWindow();
  1218. bar.Create(pWnd);
  1219. if(bar.LoadToolBar(nToolBar)){
  1220. bar.GetIconSize(iconx,icony);
  1221. if(iconx>m_iconX)m_iconX=iconx;
  1222. if(icony>m_iconY)m_iconY=icony;
  1223. CImageList imglist;
  1224. imglist.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  1225. if(AddBitmapToImageList(&imglist,nToolBar)){
  1226. returnflag=TRUE;
  1227. for(int i=0;i<bar.GetCount();++i){
  1228. nID = bar.GetItemID(i);
  1229. if(nID && GetMenuState(nID, MF_BYCOMMAND)
  1230. !=0xFFFFFFFF){
  1231. xoffset=bar.CommandToIndex(nID);
  1232. if(xoffset>=0){
  1233. bar.GetButtonInfo(xoffset,nID,nStyle,xset);
  1234. if(xset>0)xoffset=xset;
  1235. }
  1236. ModifyODMenu(NULL,nID,&imglist,xoffset);
  1237. }
  1238. }
  1239. }
  1240. }
  1241. return(returnflag);
  1242. }
  1243. BOOL BCMenu::LoadFromToolBar(UINT nID,UINT nToolBar,int& xoffset)
  1244. {
  1245. int xset,offset;
  1246. UINT nStyle;
  1247. BOOL returnflag=FALSE;
  1248. CToolBar bar;
  1249. CWnd* pWnd = AfxGetMainWnd();
  1250. if (pWnd == NULL)pWnd = CWnd::GetDesktopWindow();
  1251. bar.Create(pWnd);
  1252. if(bar.LoadToolBar(nToolBar)){
  1253. offset=bar.CommandToIndex(nID);
  1254. if(offset>=0){
  1255. bar.GetButtonInfo(offset,nID,nStyle,xset);
  1256. if(xset>0)xoffset=xset;
  1257. returnflag=TRUE;
  1258. }
  1259. }
  1260. return(returnflag);
  1261. }
  1262. // O.S.
  1263. BCMenuData *BCMenu::FindMenuItem(UINT nID)
  1264. {
  1265. BCMenuData *pData = NULL;
  1266. int i;
  1267. for(i = 0; i <= m_MenuList.GetUpperBound(); i++){
  1268. if (m_MenuList[i]->nID == nID){
  1269. pData = m_MenuList[i];
  1270. break;
  1271. }
  1272. }
  1273. if (!pData){
  1274. int loc;
  1275. BCMenu *pMenu = FindMenuOption(nID, loc);
  1276. ASSERT(pMenu != this);
  1277. if (loc >= 0){
  1278. return pMenu->FindMenuItem(nID);
  1279. }
  1280. }
  1281. return pData;
  1282. }
  1283. BCMenu *BCMenu::FindAnotherMenuOption(int nId,int& nLoc,CArray<BCMenu*,BCMenu*>&bcsubs,
  1284. CArray<int,int&>&bclocs)
  1285. {
  1286. int i,numsubs,j;
  1287. BCMenu *psubmenu,*pgoodmenu;
  1288. BOOL foundflag;
  1289. for(i=0;i<(int)(GetMenuItemCount());++i){
  1290. #ifdef _CPPRTTI
  1291. psubmenu=dynamic_cast<BCMenu *>(GetSubMenu(i));
  1292. #else
  1293. psubmenu=(BCMenu *)GetSubMenu(i);
  1294. #endif
  1295. if(psubmenu){
  1296. pgoodmenu=psubmenu->FindAnotherMenuOption(nId,nLoc,bcsubs,bclocs);
  1297. if(pgoodmenu)return(pgoodmenu);
  1298. }
  1299. else if(nId==(int)GetMenuItemID(i)){
  1300. numsubs=bcsubs.GetSize();
  1301. foundflag=TRUE;
  1302. for(j=0;j<numsubs;++j){
  1303. if(bcsubs[j]==this&&bclocs[j]==i){
  1304. foundflag=FALSE;
  1305. break;
  1306. }
  1307. }
  1308. if(foundflag){
  1309. nLoc=i;
  1310. return(this);
  1311. }
  1312. }
  1313. }
  1314. nLoc = -1;
  1315. return(NULL);
  1316. }
  1317. BCMenu *BCMenu::FindMenuOption(int nId,int& nLoc)
  1318. {
  1319. int i;
  1320. BCMenu *psubmenu,*pgoodmenu;
  1321. for(i=0;i<(int)(GetMenuItemCount());++i){
  1322. #ifdef _CPPRTTI
  1323. psubmenu=dynamic_cast<BCMenu *>(GetSubMenu(i));
  1324. #else
  1325. psubmenu=(BCMenu *)GetSubMenu(i);
  1326. #endif
  1327. if(psubmenu){
  1328. pgoodmenu=psubmenu->FindMenuOption(nId,nLoc);
  1329. if(pgoodmenu)return(pgoodmenu);
  1330. }
  1331. else if(nId==(int)GetMenuItemID(i)){
  1332. nLoc=i;
  1333. return(this);
  1334. }
  1335. }
  1336. nLoc = -1;
  1337. return(NULL);
  1338. }
  1339. BCMenuData *BCMenu::FindMenuOption(wchar_t *lpstrText)
  1340. {
  1341. int i,j;
  1342. BCMenu *psubmenu;
  1343. BCMenuData *pmenulist;
  1344. for(i=0;i<(int)(GetMenuItemCount());++i){
  1345. #ifdef _CPPRTTI
  1346. psubmenu=dynamic_cast<BCMenu *>(GetSubMenu(i));
  1347. #else
  1348. psubmenu=(BCMenu *)GetSubMenu(i);
  1349. #endif
  1350. if(psubmenu){
  1351. pmenulist=psubmenu->FindMenuOption(lpstrText);
  1352. if(pmenulist)return(pmenulist);
  1353. }
  1354. else{
  1355. const wchar_t *szWide;//SK: we use const to prevent misuse of this Ptr
  1356. for(j=0;j<=m_MenuList.GetUpperBound();++j){
  1357. szWide = m_MenuList[j]->GetWideString ();
  1358. if(szWide && !wcscmp(lpstrText,szWide))//SK: modified for dynamic allocation
  1359. return(m_MenuList[j]);
  1360. }
  1361. }
  1362. }
  1363. return(NULL);
  1364. }
  1365. BOOL BCMenu::LoadMenu(int nResource)
  1366. {
  1367. return(BCMenu::LoadMenu(MAKEINTRESOURCE(nResource)));
  1368. };
  1369. BOOL BCMenu::LoadMenu(LPCTSTR lpszResourceName)
  1370. {
  1371. ASSERT_VALID(this);
  1372. ASSERT(lpszResourceName != NULL);
  1373. // Find the Menu Resource:
  1374. HINSTANCE hInst = AfxFindResourceHandle(lpszResourceName,RT_MENU);
  1375. HRSRC hRsrc = ::FindResource(hInst,lpszResourceName,RT_MENU);
  1376. if (hRsrc == NULL){
  1377. hInst = NULL;
  1378. hRsrc = ::FindResource(hInst,lpszResourceName,RT_MENU);
  1379. }
  1380. if(hRsrc == NULL)return FALSE;
  1381. // Load the Menu Resource:
  1382. HGLOBAL hGlobal = LoadResource(hInst, hRsrc);
  1383. if(hGlobal == NULL)return FALSE;
  1384. // first destroy the menu if we're trying to loadmenu again
  1385. DestroyMenu();
  1386. // Attempt to create us as a menu...
  1387. if(!CMenu::CreateMenu())return FALSE;
  1388. // Get Item template Header, and calculate offset of MENUITEMTEMPLATES
  1389. MENUITEMTEMPLATEHEADER *pTpHdr=
  1390. (MENUITEMTEMPLATEHEADER*)LockResource(hGlobal);
  1391. BYTE* pTp=(BYTE*)pTpHdr +
  1392. (sizeof(MENUITEMTEMPLATEHEADER) + pTpHdr->offset);
  1393. // Variables needed during processing of Menu Item Templates:
  1394. int j=0;
  1395. WORD dwFlags = 0; // Flags of the Menu Item
  1396. WORD dwID = 0; // ID of the Menu Item
  1397. UINT uFlags; // Actual Flags.
  1398. wchar_t *szCaption=NULL;
  1399. int nLen = 0; // Length of caption
  1400. CTypedPtrArray<CPtrArray, BCMenu*> m_Stack; // Popup menu stack
  1401. CArray<BOOL,BOOL> m_StackEnd; // Popup menu stack
  1402. m_Stack.Add(this); // Add it to this...
  1403. m_StackEnd.Add(FALSE);
  1404. do{
  1405. // Obtain Flags and (if necessary), the ID...
  1406. memcpy(&dwFlags, pTp, sizeof(WORD));pTp+=sizeof(WORD);// Obtain Flags
  1407. if(!(dwFlags & MF_POPUP)){
  1408. memcpy(&dwID, pTp, sizeof(WORD)); // Obtain ID
  1409. pTp+=sizeof(WORD);
  1410. }
  1411. else dwID = 0;
  1412. uFlags = (UINT)dwFlags; // Remove MF_END from the flags that will
  1413. if(uFlags & MF_END) // be passed to the Append(OD)Menu functions.
  1414. uFlags -= MF_END;
  1415. // Obtain Caption (and length)
  1416. nLen = 0;
  1417. szCaption=new wchar_t[wcslen((wchar_t *)pTp)+1];
  1418. wcscpy_s(szCaption, wcslen((wchar_t*)pTp) + 1, (wchar_t *)pTp);
  1419. pTp=&pTp[(wcslen((wchar_t *)pTp)+1)*sizeof(wchar_t)];//modified SK
  1420. // Handle popup menus first....
  1421. //WideCharToMultiByte
  1422. if(dwFlags & MF_POPUP){
  1423. if(dwFlags & MF_END)m_StackEnd.SetAt(m_Stack.GetUpperBound(),TRUE);
  1424. BCMenu* pSubMenu = new BCMenu;
  1425. pSubMenu->m_unselectcheck=m_unselectcheck;
  1426. pSubMenu->m_selectcheck=m_selectcheck;
  1427. pSubMenu->checkmaps=checkmaps;
  1428. pSubMenu->checkmapsshare=TRUE;
  1429. pSubMenu->CreatePopupMenu();
  1430. // Append it to the top of the stack:
  1431. m_Stack[m_Stack.GetUpperBound()]->AppendODMenuW(szCaption,uFlags,
  1432. (UINT)pSubMenu->m_hMenu, -1);
  1433. m_Stack.Add(pSubMenu);
  1434. m_StackEnd.Add(FALSE);
  1435. }
  1436. else {
  1437. m_Stack[m_Stack.GetUpperBound()]->AppendODMenuW(szCaption, uFlags,
  1438. dwID, -1);
  1439. if(dwFlags & MF_END)m_StackEnd.SetAt(m_Stack.GetUpperBound(),TRUE);
  1440. j = m_Stack.GetUpperBound();
  1441. while(j>=0 && m_StackEnd.GetAt(j)){
  1442. m_Stack[m_Stack.GetUpperBound()]->InsertSpaces();
  1443. m_Stack.RemoveAt(j);
  1444. m_StackEnd.RemoveAt(j);
  1445. --j;
  1446. }
  1447. }
  1448. delete[] szCaption;
  1449. }while(m_Stack.GetUpperBound() != -1);
  1450. for(int i=0;i<(int)GetMenuItemCount();++i){
  1451. CString str=m_MenuList[i]->GetString();
  1452. if(GetSubMenu(i)){
  1453. m_MenuList[i]->nFlags=MF_POPUP|MF_BYPOSITION;
  1454. ModifyMenu(i,MF_POPUP|MF_BYPOSITION,
  1455. (UINT)GetSubMenu(i)->m_hMenu,str);
  1456. }
  1457. else{
  1458. m_MenuList[i]->nFlags=MF_STRING|MF_BYPOSITION;
  1459. ModifyMenu(i,MF_STRING|MF_BYPOSITION,m_MenuList[i]->nID,str);
  1460. }
  1461. }
  1462. m_loadmenu=TRUE;
  1463. return(TRUE);
  1464. }
  1465. int BCMenu::GetMenuStart(void)
  1466. {
  1467. if(!m_loadmenu)return(0);
  1468. CString name,str;
  1469. int menuloc=-1,listloc=-1,menustart=0,i=0,j=0;
  1470. int nummenulist=m_MenuList.GetSize();
  1471. int nummenu=(int)GetMenuItemCount();
  1472. while(i<nummenu&&menuloc==-1){
  1473. GetMenuString (i, name, MF_BYPOSITION);
  1474. if(name.GetLength()>0){
  1475. for(j=0;j<nummenulist;++j){
  1476. str=m_MenuList[j]->GetString();
  1477. if(name==str){
  1478. menuloc=i;
  1479. listloc=j;
  1480. break;
  1481. }
  1482. }
  1483. }
  1484. ++i;
  1485. }
  1486. if(menuloc>=0&&listloc>=0&&menuloc>=listloc)menustart=menuloc-listloc;
  1487. return(menustart);
  1488. }
  1489. void BCMenu::RemoveTopLevelOwnerDraw(void)
  1490. {
  1491. CString str;
  1492. int i=0,j=0;
  1493. int nummenulist=m_MenuList.GetSize(),menustart;
  1494. menustart=GetMenuStart();
  1495. for(i=menustart,j=0;i<(int)GetMenuItemCount();++i,++j){
  1496. if(j<nummenulist){
  1497. str=m_MenuList[j]->GetString();
  1498. if(GetSubMenu(i)){
  1499. m_MenuList[j]->nFlags=MF_POPUP|MF_BYPOSITION;
  1500. ModifyMenu(i,MF_POPUP|MF_BYPOSITION,
  1501. (UINT)GetSubMenu(i)->m_hMenu,str);
  1502. }
  1503. }
  1504. }
  1505. }
  1506. void BCMenu::InsertSpaces(void)
  1507. {
  1508. int i,j,numitems,maxlength;
  1509. CString string,newstring;
  1510. CSize t;
  1511. CFont m_fontMenu;
  1512. LOGFONT m_lf;
  1513. ZeroMemory ((PVOID) &m_lf,sizeof (LOGFONT));
  1514. NONCLIENTMETRICS nm;
  1515. nm.cbSize = sizeof (NONCLIENTMETRICS);
  1516. VERIFY (SystemParametersInfo (SPI_GETNONCLIENTMETRICS,nm.cbSize,&nm,0));
  1517. m_lf = nm.lfMenuFont;
  1518. m_fontMenu.CreateFontIndirect (&m_lf);
  1519. CWnd *pWnd = AfxGetMainWnd();
  1520. if (pWnd == NULL)pWnd = CWnd::GetDesktopWindow();
  1521. CDC *pDC = pWnd->GetDC();
  1522. CFont* pFont = pDC->SelectObject (&m_fontMenu);
  1523. numitems=GetMenuItemCount();
  1524. maxlength = -1;
  1525. for(i=0;i<numitems;++i){
  1526. string=m_MenuList[i]->GetString();
  1527. j=string.Find((char)9);
  1528. newstring.Empty();
  1529. if(j!=-1)newstring=string.Left(j);
  1530. else newstring=string;
  1531. newstring+=_T(" ");//SK: modified for Unicode correctness.
  1532. LPCTSTR lpstrText = (LPCTSTR)newstring;
  1533. t=pDC->GetTextExtent(lpstrText,_tcslen(lpstrText));
  1534. if(t.cx>maxlength)maxlength = t.cx;
  1535. }
  1536. for(i=0;i<numitems;++i){
  1537. string=m_MenuList[i]->GetString();
  1538. j=string.Find((char)9);
  1539. if(j!=-1){
  1540. newstring.Empty();
  1541. newstring=string.Left(j);
  1542. LPCTSTR lpstrText = (LPCTSTR)(newstring);
  1543. t=pDC->GetTextExtent(lpstrText,_tcslen(lpstrText));
  1544. while(t.cx<maxlength){
  1545. newstring+=_T(' ');//SK: modified for Unicode correctness
  1546. LPCTSTR lpstrText = (LPCTSTR)(newstring);
  1547. t=pDC->GetTextExtent(lpstrText,_tcslen(lpstrText));
  1548. }
  1549. newstring+=string.Mid(j);
  1550. #ifdef UNICODE
  1551. m_MenuList[i]->SetWideString(newstring);//SK: modified for dynamic allocation
  1552. #else
  1553. m_MenuList[i]->SetAnsiString(newstring);
  1554. #endif
  1555. }
  1556. }
  1557. pDC->SelectObject (pFont); // Select old font in
  1558. pWnd->ReleaseDC(pDC); // Release the DC
  1559. m_fontMenu.DeleteObject();
  1560. }
  1561. void BCMenu::LoadCheckmarkBitmap(int unselect, int select)
  1562. {
  1563. if(unselect>0 && select>0){
  1564. m_selectcheck=select;
  1565. m_unselectcheck=unselect;
  1566. if(checkmaps)checkmaps->DeleteImageList();
  1567. else checkmaps=new(CImageList);
  1568. checkmaps->Create(m_iconX,m_iconY,ILC_MASK,2,1);
  1569. BOOL flag1=AddBitmapToImageList(checkmaps,unselect);
  1570. BOOL flag2=AddBitmapToImageList(checkmaps,select);
  1571. if(!flag1||!flag2){
  1572. checkmaps->DeleteImageList();
  1573. delete checkmaps;
  1574. checkmaps=NULL;
  1575. }
  1576. }
  1577. }
  1578. //--------------------------------------------------------------------------
  1579. //[18.06.99 rj]
  1580. BOOL BCMenu::GetMenuText(UINT id, CString& string, UINT nFlags/*= MF_BYPOSITION*/)
  1581. {
  1582. BOOL returnflag=FALSE;
  1583. if(MF_BYPOSITION&nFlags){
  1584. UINT numMenuItems = m_MenuList.GetUpperBound();
  1585. if(id<=numMenuItems){
  1586. string=m_MenuList[id]->GetString();
  1587. returnflag=TRUE;
  1588. }
  1589. }
  1590. else{
  1591. int uiLoc;
  1592. BCMenu* pMenu = FindMenuOption(id,uiLoc);
  1593. if(NULL!=pMenu) returnflag = pMenu->GetMenuText(uiLoc,string);
  1594. }
  1595. return(returnflag);
  1596. }
  1597. void BCMenu::DrawRadioDot(CDC *pDC,int x,int y,COLORREF color)
  1598. {
  1599. CRect rcDot(x,y,x+6,y+6);
  1600. CBrush brush;
  1601. CPen pen;
  1602. brush.CreateSolidBrush(color);
  1603. pen.CreatePen(PS_SOLID,0,color);
  1604. CBrush *pOldBrush=pDC->SelectObject(&brush);
  1605. CPen *pOldPen=pDC->SelectObject(&pen);
  1606. pDC->Ellipse(&rcDot);
  1607. pDC->SelectObject(pOldBrush);
  1608. pDC->SelectObject(pOldPen);
  1609. pen.DeleteObject();
  1610. brush.DeleteObject();
  1611. }
  1612. void BCMenu::DrawCheckMark(CDC* pDC,int x,int y,COLORREF color)
  1613. {
  1614. CPen m_penBack;
  1615. m_penBack.CreatePen (PS_SOLID,0,color);
  1616. CPen *pOldPen = pDC->SelectObject (&m_penBack);
  1617. pDC->MoveTo(x,y+2);
  1618. pDC->LineTo(x,y+5);
  1619. pDC->MoveTo(x+1,y+3);
  1620. pDC->LineTo(x+1,y+6);
  1621. pDC->MoveTo(x+2,y+4);
  1622. pDC->LineTo(x+2,y+7);
  1623. pDC->MoveTo(x+3,y+3);
  1624. pDC->LineTo(x+3,y+6);
  1625. pDC->MoveTo(x+4,y+2);
  1626. pDC->LineTo(x+4,y+5);
  1627. pDC->MoveTo(x+5,y+1);
  1628. pDC->LineTo(x+5,y+4);
  1629. pDC->MoveTo(x+6,y);
  1630. pDC->LineTo(x+6,y+3);
  1631. pDC->SelectObject (pOldPen);
  1632. m_penBack.DeleteObject();
  1633. }
  1634. BCMenuData *BCMenu::FindMenuList(UINT nID)
  1635. {
  1636. for(int i=0;i<=m_MenuList.GetUpperBound();++i){
  1637. if(m_MenuList[i]->nID==nID && !m_MenuList[i]->syncflag){
  1638. m_MenuList[i]->syncflag=1;
  1639. return(m_MenuList[i]);
  1640. }
  1641. }
  1642. return(NULL);
  1643. }
  1644. void BCMenu::InitializeMenuList(int value)
  1645. {
  1646. for(int i=0;i<=m_MenuList.GetUpperBound();++i)
  1647. m_MenuList[i]->syncflag=value;
  1648. }
  1649. void BCMenu::DeleteMenuList(void)
  1650. {
  1651. for(int i=0;i<=m_MenuList.GetUpperBound();++i){
  1652. if(!m_MenuList[i]->syncflag){
  1653. delete m_MenuList[i];
  1654. }
  1655. }
  1656. }
  1657. void BCMenu::SynchronizeMenu(void)
  1658. {
  1659. CTypedPtrArray<CPtrArray, BCMenuData*> temp;
  1660. BCMenuData *mdata;
  1661. CString string;
  1662. UINT submenu,nID=0,state,j;
  1663. InitializeMenuList(0);
  1664. for(j=0;j<GetMenuItemCount();++j){
  1665. mdata=NULL;
  1666. state=GetMenuState(j,MF_BYPOSITION);
  1667. if(state&MF_POPUP){
  1668. submenu=(UINT)GetSubMenu(j)->m_hMenu;
  1669. mdata=FindMenuList(submenu);
  1670. GetMenuString(j,string,MF_BYPOSITION);
  1671. if(!mdata)mdata=NewODMenu(j,
  1672. (state&0xFF)|MF_BYPOSITION|MF_POPUP|MF_OWNERDRAW,submenu,string);
  1673. else if(string.GetLength()>0)
  1674. #ifdef UNICODE
  1675. mdata->SetWideString(string); //SK: modified for dynamic allocation
  1676. #else
  1677. mdata->SetAnsiString(string);
  1678. #endif
  1679. }
  1680. else if(state&MF_SEPARATOR){
  1681. mdata=FindMenuList(0);
  1682. if(!mdata)mdata=NewODMenu(j,
  1683. state|MF_BYPOSITION|MF_SEPARATOR|MF_OWNERDRAW,0,_T(""));//SK: modified for Unicode correctness
  1684. else ModifyMenu(j,mdata->nFlags,nID,(LPCTSTR)mdata);
  1685. }
  1686. else{
  1687. nID=GetMenuItemID(j);
  1688. mdata=FindMenuList(nID);
  1689. GetMenuString(j,string,MF_BYPOSITION);
  1690. if(!mdata)mdata=NewODMenu(j,state|MF_BYPOSITION|MF_OWNERDRAW,
  1691. nID,string);
  1692. else{
  1693. mdata->nFlags=state|MF_BYPOSITION|MF_OWNERDRAW;
  1694. if(string.GetLength()>0)
  1695. #ifdef UNICODE
  1696. mdata->SetWideString(string);//SK: modified for dynamic allocation
  1697. #else
  1698. mdata->SetAnsiString(string);
  1699. #endif
  1700. ModifyMenu(j,mdata->nFlags,nID,(LPCTSTR)mdata);
  1701. }
  1702. }
  1703. if(mdata)temp.Add(mdata);
  1704. }
  1705. DeleteMenuList();
  1706. m_MenuList.RemoveAll();
  1707. m_MenuList.Append(temp);
  1708. temp.RemoveAll();
  1709. }
  1710. void BCMenu::UpdateMenu(CMenu *pmenu)
  1711. {
  1712. #ifdef _CPPRTTI
  1713. BCMenu *psubmenu = dynamic_cast<BCMenu *>(pmenu);
  1714. #else
  1715. BCMenu *psubmenu = (BCMenu *)pmenu;
  1716. #endif
  1717. if(psubmenu)psubmenu->SynchronizeMenu();
  1718. }
  1719. LRESULT BCMenu::FindKeyboardShortcut(UINT nChar, UINT nFlags,
  1720. CMenu *pMenu)
  1721. {
  1722. #ifdef _CPPRTTI
  1723. BCMenu *pBCMenu = dynamic_cast<BCMenu *>(pMenu);
  1724. #else
  1725. BCMenu *pBCMenu = (BCMenu *)pMenu;
  1726. #endif
  1727. if(pBCMenu && nFlags&MF_POPUP){
  1728. CString key(_T('&'),2);//SK: modified for Unicode correctness
  1729. key.SetAt(1,(TCHAR)nChar);
  1730. key.MakeLower();
  1731. CString menutext;
  1732. int menusize = (int)pBCMenu->GetMenuItemCount();
  1733. if(menusize!=(pBCMenu->m_MenuList.GetUpperBound()+1))
  1734. pBCMenu->SynchronizeMenu();
  1735. for(int i=0;i<menusize;++i){
  1736. if(pBCMenu->GetMenuText(i,menutext)){
  1737. menutext.MakeLower();
  1738. if(menutext.Find(key)>=0)return(MAKELRESULT(i,2));
  1739. }
  1740. }
  1741. }
  1742. return(0);
  1743. }
  1744. void BCMenu::DitherBlt (HDC hdcDest, int nXDest, int nYDest, int nWidth,
  1745. int nHeight, HBITMAP hbm, int nXSrc, int nYSrc,
  1746. COLORREF bgcolor)
  1747. {
  1748. ASSERT(hdcDest && hbm);
  1749. ASSERT(nWidth > 0 && nHeight > 0);
  1750. // Create a generic DC for all BitBlts
  1751. HDC hDC = CreateCompatibleDC(hdcDest);
  1752. ASSERT(hDC);
  1753. if (hDC)
  1754. {
  1755. // Create a DC for the monochrome DIB section
  1756. HDC bwDC = CreateCompatibleDC(hDC);
  1757. ASSERT(bwDC);
  1758. if (bwDC)
  1759. {
  1760. // Create the monochrome DIB section with a black and white palette
  1761. struct {
  1762. BITMAPINFOHEADER bmiHeader;
  1763. RGBQUAD bmiColors[2];
  1764. } RGBBWBITMAPINFO = {
  1765. { // a BITMAPINFOHEADER
  1766. sizeof(BITMAPINFOHEADER), // biSize
  1767. nWidth, // biWidth;
  1768. nHeight, // biHeight;
  1769. 1, // biPlanes;
  1770. 1, // biBitCount
  1771. BI_RGB, // biCompression;
  1772. 0, // biSizeImage;
  1773. 0, // biXPelsPerMeter;
  1774. 0, // biYPelsPerMeter;
  1775. 0, // biClrUsed;
  1776. 0 // biClrImportant;
  1777. },
  1778. {
  1779. { 0x00, 0x00, 0x00, 0x00 }, { 0xFF, 0xFF, 0xFF, 0x00 }
  1780. }
  1781. };
  1782. VOID *pbitsBW;
  1783. HBITMAP hbmBW = CreateDIBSection(bwDC,
  1784. (LPBITMAPINFO)&RGBBWBITMAPINFO, DIB_RGB_COLORS, &pbitsBW, NULL, 0);
  1785. ASSERT(hbmBW);
  1786. if (hbmBW)
  1787. {
  1788. // Attach the monochrome DIB section and the bitmap to the DCs
  1789. HBITMAP olddib = (HBITMAP)SelectObject(bwDC, hbmBW);
  1790. HBITMAP hdcolddib = (HBITMAP)SelectObject(hDC, hbm);
  1791. // BitBlt the bitmap into the monochrome DIB section
  1792. BitBlt(bwDC, 0, 0, nWidth, nHeight, hDC, nXSrc, nYSrc, SRCCOPY);
  1793. // Paint the destination rectangle in gray
  1794. FillRect(hdcDest, CRect(nXDest, nYDest, nXDest + nWidth, nYDest +
  1795. nHeight), GetSysColorBrush(bgcolor));
  1796. //SK: looks better on the old shell
  1797. // BitBlt the black bits in the monochrome bitmap into COLOR_3DHILIGHT
  1798. // bits in the destination DC
  1799. // The magic ROP comes from the Charles Petzold's book
  1800. HBRUSH hb = CreateSolidBrush(GetSysColor(COLOR_3DHILIGHT));
  1801. HBRUSH oldBrush = (HBRUSH)SelectObject(hdcDest, hb);
  1802. BitBlt(hdcDest,nXDest+1,nYDest+1,nWidth,nHeight,bwDC,0,0,0xB8074A);
  1803. // BitBlt the black bits in the monochrome bitmap into COLOR_3DSHADOW
  1804. // bits in the destination DC
  1805. hb = CreateSolidBrush(GetSysColor(COLOR_3DSHADOW));
  1806. DeleteObject(SelectObject(hdcDest, hb));
  1807. BitBlt(hdcDest, nXDest, nYDest, nWidth, nHeight,bwDC,0,0,0xB8074A);
  1808. DeleteObject(SelectObject(hdcDest, oldBrush));
  1809. VERIFY(DeleteObject(SelectObject(bwDC, olddib)));
  1810. SelectObject(hDC, hdcolddib);
  1811. }
  1812. VERIFY(DeleteDC(bwDC));
  1813. }
  1814. VERIFY(DeleteDC(hDC));
  1815. }
  1816. }
  1817. void BCMenu::GetFadedBitmap(CBitmap &bmp)
  1818. {
  1819. CDC ddc;
  1820. COLORREF bgcol,col;
  1821. BITMAP BitMap;
  1822. bmp.GetBitmap(&BitMap);
  1823. ddc.CreateCompatibleDC(NULL);
  1824. CBitmap * pddcOldBmp = ddc.SelectObject(&bmp);
  1825. // use this to get the background color, takes into account color shifting
  1826. CDC ddc2;
  1827. CBrush brush;
  1828. CBitmap bmp2;
  1829. ddc2.CreateCompatibleDC(NULL);
  1830. bmp2.CreateCompatibleBitmap(&ddc,BitMap.bmWidth,BitMap.bmHeight);
  1831. col=GetSysColor(COLOR_3DFACE);
  1832. brush.CreateSolidBrush(col);
  1833. CBitmap * pddcOldBmp2 = ddc2.SelectObject(&bmp2);
  1834. CRect rect(0,0,BitMap.bmWidth,BitMap.bmHeight);
  1835. ddc2.FillRect(rect,&brush);
  1836. bgcol=ddc2.GetPixel(1,1);
  1837. brush.DeleteObject();
  1838. ddc2.SelectObject(pddcOldBmp2);
  1839. for(int i=0;i<BitMap.bmWidth;++i){
  1840. for(int j=0;j<BitMap.bmHeight;++j){
  1841. col=ddc.GetPixel(i,j);
  1842. if(col!=bgcol)ddc.SetPixel(i,j,LightenColor(col,0.3));
  1843. }
  1844. }
  1845. ddc.SelectObject(pddcOldBmp);
  1846. }
  1847. void BCMenu::GetShadowBitmap(CBitmap &bmp)
  1848. {
  1849. CDC ddc;
  1850. COLORREF bgcol,col,shadowcol=GetSysColor(COLOR_BTNSHADOW);
  1851. BITMAP BitMap;
  1852. if(!IsWinXPLuna())shadowcol=LightenColor(shadowcol,0.49);
  1853. bmp.GetBitmap(&BitMap);
  1854. ddc.CreateCompatibleDC(NULL);
  1855. CBitmap * pddcOldBmp = ddc.SelectObject(&bmp);
  1856. // use this to get the background color, takes into account color shifting
  1857. CDC ddc2;
  1858. CBrush brush;
  1859. CBitmap bmp2;
  1860. ddc2.CreateCompatibleDC(NULL);
  1861. bmp2.CreateCompatibleBitmap(&ddc,BitMap.bmWidth,BitMap.bmHeight);
  1862. col=GetSysColor(COLOR_3DFACE);
  1863. brush.CreateSolidBrush(col);
  1864. CBitmap * pddcOldBmp2 = ddc2.SelectObject(&bmp2);
  1865. CRect rect(0,0,BitMap.bmWidth,BitMap.bmHeight);
  1866. ddc2.FillRect(rect,&brush);
  1867. bgcol=ddc2.GetPixel(1,1);
  1868. brush.DeleteObject();
  1869. ddc2.SelectObject(pddcOldBmp2);
  1870. for(int i=0;i<BitMap.bmWidth;++i){
  1871. for(int j=0;j<BitMap.bmHeight;++j){
  1872. col=ddc.GetPixel(i,j);
  1873. if(col!=bgcol)ddc.SetPixel(i,j,shadowcol);
  1874. }
  1875. }
  1876. ddc.SelectObject(pddcOldBmp);
  1877. }
  1878. BOOL BCMenu::AddBitmapToImageList(CImageList *bmplist,UINT nResourceID)
  1879. {
  1880. BOOL bReturn=FALSE;
  1881. // O.S.
  1882. if (m_bDynIcons){
  1883. bmplist->Add((HICON)nResourceID);
  1884. bReturn=TRUE;
  1885. }
  1886. else{
  1887. HBITMAP hbmp=LoadSysColorBitmap(nResourceID);
  1888. if(hbmp){
  1889. CBitmap bmp;
  1890. bmp.Attach(hbmp);
  1891. if(m_bitmapBackgroundFlag){
  1892. if(bmplist->Add(&bmp,m_bitmapBackground)>=0)bReturn=TRUE;
  1893. }
  1894. else{
  1895. if(bmplist->Add(&bmp,GetSysColor(COLOR_3DFACE))>=0)bReturn=TRUE;
  1896. }
  1897. bmp.Detach();
  1898. DeleteObject(hbmp);
  1899. }
  1900. else{
  1901. CBitmap mybmp;
  1902. if(mybmp.LoadBitmap(nResourceID)){
  1903. if(m_bitmapBackgroundFlag){
  1904. if(bmplist->Add(&mybmp,m_bitmapBackground)>=0)bReturn=TRUE;
  1905. }
  1906. else{
  1907. if(bmplist->Add(&mybmp,GetSysColor(COLOR_3DFACE))>=0)bReturn=TRUE;
  1908. }
  1909. }
  1910. }
  1911. }
  1912. if(bReturn&&IsLunaMenuStyle()&&xp_draw_3D_bitmaps){
  1913. CWnd *pWnd = AfxGetMainWnd(); // Get main window
  1914. if (pWnd == NULL) pWnd = CWnd::GetDesktopWindow();
  1915. CDC *pDC = pWnd->GetDC(); // Get device context
  1916. CBitmap bmp,bmp2;
  1917. GetBitmapFromImageList(pDC,bmplist,0,bmp);
  1918. GetFadedBitmap(bmp);
  1919. bmplist->Add(&bmp,GetSysColor(COLOR_3DFACE));
  1920. GetBitmapFromImageList(pDC,bmplist,0,bmp2);
  1921. GetShadowBitmap(bmp2);
  1922. bmplist->Add(&bmp2,GetSysColor(COLOR_3DFACE));
  1923. pWnd->ReleaseDC(pDC); // Release the DC
  1924. }
  1925. return(bReturn);
  1926. }
  1927. void BCMenu::SetBitmapBackground(COLORREF color)
  1928. {
  1929. m_bitmapBackground=color;
  1930. m_bitmapBackgroundFlag=TRUE;
  1931. }
  1932. void BCMenu::UnSetBitmapBackground(void)
  1933. {
  1934. m_bitmapBackgroundFlag=FALSE;
  1935. }
  1936. // Given a toolbar, append all the options from it to this menu
  1937. // Passed a ptr to the toolbar object and the toolbar ID
  1938. // Author : Robert Edward Caldecott
  1939. void BCMenu::AddFromToolBar(CToolBar* pToolBar, int nResourceID)
  1940. {
  1941. for (int i = 0; i < pToolBar->GetCount(); i++) {
  1942. UINT nID = pToolBar->GetItemID(i);
  1943. // See if this toolbar option
  1944. // appears as a command on this
  1945. // menu or is a separator
  1946. if (nID == 0 || GetMenuState(nID, MF_BYCOMMAND) == 0xFFFFFFFF)
  1947. continue; // Item doesn't exist
  1948. UINT nStyle;
  1949. int nImage;
  1950. // Get the toolbar button info
  1951. pToolBar->GetButtonInfo(i, nID, nStyle, nImage);
  1952. // OK, we have the command ID of the toolbar
  1953. // option, and the tollbar bitmap offset
  1954. int nLoc;
  1955. BCMenuData* pData;
  1956. BCMenu *pSubMenu = FindMenuOption(nID, nLoc);
  1957. if (pSubMenu && nLoc >= 0)pData = pSubMenu->m_MenuList[nLoc];
  1958. else {
  1959. // Create a new BCMenuData structure
  1960. pData = new BCMenuData;
  1961. m_MenuList.Add(pData);
  1962. }
  1963. // Set some default structure members
  1964. pData->menuIconNormal = nResourceID;
  1965. pData->nID = nID;
  1966. pData->nFlags = MF_BYCOMMAND | MF_OWNERDRAW;
  1967. pData->xoffset = nImage;
  1968. if (pData->bitmap)pData->bitmap->DeleteImageList();
  1969. else pData->bitmap = new CImageList;
  1970. pData->bitmap->Create(m_iconX, m_iconY,ILC_COLORDDB|ILC_MASK, 1, 1);
  1971. if(!AddBitmapToImageList(pData->bitmap, nResourceID)){
  1972. pData->bitmap->DeleteImageList();
  1973. delete pData->bitmap;
  1974. pData->bitmap=NULL;
  1975. pData->menuIconNormal = -1;
  1976. pData->xoffset = -1;
  1977. }
  1978. // Modify our menu
  1979. ModifyMenu(nID,pData->nFlags,nID,(LPCTSTR)pData);
  1980. }
  1981. }
  1982. BOOL BCMenu::Draw3DCheckmark(CDC *dc, const CRect& rc,
  1983. BOOL bSelected, HBITMAP hbmCheck)
  1984. {
  1985. CRect rcDest = rc;
  1986. CBrush brush;
  1987. COLORREF col;
  1988. if(IsWinXPLuna())col=GetSysColor(COLOR_3DFACE);
  1989. else col=GetSysColor(COLOR_MENU);
  1990. if(!bSelected)col = LightenColor(col,0.6);
  1991. brush.CreateSolidBrush(col);
  1992. dc->FillRect(rcDest,&brush);
  1993. brush.DeleteObject();
  1994. if (IsNewShell()) //SK: looks better on the old shell
  1995. dc->DrawEdge(&rcDest, BDR_SUNKENOUTER, BF_RECT);
  1996. if (!hbmCheck)DrawCheckMark(dc,rc.left+4,rc.top+4,GetSysColor(COLOR_MENUTEXT));
  1997. else DrawRadioDot(dc,rc.left+5,rc.top+4,GetSysColor(COLOR_MENUTEXT));
  1998. return TRUE;
  1999. }
  2000. BOOL BCMenu::DrawXPCheckmark(CDC *dc, const CRect& rc, HBITMAP hbmCheck,COLORREF &colorout)
  2001. {
  2002. CBrush brushin;
  2003. brushin.CreateSolidBrush(LightenColor(colorout,0.85));
  2004. dc->FillRect(rc,&brushin);
  2005. brushin.DeleteObject();
  2006. dc->Draw3dRect (rc,colorout,colorout);
  2007. if (!hbmCheck)DrawCheckMark(dc,rc.left+5,rc.top+4,GetSysColor(COLOR_MENUTEXT));
  2008. else DrawRadioDot(dc,rc.left+5,rc.top+4,GetSysColor(COLOR_MENUTEXT));
  2009. return TRUE;
  2010. }
  2011. void BCMenu::DitherBlt2(CDC *drawdc, int nXDest, int nYDest, int nWidth,
  2012. int nHeight, CBitmap &bmp, int nXSrc, int nYSrc,
  2013. COLORREF bgcolor)
  2014. {
  2015. // create a monochrome memory DC
  2016. CDC ddc;
  2017. ddc.CreateCompatibleDC(0);
  2018. CBitmap bwbmp;
  2019. bwbmp.CreateCompatibleBitmap(&ddc, nWidth, nHeight);
  2020. CBitmap * pddcOldBmp = ddc.SelectObject(&bwbmp);
  2021. CDC dc;
  2022. dc.CreateCompatibleDC(0);
  2023. CBitmap * pdcOldBmp = dc.SelectObject(&bmp);
  2024. // build a mask
  2025. ddc.PatBlt(0, 0, nWidth, nHeight, WHITENESS);
  2026. dc.SetBkColor(GetSysColor(COLOR_BTNFACE));
  2027. ddc.BitBlt(0, 0, nWidth, nHeight, &dc, nXSrc,nYSrc, SRCCOPY);
  2028. dc.SetBkColor(GetSysColor(COLOR_BTNHILIGHT));
  2029. ddc.BitBlt(0, 0, nWidth, nHeight, &dc, nXSrc,nYSrc, SRCPAINT);
  2030. // Copy the image from the toolbar into the memory DC
  2031. // and draw it (grayed) back into the toolbar.
  2032. dc.FillSolidRect(0,0, nWidth, nHeight, bgcolor);
  2033. //SK: Looks better on the old shell
  2034. dc.SetBkColor(RGB(0, 0, 0));
  2035. dc.SetTextColor(RGB(255, 255, 255));
  2036. CBrush brShadow, brHilight;
  2037. brHilight.CreateSolidBrush(GetSysColor(COLOR_BTNHILIGHT));
  2038. brShadow.CreateSolidBrush(GetSysColor(COLOR_BTNSHADOW));
  2039. CBrush * pOldBrush = dc.SelectObject(&brHilight);
  2040. dc.BitBlt(0,0, nWidth, nHeight, &ddc, 0, 0, 0x00E20746L);
  2041. drawdc->BitBlt(nXDest+1,nYDest+1,nWidth, nHeight, &dc,0,0,SRCCOPY);
  2042. dc.BitBlt(1,1, nWidth, nHeight, &ddc, 0, 0, 0x00E20746L);
  2043. dc.SelectObject(&brShadow);
  2044. dc.BitBlt(0,0, nWidth, nHeight, &ddc, 0, 0, 0x00E20746L);
  2045. drawdc->BitBlt(nXDest,nYDest,nWidth, nHeight, &dc,0,0,SRCCOPY);
  2046. // reset DCs
  2047. ddc.SelectObject(pddcOldBmp);
  2048. ddc.DeleteDC();
  2049. dc.SelectObject(pOldBrush);
  2050. dc.SelectObject(pdcOldBmp);
  2051. dc.DeleteDC();
  2052. brShadow.DeleteObject();
  2053. brHilight.DeleteObject();
  2054. bwbmp.DeleteObject();
  2055. }
  2056. void BCMenu::SetDisableOldStyle(void)
  2057. {
  2058. disable_old_style=TRUE;
  2059. }
  2060. void BCMenu::UnSetDisableOldStyle(void)
  2061. {
  2062. disable_old_style=FALSE;
  2063. }
  2064. BOOL BCMenu::GetDisableOldStyle(void)
  2065. {
  2066. return(disable_old_style);
  2067. }
  2068. WORD BCMenu::NumBitmapColors(LPBITMAPINFOHEADER lpBitmap)
  2069. {
  2070. if ( lpBitmap->biClrUsed != 0)
  2071. return (WORD)lpBitmap->biClrUsed;
  2072. switch (lpBitmap->biBitCount){
  2073. case 1:
  2074. return 2;
  2075. case 4:
  2076. return 16;
  2077. case 8:
  2078. return 256;
  2079. default:
  2080. return 0;
  2081. }
  2082. return 0;
  2083. }
  2084. HBITMAP BCMenu::LoadSysColorBitmap(int nResourceId)
  2085. {
  2086. HINSTANCE hInst =
  2087. AfxFindResourceHandle(MAKEINTRESOURCE(nResourceId),RT_BITMAP);
  2088. HRSRC hRsrc =
  2089. ::FindResource(hInst,MAKEINTRESOURCE(nResourceId),RT_BITMAP);
  2090. if (hRsrc == NULL){
  2091. hInst = NULL;
  2092. hRsrc = ::FindResource(hInst,MAKEINTRESOURCE(nResourceId),RT_BITMAP);
  2093. }
  2094. if (hRsrc == NULL)return NULL;
  2095. // determine how many colors in the bitmap
  2096. HGLOBAL hglb;
  2097. if ((hglb = LoadResource(hInst, hRsrc)) == NULL)
  2098. return NULL;
  2099. LPBITMAPINFOHEADER lpBitmap = (LPBITMAPINFOHEADER)LockResource(hglb);
  2100. if (lpBitmap == NULL)return NULL;
  2101. WORD numcol = NumBitmapColors(lpBitmap);
  2102. ::FreeResource(hglb);
  2103. if(numcol!=16)return(NULL);
  2104. return AfxLoadSysColorBitmap(hInst, hRsrc, FALSE);
  2105. }
  2106. BOOL BCMenu::RemoveMenu(UINT uiId,UINT nFlags)
  2107. {
  2108. if(MF_BYPOSITION&nFlags){
  2109. UINT uint = GetMenuState(uiId,MF_BYPOSITION);
  2110. if(uint&MF_SEPARATOR && !(uint&MF_POPUP)){
  2111. delete m_MenuList.GetAt(uiId);
  2112. m_MenuList.RemoveAt(uiId);
  2113. }
  2114. else{
  2115. BCMenu* pSubMenu = (BCMenu*) GetSubMenu(uiId);
  2116. if(NULL==pSubMenu){
  2117. UINT uiCommandId = GetMenuItemID(uiId);
  2118. for(int i=0;i<m_MenuList.GetSize(); i++){
  2119. if(m_MenuList[i]->nID==uiCommandId){
  2120. delete m_MenuList.GetAt(i);
  2121. m_MenuList.RemoveAt(i);
  2122. break;
  2123. }
  2124. }
  2125. }
  2126. else{
  2127. int numSubMenus = m_SubMenus.GetUpperBound();
  2128. for(int m = numSubMenus; m >= 0; m--){
  2129. if(m_SubMenus[m]==pSubMenu->m_hMenu){
  2130. int numAllSubMenus = m_AllSubMenus.GetUpperBound();
  2131. for(int n = numAllSubMenus; n>= 0; n--){
  2132. if(m_AllSubMenus[n]==m_SubMenus[m])m_AllSubMenus.RemoveAt(n);
  2133. }
  2134. m_SubMenus.RemoveAt(m);
  2135. }
  2136. }
  2137. int num = pSubMenu->GetMenuItemCount();
  2138. for(int i=num-1;i>=0;--i)pSubMenu->RemoveMenu(i,MF_BYPOSITION);
  2139. for(int i=m_MenuList.GetUpperBound();i>=0;i--){
  2140. if(m_MenuList[i]->nID==(UINT)pSubMenu->m_hMenu){
  2141. delete m_MenuList.GetAt(i);
  2142. m_MenuList.RemoveAt(i);
  2143. break;
  2144. }
  2145. }
  2146. delete pSubMenu;
  2147. }
  2148. }
  2149. }
  2150. else{
  2151. int iPosition =0;
  2152. BCMenu* pMenu = FindMenuOption(uiId,iPosition);
  2153. // bug fix RIA 14th September 2000
  2154. // failed to return correct value on call to remove menu as the item was
  2155. // removed twice. The second time its not found
  2156. // so a value of 0 was being returned
  2157. if(pMenu) return pMenu->RemoveMenu(iPosition,MF_BYPOSITION); // added return
  2158. }
  2159. return CMenu::RemoveMenu(uiId,nFlags);
  2160. }
  2161. BOOL BCMenu::DeleteMenu(UINT uiId,UINT nFlags)
  2162. {
  2163. if(MF_BYPOSITION&nFlags){
  2164. UINT uint = GetMenuState(uiId,MF_BYPOSITION);
  2165. if(uint&MF_SEPARATOR && !(uint&MF_POPUP)){
  2166. // make sure it's a separator
  2167. int menulistsize=m_MenuList.GetSize();
  2168. if(uiId<(UINT)menulistsize){
  2169. CString str=m_MenuList[uiId]->GetString();
  2170. if(str==""){
  2171. delete m_MenuList.GetAt(uiId);
  2172. m_MenuList.RemoveAt(uiId);
  2173. }
  2174. }
  2175. }
  2176. else{
  2177. BCMenu* pSubMenu = (BCMenu*) GetSubMenu(uiId);
  2178. if(NULL==pSubMenu){
  2179. UINT uiCommandId = GetMenuItemID(uiId);
  2180. for(int i=0;i<m_MenuList.GetSize(); i++){
  2181. if(m_MenuList[i]->nID==uiCommandId){
  2182. delete m_MenuList.GetAt(i);
  2183. m_MenuList.RemoveAt(i);
  2184. break;
  2185. }
  2186. }
  2187. }
  2188. else{
  2189. int numSubMenus = m_SubMenus.GetUpperBound();
  2190. for(int m = numSubMenus; m >= 0; m--){
  2191. if(m_SubMenus[m]==pSubMenu->m_hMenu){
  2192. int numAllSubMenus = m_AllSubMenus.GetUpperBound();
  2193. for(int n = numAllSubMenus; n>= 0; n--){
  2194. if(m_AllSubMenus[n]==m_SubMenus[m])m_AllSubMenus.RemoveAt(n);
  2195. }
  2196. m_SubMenus.RemoveAt(m);
  2197. }
  2198. }
  2199. int num = pSubMenu->GetMenuItemCount();
  2200. for(int i=num-1;i>=0;--i)pSubMenu->DeleteMenu(i,MF_BYPOSITION);
  2201. for(int i=m_MenuList.GetUpperBound();i>=0;i--){
  2202. if(m_MenuList[i]->nID==(UINT)pSubMenu->m_hMenu){
  2203. delete m_MenuList.GetAt(i);
  2204. m_MenuList.RemoveAt(i);
  2205. break;
  2206. }
  2207. }
  2208. delete pSubMenu;
  2209. }
  2210. }
  2211. }
  2212. else{
  2213. int iPosition =0;
  2214. BCMenu* pMenu = FindMenuOption(uiId,iPosition);
  2215. if(pMenu)return pMenu->DeleteMenu(iPosition,MF_BYPOSITION);
  2216. }
  2217. return(CMenu::DeleteMenu(uiId,nFlags));
  2218. }
  2219. BOOL BCMenu::AppendMenuA(UINT nFlags,UINT nIDNewItem,const char *lpszNewItem,int nIconNormal)
  2220. {
  2221. USES_CONVERSION;
  2222. return AppendMenuW(nFlags,nIDNewItem,A2W(lpszNewItem),nIconNormal);
  2223. }
  2224. BOOL BCMenu::AppendMenuW(UINT nFlags,UINT nIDNewItem,wchar_t *lpszNewItem,int nIconNormal)
  2225. {
  2226. return AppendODMenuW(lpszNewItem,nFlags,nIDNewItem,nIconNormal);
  2227. }
  2228. BOOL BCMenu::AppendMenuA(UINT nFlags,UINT nIDNewItem,const char *lpszNewItem,CImageList *il,int xoffset)
  2229. {
  2230. USES_CONVERSION;
  2231. return AppendMenuW(nFlags,nIDNewItem,A2W(lpszNewItem),il,xoffset);
  2232. }
  2233. BOOL BCMenu::AppendMenuW(UINT nFlags,UINT nIDNewItem,wchar_t *lpszNewItem,CImageList *il,int xoffset)
  2234. {
  2235. return AppendODMenuW(lpszNewItem,nFlags,nIDNewItem,il,xoffset);
  2236. }
  2237. BOOL BCMenu::AppendMenuA(UINT nFlags,UINT nIDNewItem,const char *lpszNewItem,CBitmap *bmp)
  2238. {
  2239. USES_CONVERSION;
  2240. return AppendMenuW(nFlags,nIDNewItem,A2W(lpszNewItem),bmp);
  2241. }
  2242. BOOL BCMenu::AppendMenuW(UINT nFlags,UINT nIDNewItem,wchar_t *lpszNewItem,CBitmap *bmp)
  2243. {
  2244. if(bmp){
  2245. CImageList temp;
  2246. temp.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  2247. if(m_bitmapBackgroundFlag)temp.Add(bmp,m_bitmapBackground);
  2248. else temp.Add(bmp,GetSysColor(COLOR_3DFACE));
  2249. return AppendODMenuW(lpszNewItem,nFlags,nIDNewItem,&temp,0);
  2250. }
  2251. return AppendODMenuW(lpszNewItem,nFlags,nIDNewItem,NULL,0);
  2252. }
  2253. BOOL BCMenu::InsertMenuA(UINT nPosition,UINT nFlags,UINT nIDNewItem,const char *lpszNewItem,int nIconNormal)
  2254. {
  2255. USES_CONVERSION;
  2256. return InsertMenuW(nPosition,nFlags,nIDNewItem,A2W(lpszNewItem),nIconNormal);
  2257. }
  2258. BOOL BCMenu::InsertMenuW(UINT nPosition,UINT nFlags,UINT nIDNewItem,wchar_t *lpszNewItem,int nIconNormal)
  2259. {
  2260. return InsertODMenuW(nPosition,lpszNewItem,nFlags,nIDNewItem,nIconNormal);
  2261. }
  2262. BOOL BCMenu::InsertMenuA(UINT nPosition,UINT nFlags,UINT nIDNewItem,const char *lpszNewItem,CImageList *il,int xoffset)
  2263. {
  2264. USES_CONVERSION;
  2265. return InsertMenuW(nPosition,nFlags,nIDNewItem,A2W(lpszNewItem),il,xoffset);
  2266. }
  2267. BOOL BCMenu::InsertMenuW(UINT nPosition,UINT nFlags,UINT nIDNewItem,wchar_t *lpszNewItem,CImageList *il,int xoffset)
  2268. {
  2269. return InsertODMenuW(nPosition,lpszNewItem,nFlags,nIDNewItem,il,xoffset);
  2270. }
  2271. BOOL BCMenu::InsertMenuA(UINT nPosition,UINT nFlags,UINT nIDNewItem,const char *lpszNewItem,CBitmap *bmp)
  2272. {
  2273. USES_CONVERSION;
  2274. return InsertMenuW(nPosition,nFlags,nIDNewItem,A2W(lpszNewItem),bmp);
  2275. }
  2276. BOOL BCMenu::InsertMenuW(UINT nPosition,UINT nFlags,UINT nIDNewItem,wchar_t *lpszNewItem,CBitmap *bmp)
  2277. {
  2278. if(bmp){
  2279. CImageList temp;
  2280. temp.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  2281. if(m_bitmapBackgroundFlag)temp.Add(bmp,m_bitmapBackground);
  2282. else temp.Add(bmp,GetSysColor(COLOR_3DFACE));
  2283. return InsertODMenuW(nPosition,lpszNewItem,nFlags,nIDNewItem,&temp,0);
  2284. }
  2285. return InsertODMenuW(nPosition,lpszNewItem,nFlags,nIDNewItem,NULL,0);
  2286. }
  2287. //--------------------------------------------------------------------------
  2288. //[21.06.99 rj]
  2289. BCMenu* BCMenu::AppendODPopupMenuW(wchar_t *lpstrText)
  2290. {
  2291. BCMenu* pSubMenu = new BCMenu;
  2292. pSubMenu->m_unselectcheck=m_unselectcheck;
  2293. pSubMenu->m_selectcheck=m_selectcheck;
  2294. pSubMenu->checkmaps=checkmaps;
  2295. pSubMenu->checkmapsshare=TRUE;
  2296. pSubMenu->CreatePopupMenu();
  2297. AppendODMenuW(lpstrText,MF_POPUP,(UINT)pSubMenu->m_hMenu, -1);
  2298. return pSubMenu;
  2299. }
  2300. //--------------------------------------------------------------------------
  2301. //[21.06.99 rj]
  2302. BCMenu* BCMenu::AppendODPopupMenuA(LPCSTR lpstrText)
  2303. {
  2304. USES_CONVERSION;
  2305. return AppendODPopupMenuW(A2W(lpstrText));
  2306. }
  2307. BOOL BCMenu::ImageListDuplicate(CImageList *il,int xoffset,CImageList *newlist)
  2308. {
  2309. if (il == NULL||newlist==NULL||xoffset<0) return FALSE;
  2310. HICON hIcon = il->ExtractIcon(xoffset);
  2311. int cx, cy;
  2312. ImageList_GetIconSize(il->m_hImageList, &cx, &cy);
  2313. newlist->Create(cx,cy,ILC_COLORDDB|ILC_MASK,1,1);
  2314. newlist->Add(hIcon);
  2315. ::DestroyIcon(hIcon);
  2316. if(IsLunaMenuStyle()&&xp_draw_3D_bitmaps){
  2317. CWnd *pWnd = AfxGetMainWnd(); // Get main window
  2318. if (pWnd == NULL) pWnd = CWnd::GetDesktopWindow();
  2319. CDC *pDC = pWnd->GetDC(); // Get device context
  2320. CBitmap bmp,bmp2;
  2321. GetBitmapFromImageList(pDC,newlist,0,bmp);
  2322. GetFadedBitmap(bmp);
  2323. newlist->Add(&bmp,GetSysColor(COLOR_3DFACE));
  2324. GetBitmapFromImageList(pDC,newlist,0,bmp2);
  2325. GetShadowBitmap(bmp2);
  2326. newlist->Add(&bmp2,GetSysColor(COLOR_3DFACE));
  2327. pWnd->ReleaseDC(pDC); // Release the DC
  2328. }
  2329. return TRUE;
  2330. }
  2331. // 2001-07-12, Damir Valiulin:
  2332. // Added GetSubMenu (LPCTSTR lpszSubMenuName) function
  2333. //
  2334. CMenu* BCMenu::GetSubMenu(int nPos)
  2335. {
  2336. return CMenu::GetSubMenu (nPos);
  2337. }
  2338. CMenu* BCMenu::GetSubMenu(LPCTSTR lpszSubMenuName)
  2339. {
  2340. int num = GetMenuItemCount ();
  2341. CString name;
  2342. for (int i=0; i<num; i++)
  2343. {
  2344. GetMenuString (i, name, MF_BYPOSITION);
  2345. if (name.Compare (lpszSubMenuName) == 0)
  2346. {
  2347. return CMenu::GetSubMenu (i);
  2348. }
  2349. }
  2350. return NULL;
  2351. }
  2352. BCMenu* BCMenu::GetSubBCMenu(char* pText)
  2353. {
  2354. USES_CONVERSION;
  2355. return GetSubBCMenu(A2W(pText));
  2356. }
  2357. BCMenu* BCMenu::GetSubBCMenu(wchar_t* lpszSubMenuName)
  2358. {
  2359. BCMenuData *mdata;
  2360. mdata=FindMenuOption(lpszSubMenuName);
  2361. if(mdata){
  2362. HMENU bchmenu=(HMENU)mdata->nID;
  2363. CMenu *ptr=FromHandle(bchmenu);
  2364. BOOL flag=ptr->IsKindOf(RUNTIME_CLASS( BCMenu ));
  2365. if(flag)return((BCMenu *)ptr);
  2366. }
  2367. return NULL;
  2368. }
  2369. // Tongzhe Cui, Functions to remove a popup menu based on its name. Seperators
  2370. // before and after the popup menu can also be removed if they exist.
  2371. int BCMenu::GetMenuPosition(char* pText)
  2372. {
  2373. USES_CONVERSION;
  2374. return GetMenuPosition(A2W(pText));
  2375. }
  2376. int BCMenu::GetMenuPosition(wchar_t* pText)
  2377. {
  2378. int i,j;
  2379. BCMenu* psubmenu;
  2380. for(i=0;i<(int)(GetMenuItemCount());++i)
  2381. {
  2382. psubmenu=(BCMenu *)GetSubMenu(i);
  2383. if(!psubmenu)
  2384. {
  2385. const wchar_t *szWide;//SK: we use const to prevent misuse of this Ptr
  2386. for(j=0;j<=m_MenuList.GetUpperBound();++j)
  2387. {
  2388. szWide = m_MenuList[j]->GetWideString ();
  2389. if(szWide && !wcscmp(pText,szWide))//SK: modified for dynamic allocation
  2390. return j;
  2391. }
  2392. }
  2393. }
  2394. // means no found;
  2395. return -1;
  2396. }
  2397. int BCMenu::RemoveMenu(char* pText, BC_Seperator sPos)
  2398. {
  2399. USES_CONVERSION;
  2400. return RemoveMenu(A2W(pText), sPos);
  2401. }
  2402. int BCMenu::RemoveMenu(wchar_t* pText, BC_Seperator sPos)
  2403. {
  2404. int nPos = GetMenuPosition(pText);
  2405. if(nPos != -1)
  2406. {
  2407. RemoveMenu(nPos, MF_BYPOSITION);
  2408. }
  2409. if(sPos == BCMENU_HEAD)
  2410. {
  2411. ASSERT(nPos - 1 >= 0);
  2412. RemoveMenu(nPos-1, MF_BYPOSITION);
  2413. }
  2414. else if(sPos == BCMENU_TAIL)
  2415. {
  2416. RemoveMenu(nPos-1, MF_BYPOSITION);
  2417. }
  2418. else if(sPos == BCMENU_BOTH)
  2419. {
  2420. // remove the end first;
  2421. RemoveMenu(nPos-1, MF_BYPOSITION);
  2422. // remove the head;
  2423. ASSERT(nPos - 1 >= 0);
  2424. RemoveMenu(nPos-1, MF_BYPOSITION);
  2425. }
  2426. return nPos;
  2427. }
  2428. int BCMenu::DeleteMenu(char* pText, BC_Seperator sPos)
  2429. {
  2430. USES_CONVERSION;
  2431. return DeleteMenu(A2W(pText), sPos);
  2432. }
  2433. int BCMenu::DeleteMenu(wchar_t* pText, BC_Seperator sPos)
  2434. {
  2435. int nPos = GetMenuPosition(pText);
  2436. if(nPos != -1)
  2437. {
  2438. DeleteMenu(nPos, MF_BYPOSITION);
  2439. }
  2440. if(sPos == BCMENU_HEAD)
  2441. {
  2442. ASSERT(nPos - 1 >= 0);
  2443. DeleteMenu(nPos-1, MF_BYPOSITION);
  2444. }
  2445. else if(sPos == BCMENU_TAIL)
  2446. {
  2447. DeleteMenu(nPos-1, MF_BYPOSITION);
  2448. }
  2449. else if(sPos == BCMENU_BOTH)
  2450. {
  2451. // remove the end first;
  2452. DeleteMenu(nPos-1, MF_BYPOSITION);
  2453. // remove the head;
  2454. ASSERT(nPos - 1 >= 0);
  2455. DeleteMenu(nPos-1, MF_BYPOSITION);
  2456. }
  2457. return nPos;
  2458. }
  2459. // Larry Antram
  2460. BOOL BCMenu::SetMenuText(UINT id, CString string, UINT nFlags/*= MF_BYPOSITION*/ )
  2461. {
  2462. BOOL returnflag=FALSE;
  2463. if(MF_BYPOSITION&nFlags)
  2464. {
  2465. UINT numMenuItems = m_MenuList.GetUpperBound();
  2466. if(id<=numMenuItems){
  2467. #ifdef UNICODE
  2468. m_MenuList[id]->SetWideString((LPCTSTR)string);
  2469. #else
  2470. m_MenuList[id]->SetAnsiString(string);
  2471. #endif
  2472. returnflag=TRUE;
  2473. }
  2474. }
  2475. else{
  2476. int uiLoc;
  2477. BCMenu* pMenu = FindMenuOption(id,uiLoc);
  2478. if(NULL!=pMenu) returnflag = pMenu->SetMenuText(uiLoc,string);
  2479. }
  2480. return(returnflag);
  2481. }
  2482. // courtesy of Warren Stevens
  2483. void BCMenu::ColorBitmap(CDC* pDC,CBitmap& bmp,CSize bitmap_size,CSize icon_size,COLORREF fill,COLORREF border,int hatchstyle)
  2484. {
  2485. CDC bmpdc;
  2486. COLORREF m_newclrBack;
  2487. int x1=0,y1=0,x2=bitmap_size.cx,y2=bitmap_size.cy;
  2488. if(IsWinXPLuna())m_newclrBack=GetSysColor(COLOR_3DFACE);
  2489. else m_newclrBack=GetSysColor(COLOR_MENU);
  2490. bmpdc.CreateCompatibleDC(pDC);
  2491. bmp.CreateCompatibleBitmap(pDC, icon_size.cx, icon_size.cy);
  2492. CBitmap* pOldBitmap = bmpdc.SelectObject(&bmp);
  2493. if(bitmap_size!=icon_size){
  2494. CBrush background_brush;
  2495. background_brush.CreateSolidBrush(m_newclrBack);
  2496. CRect rect(0,0, icon_size.cx, icon_size.cy);
  2497. bmpdc.FillRect(rect,&background_brush);
  2498. x1 = (icon_size.cx-bitmap_size.cx)/2;
  2499. y1 = (icon_size.cy-bitmap_size.cy)/2;
  2500. x2 = x1+bitmap_size.cx;
  2501. y2 = y1+bitmap_size.cy;
  2502. background_brush.DeleteObject();
  2503. }
  2504. CPen border_pen(PS_SOLID, 1, border);
  2505. CBrush fill_brush;
  2506. if(hatchstyle!=-1) { fill_brush.CreateHatchBrush(hatchstyle, fill); }
  2507. else { fill_brush.CreateSolidBrush(fill); }
  2508. CPen* pOldPen = bmpdc.SelectObject(&border_pen);
  2509. CBrush* pOldBrush = bmpdc.SelectObject(&fill_brush);
  2510. bmpdc.Rectangle(x1,y1,x2,y2);
  2511. if(NULL!=pOldBrush) { bmpdc.SelectObject(pOldBrush); }
  2512. if(NULL!=pOldPen) { bmpdc.SelectObject(pOldPen); }
  2513. if(NULL!=pOldBitmap) { bmpdc.SelectObject(pOldBitmap); }
  2514. }
  2515. BOOL BCMenu::IsWindowsClassicTheme(void)
  2516. {
  2517. TCHAR Buf[_MAX_PATH+10];
  2518. HKEY hKey;
  2519. DWORD size,type;
  2520. long lRetCode;
  2521. static BOOL XPTheme_returnflag=FALSE;
  2522. static BOOL XPTheme_checkflag=FALSE;
  2523. if(XPTheme_checkflag)return(XPTheme_returnflag);
  2524. XPTheme_checkflag=TRUE;
  2525. lRetCode = RegOpenKeyEx ( HKEY_CURRENT_USER,
  2526. _T("Software\\Microsoft\\Plus!\\Themes\\Current"),
  2527. 0,KEY_READ,&hKey);
  2528. if (lRetCode == ERROR_SUCCESS){
  2529. size = _MAX_PATH;type=REG_SZ;
  2530. lRetCode=::RegQueryValueEx(hKey,NULL,NULL,&type,
  2531. (unsigned char *)Buf,&size);
  2532. if(lRetCode == ERROR_SUCCESS){
  2533. TCHAR szClassicTheme[]=_T("Windows Classic.theme");
  2534. int len=lstrlen(Buf);
  2535. if(len>=lstrlen(szClassicTheme)){
  2536. if(!lstrcmpi(&Buf[len-lstrlen(szClassicTheme)],szClassicTheme)){
  2537. XPTheme_returnflag=TRUE;
  2538. }
  2539. }
  2540. }
  2541. RegCloseKey(hKey);
  2542. }
  2543. return(XPTheme_returnflag);
  2544. }
  2545. int BCMenu::GlobalImageListOffset(int nID)
  2546. {
  2547. int numcurrent=m_AllImagesID.GetSize();
  2548. int existsloc = -1;
  2549. for(int i=0;i<numcurrent;++i){
  2550. if(m_AllImagesID[i]==nID){
  2551. existsloc=i;
  2552. break;
  2553. }
  2554. }
  2555. return existsloc;
  2556. }
  2557. BOOL BCMenu::CanDraw3DImageList(int offset)
  2558. {
  2559. BOOL retflag=FALSE;
  2560. int numcurrent=m_AllImagesID.GetSize();
  2561. if(offset+1<numcurrent&&offset+2<numcurrent){
  2562. int nID=m_AllImagesID[offset];
  2563. if(m_AllImagesID[offset+1]==nID&&m_AllImagesID[offset+2]==nID)retflag=TRUE;
  2564. }
  2565. return(retflag);
  2566. }
  2567. int BCMenu::AddToGlobalImageList(CImageList *il,int xoffset,int nID)
  2568. {
  2569. int loc = -1;
  2570. HIMAGELIST hImageList = m_AllImages.m_hImageList;
  2571. if(!hImageList){
  2572. m_AllImages.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  2573. }
  2574. HICON hIcon = il->ExtractIcon(xoffset);
  2575. if(hIcon){
  2576. CBitmap bmp,bmp2;
  2577. if(IsLunaMenuStyle()&&xp_draw_3D_bitmaps){
  2578. CWnd *pWnd = AfxGetMainWnd(); // Get main window
  2579. if (pWnd == NULL) pWnd = CWnd::GetDesktopWindow();
  2580. CDC *pDC = pWnd->GetDC(); // Get device context
  2581. GetBitmapFromImageList(pDC,il,xoffset,bmp);
  2582. GetFadedBitmap(bmp);
  2583. GetBitmapFromImageList(pDC,il,xoffset,bmp2);
  2584. GetShadowBitmap(bmp2);
  2585. pWnd->ReleaseDC(pDC); // Release the DC
  2586. }
  2587. int numcurrent=m_AllImagesID.GetSize();
  2588. int existsloc = -1;
  2589. for(int i=0;i<numcurrent;++i){
  2590. if(m_AllImagesID[i]==nID){
  2591. existsloc=i;
  2592. break;
  2593. }
  2594. }
  2595. if(existsloc>=0){
  2596. m_AllImages.Replace(existsloc,hIcon);
  2597. loc = existsloc;
  2598. if(IsLunaMenuStyle()&&xp_draw_3D_bitmaps){
  2599. if(existsloc+1<numcurrent&&m_AllImagesID[existsloc+1]==nID){
  2600. if(existsloc+2<numcurrent&&m_AllImagesID[existsloc+2]==nID){
  2601. CImageList il2;
  2602. il2.Create(m_iconX,m_iconY,ILC_COLORDDB|ILC_MASK,1,1);
  2603. il2.Add(&bmp,GetSysColor(COLOR_3DFACE));
  2604. HICON hIcon2 = il2.ExtractIcon(0);
  2605. m_AllImages.Replace(existsloc+1,hIcon2);
  2606. il2.Add(&bmp2,GetSysColor(COLOR_3DFACE));
  2607. HICON hIcon3 = il2.ExtractIcon(1);
  2608. m_AllImages.Replace(existsloc+2,hIcon3);
  2609. ::DestroyIcon(hIcon2);
  2610. ::DestroyIcon(hIcon3);
  2611. }
  2612. }
  2613. }
  2614. }
  2615. else{
  2616. m_AllImages.Add(hIcon);
  2617. m_AllImagesID.Add(nID);
  2618. loc=numcurrent;
  2619. if(IsLunaMenuStyle()&&xp_draw_3D_bitmaps){
  2620. m_AllImages.Add(&bmp,GetSysColor(COLOR_3DFACE));
  2621. m_AllImages.Add(&bmp2,GetSysColor(COLOR_3DFACE));
  2622. m_AllImagesID.Add(nID);
  2623. m_AllImagesID.Add(nID);
  2624. }
  2625. }
  2626. ::DestroyIcon(hIcon);
  2627. }
  2628. return(loc);
  2629. }
  2630. //*************************************************************************
  2631. /////////////////////////////////////////////////////////////////////////////
  2632. // Toolbar for loading images
  2633. /////////////////////////////////////////////////////////////////////////////
  2634. struct CToolBarData
  2635. {
  2636. WORD wVersion;
  2637. WORD wWidth;
  2638. WORD wHeight;
  2639. WORD wItemCount;
  2640. WORD* items(){
  2641. return (WORD*)(this+1);
  2642. }
  2643. };
  2644. BOOL BCMenuToolBar::LoadToolBar(LPCTSTR lpszResourceName)
  2645. {
  2646. ASSERT_VALID(this);
  2647. ASSERT(lpszResourceName != NULL);
  2648. HINSTANCE hInst = AfxFindResourceHandle(lpszResourceName,RT_TOOLBAR);
  2649. HRSRC hRsrc = ::FindResource(hInst,lpszResourceName,RT_TOOLBAR);
  2650. if (hRsrc == NULL){
  2651. hInst = NULL;
  2652. hRsrc = ::FindResource(hInst,lpszResourceName,RT_TOOLBAR);
  2653. }
  2654. if (hRsrc == NULL)
  2655. return FALSE;
  2656. HGLOBAL hGlobal = ::LoadResource(hInst,hRsrc);
  2657. if (hGlobal == NULL)
  2658. return FALSE;
  2659. CToolBarData* pData = (CToolBarData*)::LockResource(hGlobal);
  2660. if (pData == NULL)
  2661. return FALSE;
  2662. ASSERT(pData->wVersion == 1);
  2663. UINT* pItems = new UINT[pData->wItemCount];
  2664. for (int i = 0; i < pData->wItemCount; i++)
  2665. pItems[i] = pData->items()[i];
  2666. BOOL bResult = SetButtons(pItems,pData->wItemCount);
  2667. delete[] pItems;
  2668. if (bResult){
  2669. CSize sizeImage(pData->wWidth,pData->wHeight);
  2670. CSize sizeButton(pData->wWidth+7,pData->wHeight+7);
  2671. SetSizes(sizeButton,sizeImage);
  2672. m_iconX=pData->wWidth;
  2673. m_iconY=pData->wHeight;
  2674. bResult = LoadBitmap(lpszResourceName);
  2675. }
  2676. ::UnlockResource(hGlobal);
  2677. ::FreeResource(hGlobal);
  2678. return bResult;
  2679. }
  2680. BOOL BCMenuToolBar::LoadBitmap(LPCTSTR lpszResourceName)
  2681. {
  2682. ASSERT_VALID(this);
  2683. ASSERT(lpszResourceName != NULL);
  2684. HINSTANCE hInstImageWell = AfxFindResourceHandle(lpszResourceName,RT_BITMAP);
  2685. HRSRC hRsrcImageWell = ::FindResource(hInstImageWell,lpszResourceName,RT_BITMAP);
  2686. if (hRsrcImageWell == NULL){
  2687. hInstImageWell = NULL;
  2688. hRsrcImageWell = ::FindResource(hInstImageWell,lpszResourceName,RT_BITMAP);
  2689. }
  2690. if (hRsrcImageWell == NULL)
  2691. return FALSE;
  2692. HBITMAP hbmImageWell;
  2693. hbmImageWell = AfxLoadSysColorBitmap(hInstImageWell,hRsrcImageWell);
  2694. if (!AddReplaceBitmap(hbmImageWell))
  2695. return FALSE;
  2696. m_hInstImageWell = hInstImageWell;
  2697. m_hRsrcImageWell = hRsrcImageWell;
  2698. return TRUE;
  2699. }