skinsb.c 45 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823
  1. //////////////////////////////////////////////////////////////////////////////////
  2. // Module Name : UISkin scrollbar library version 1.0
  3. // Copyright (C) LiJun(Chinase) 2006
  4. //
  5. // Descrption : Implement custom draw internal window of scrollbar
  6. //
  7. // Email : Notoldtree@126.com
  8. //
  9. // CREATE DATE : 2006.10
  10. //
  11. // VERSION HISTORY :
  12. //
  13. // 2008.06
  14. //
  15. //
  16. //
  17. //////////////////////////////////////////////////////////////////////////////////
  18. //#include "stdafx.h"
  19. #include "internal.h"
  20. #include "skinsb.h"
  21. //#include <stdio.h>
  22. #define WIN32_LEAN_AND_MEAN
  23. static TCHAR g_szPropSB[] = TEXT("PROP_SkinSB");
  24. // The skin bitmap resourece array
  25. static POINT ptArray[6][6] = {
  26. { {0, 0}, {19, 0}, {38, 0}, {57, 0}, {76, 0}, {95, 0} },
  27. { {0, 19}, {19, 19}, {38, 19}, {57, 19}, {76, 19}, {95, 19} },
  28. { {0, 38}, {19, 38}, {38, 38}, {57, 38}, {76, 38}, {95, 38} },
  29. { {0, 57}, {19, 57}, {38, 57}, {57, 57}, {76, 57}, {95, 57} },
  30. { {0, 76}, {19, 76}, {38, 76}, {57, 76}, {76, 76}, {95, 76} },
  31. { {0, 95}, {19, 95}, {38, 95}, {57, 95}, {76, 95}, {95, 95} }
  32. };
  33. // Public interfaces
  34. //----------------------------------------------------------
  35. // Name : SkinSB_Init()
  36. // Desc : Initialize the skin scrollbar library
  37. //----------------------------------------------------------
  38. BOOL WINAPI SkinSB_Init(HWND hwnd, HBITMAP hBmp)
  39. {
  40. LPSB psb;
  41. DWORD dwStyle;
  42. if( !IsWindow(hwnd) ) {
  43. SetLastError(ERROR_INVALID_HANDLE);
  44. return FALSE;
  45. }
  46. if( SkinSB_IsValid(hwnd) )
  47. return FALSE;
  48. // Allocates memory
  49. psb = (LPSB)LocalAlloc(LPTR, sizeof(SB));
  50. if( psb == NULL )
  51. return FALSE;
  52. ZeroMemory(psb, sizeof(SB));
  53. dwStyle = GetWindowLong(hwnd, GWL_STYLE);
  54. psb->Horz.cbSize = psb->Vert.cbSize = sizeof(SCROLLINFO);
  55. psb->Horz.fMask = psb->Vert.fMask = SIF_ALL;
  56. if( dwStyle & WS_HSCROLL )
  57. GetScrollInfo(hwnd, SB_HORZ, &psb->Horz);
  58. if( dwStyle & WS_VSCROLL )
  59. GetScrollInfo(hwnd, SB_VERT, &psb->Vert);
  60. // Is left scrollbar style
  61. if( GetWindowLong(hwnd, GWL_EXSTYLE) & WS_EX_LEFTSCROLLBAR )
  62. psb->fLeftScrollBar = TRUE;
  63. if( !hBmp ) {
  64. SetLastError(ERROR_INVALID_HANDLE);
  65. return FALSE;
  66. }
  67. psb->hBmp = hBmp;
  68. // Reaplace the window procedure
  69. psb->hwnd = hwnd;
  70. psb->pfnOldProc = (WNDPROC)(LONG_PTR)SetWindowLong(hwnd,
  71. GWL_WNDPROC, (LONG)(LONG_PTR)SkinSB_Proc);
  72. psb->fPreventStyleChange = FALSE;
  73. psb->fTracking = FALSE;
  74. // Whether richedit control
  75. TCHAR szClassName[255] = { 0 };
  76. GetClassName(hwnd, szClassName, sizeof(szClassName));
  77. if( _strcmpi(szClassName, TEXT("RichEdit20A")) == 0 ||
  78. _strcmpi(szClassName, TEXT("RichEdit20W")) == 0 ) {
  79. psb->fRichEdit = TRUE;
  80. }
  81. // Set the window property
  82. if( !SetProp(hwnd, g_szPropSB, (HANDLE)psb) )
  83. return FALSE;
  84. // Redraw the window noclient
  85. SetWindowPos(hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER |
  86. SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_DRAWFRAME);
  87. return TRUE;
  88. }
  89. //---------------------------------------------------------
  90. // Name : SkinSB_Uninit()
  91. // Desc : Finalize the skin scrollbar library
  92. //---------------------------------------------------------
  93. BOOL WINAPI SkinSB_Uninit(HWND hwnd)
  94. {
  95. LPSB psb;
  96. SCROLLINFO vsi;
  97. SCROLLINFO hsi;
  98. BOOL vValid, hValid;
  99. UINT vFlags, hFlags;
  100. if( (psb = SkinSB_GetSB(hwnd)) == NULL )
  101. return FALSE;
  102. vsi.cbSize = hsi.cbSize = sizeof(SCROLLINFO);
  103. vsi.fMask = hsi.fMask = SIF_ALL | SIF_DISABLENOSCROLL;
  104. vValid = SkinSB_GetScrollInfo(hwnd, SB_VERT, &vsi);
  105. vFlags = SkinSB_GetDisableFlags(psb, TRUE);
  106. hValid = SkinSB_GetScrollInfo(hwnd, SB_HORZ, &hsi);
  107. hFlags = SkinSB_GetDisableFlags(psb, FALSE);
  108. // Remove the subclass procedure
  109. SetWindowLong(hwnd, GWL_WNDPROC, (LONG)(LONG_PTR)psb->pfnOldProc);
  110. RemoveProp(hwnd, g_szPropSB);
  111. LocalFree((HLOCAL)psb);
  112. // Restore system scroll parameters
  113. if( vValid ) {
  114. SetScrollInfo(hwnd, SB_VERT, &vsi, TRUE);
  115. EnableScrollBar(hwnd, SB_VERT, vFlags);
  116. }
  117. if( hValid ) {
  118. SetScrollInfo(hwnd, SB_HORZ, &hsi, TRUE);
  119. EnableScrollBar(hwnd, SB_HORZ, hFlags);
  120. }
  121. return TRUE;
  122. }
  123. //---------------------------------------------------------
  124. // Name : SkinSB_IsValid()
  125. // Desc : Is initialized ??
  126. //---------------------------------------------------------
  127. BOOL WINAPI SkinSB_IsValid(HWND hwnd)
  128. {
  129. return (SkinSB_GetSB(hwnd) != NULL);
  130. }
  131. //---------------------------------------------------------
  132. // Name : SkinSB_GetScrollInfo()
  133. // Desc : Overload API GetScrollInfo()
  134. //---------------------------------------------------------
  135. BOOL WINAPI SkinSB_GetScrollInfo(HWND hwnd, int fnBar, LPSCROLLINFO lpsi)
  136. {
  137. LPSB psb;
  138. LPSCROLLINFO psi;
  139. BOOL fCopied = FALSE;
  140. if( !lpsi || lpsi->cbSize != sizeof(SCROLLINFO) )
  141. return FALSE;
  142. // Is initialized ?
  143. if( (psb = SkinSB_GetSB(hwnd)) == NULL )
  144. return FALSE;
  145. // If be scrollbar control then call failed
  146. if( fnBar == SB_HORZ )
  147. psi = &psb->Horz;
  148. else if( fnBar == SB_VERT )
  149. psi = &psb->Vert;
  150. else if( fnBar == SB_CTL )
  151. return FALSE;
  152. if( lpsi->fMask & SIF_PAGE ) {
  153. lpsi->nPage = psi->nPage;
  154. fCopied = TRUE;
  155. }
  156. if( lpsi->fMask & SIF_POS ) {
  157. lpsi->nPos = psi->nPos;
  158. fCopied = TRUE;
  159. }
  160. if( lpsi->fMask & SIF_TRACKPOS ) {
  161. lpsi->nTrackPos = psi->nTrackPos;
  162. fCopied = TRUE;
  163. }
  164. if( lpsi->fMask & SIF_RANGE ) {
  165. lpsi->nMin = psi->nMin;
  166. lpsi->nMax = psi->nMax;
  167. fCopied = TRUE;
  168. }
  169. return fCopied;
  170. }
  171. //---------------------------------------------------------
  172. // Name : SkinSB_SetScrollInfo()
  173. // Desc : Overload API SetScrollInfo()
  174. //---------------------------------------------------------
  175. int WINAPI SkinSB_SetScrollInfo(HWND hwnd, int fnBar, LPCSCROLLINFO psi, BOOL fRedraw)
  176. {
  177. LPSB psb;
  178. LPSCROLLINFO mysi;
  179. int nRet;
  180. DWORD dwStyle;
  181. BOOL fVert;
  182. UINT wScroll;
  183. BOOL fScroll;
  184. BOOL fOldScroll;
  185. BOOL bReturnOldPos;
  186. // if be scrollbar control the call failed
  187. if( fnBar == SB_CTL )
  188. return 0;
  189. if( (psb = SkinSB_GetSB(hwnd)) == NULL )
  190. return 0;
  191. if( fRedraw )
  192. fRedraw = IsWindowVisible(hwnd);
  193. fVert = (fnBar == SB_VERT);
  194. bReturnOldPos = (psi->fMask & SIF_POS);
  195. dwStyle = GetWindowLong(hwnd, GWL_STYLE);
  196. wScroll = fVert ? WS_VSCROLL : WS_HSCROLL;
  197. fScroll = fOldScroll = (dwStyle & wScroll) ? TRUE : FALSE;
  198. // Don't do anything if we're setting position of a nonexistent scroll bar.
  199. if( !(psi->fMask & SIF_RANGE) && !fOldScroll )
  200. return 0;
  201. mysi = (fVert ? &psb->Vert : &psb->Horz);
  202. if( !SkinSB_SetSBParms(mysi, *psi, &fScroll, &nRet, bReturnOldPos) &&
  203. !(psi->fMask & SIF_DISABLENOSCROLL) ) {
  204. if( fOldScroll && fRedraw )
  205. goto redrawAfterSet;
  206. return nRet;
  207. }
  208. if( fScroll )
  209. psb->style |= wScroll;
  210. else
  211. psb->style &= ~wScroll;
  212. // Keep the owner window scroll style
  213. SetWindowLong(hwnd, GWL_STYLE, dwStyle | wScroll);
  214. if( psi->fMask & SIF_DISABLENOSCROLL ) {
  215. if( fOldScroll ) {
  216. fScroll = ((int)mysi->nPage <= (mysi->nMax - mysi->nMin));
  217. psb->style |= wScroll;
  218. SetWindowLong(hwnd, GWL_STYLE, dwStyle | wScroll);
  219. SkinSB_EnableArrows(psb, fnBar, fScroll ? ESB_ENABLE_BOTH : ESB_DISABLE_BOTH);
  220. }
  221. }
  222. else if( fOldScroll ^ fScroll ) {
  223. SetWindowPos(hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER |
  224. SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_DRAWFRAME);
  225. return nRet;
  226. }
  227. if( fScroll && fRedraw ) {
  228. redrawAfterSet:
  229. if(dwStyle & wScroll) {
  230. HDC hDC = GetWindowDC(hwnd);
  231. SkinSB_DrawThumb(psb, hDC, fVert);
  232. ReleaseDC(hwnd, hDC);
  233. }
  234. else {
  235. if( SkinSB_IsScrollInfoActive(psi) )
  236. SetWindowPos(hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER |
  237. SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_DRAWFRAME);
  238. }
  239. }
  240. return mysi->nPos; //lres;
  241. }
  242. //---------------------------------------------------------
  243. // Name : SkinSB_GetScrollPos()
  244. // Desc : Overload API GetScrollPos()
  245. //---------------------------------------------------------
  246. int WINAPI SkinSB_GetScrollPos(HWND hwnd, int nBar)
  247. {
  248. LPSB psb;
  249. int nPos;
  250. if( !(psb = SkinSB_GetSB(hwnd)) )
  251. return FALSE;
  252. if( nBar == SB_HORZ )
  253. nPos = psb->Horz.nPos;
  254. else if( nBar == SB_VERT )
  255. nPos = psb->Vert.nPos;
  256. return nPos;
  257. }
  258. //---------------------------------------------------------
  259. // Name : SkinSB_SetScrollPos()
  260. // Desc : Overload API SetScrollPos()
  261. //---------------------------------------------------------
  262. int WINAPI SkinSB_SetScrollPos(HWND hwnd, int nBar, int nPos, BOOL fRedraw)
  263. {
  264. LPSB psb;
  265. LPSCROLLINFO psi;
  266. int nOldPos;
  267. if( (psb = SkinSB_GetSB(hwnd)) == NULL)
  268. return FALSE;
  269. if( nBar == SB_HORZ )
  270. psi = &psb->Horz;
  271. else if( nBar == SB_VERT )
  272. psi = &psb->Vert;
  273. else
  274. return FALSE;
  275. nOldPos = psi->nPos;
  276. psi->nPos = nPos;
  277. if( fRedraw ) {
  278. HDC hDC = GetWindowDC(hwnd);
  279. SkinSB_DrawScrollBar(psb, hDC, (nBar == SB_VERT) );
  280. ReleaseDC(hwnd, hDC);
  281. }
  282. return nOldPos;
  283. }
  284. //---------------------------------------------------------
  285. // Name : SkinSB_GetScrollRange()
  286. // Desc : Overload API GetScrollRange()
  287. //---------------------------------------------------------
  288. BOOL WINAPI SkinSB_GetScrollRange(HWND hwnd, int nBar, LPINT lpMinPos, LPINT lpMaxPos)
  289. {
  290. LPSB psb;
  291. LPSCROLLINFO psi;
  292. if( !lpMinPos || !lpMaxPos )
  293. return FALSE;
  294. if( (psb = SkinSB_GetSB(hwnd)) == NULL )
  295. return FALSE;
  296. if( nBar == SB_HORZ )
  297. psi = &psb->Horz;
  298. else if( nBar == SB_VERT )
  299. psi = &psb->Vert;
  300. *lpMinPos = psi->nMin;
  301. *lpMaxPos = psi->nMax;
  302. return TRUE;
  303. }
  304. //---------------------------------------------------------
  305. // Name : SkinSB_SetScrollRange()
  306. // Desc : Overload API SetScrollRange()
  307. //---------------------------------------------------------
  308. BOOL WINAPI SkinSB_SetScrollRange(HWND hwnd, int nBar, int nMinPos, int nMaxPos, BOOL fRedraw)
  309. {
  310. LPSB psb;
  311. LPSCROLLINFO psi;
  312. if( !(psb = SkinSB_GetSB(hwnd)) )
  313. return FALSE;
  314. if( nBar == SB_HORZ )
  315. psi = &psb->Horz;
  316. else if( nBar == SB_VERT )
  317. psi = &psb->Vert;
  318. psi->nMin = nMinPos;
  319. psi->nMax = nMaxPos;
  320. if(nMinPos == 0 && nMaxPos == 0)
  321. SkinSB_ShowScrollBar(hwnd, nBar, FALSE);
  322. if( fRedraw ) {
  323. HDC hDC = GetWindowDC(hwnd);
  324. SkinSB_DrawScrollBar(psb, hDC, (nBar == SB_VERT) );
  325. ReleaseDC(hwnd, hDC);
  326. }
  327. return TRUE;
  328. }
  329. //---------------------------------------------------------
  330. // Name : SkinSB_ShowScrollBar()
  331. // Desc : Overload API ShowScrollBar()
  332. //---------------------------------------------------------
  333. BOOL WINAPI SkinSB_ShowScrollBar(HWND hwnd, int wBar, BOOL fShow)
  334. {
  335. BOOL fChanged = FALSE;
  336. DWORD dwStyle, dwNew;
  337. switch( wBar )
  338. {
  339. case SB_CTL:
  340. ShowWindow(hwnd, fShow ? SW_SHOW : SW_HIDE);
  341. break;
  342. case SB_HORZ:
  343. dwNew = WS_HSCROLL;
  344. break;
  345. case SB_VERT:
  346. dwNew = WS_VSCROLL;
  347. break;
  348. case SB_BOTH:
  349. dwNew = WS_HSCROLL | WS_VSCROLL;
  350. break;
  351. }
  352. dwStyle = GetWindowLong(hwnd, GWL_STYLE);
  353. if( !fShow ) {
  354. if( dwStyle & dwNew ) {
  355. fChanged = TRUE;
  356. dwStyle &= ~dwNew;
  357. }
  358. }
  359. else {
  360. if( (dwStyle & dwNew) != dwNew ) {
  361. fChanged = TRUE;
  362. dwStyle |= dwNew;
  363. }
  364. }
  365. if( fChanged ) {
  366. SetWindowLong(hwnd, GWL_STYLE, dwStyle);
  367. SetWindowPos(hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER |
  368. SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_DRAWFRAME);
  369. }
  370. return TRUE;
  371. }
  372. //---------------------------------------------------------
  373. // Name : SkinSB_EnableScrollBar()
  374. // Desc : Overload API EnableScrollBar()
  375. //---------------------------------------------------------
  376. BOOL WINAPI SkinSB_EnableScrollBar(HWND hwnd, UINT wSBflags, UINT wArrows)
  377. {
  378. LPSB psb;
  379. if( !(psb = SkinSB_GetSB(hwnd)) )
  380. return FALSE;
  381. if( wSBflags == SB_CTL )
  382. return FALSE;
  383. return SkinSB_EnableArrows(psb, wSBflags, wArrows);
  384. }
  385. // Internal Function
  386. //---------------------------------------------------------
  387. // Name : SkinSB_SetSBParms()
  388. // Desc : Set scroll parameters
  389. //---------------------------------------------------------
  390. BOOL SkinSB_SetSBParms(LPSCROLLINFO psi, SCROLLINFO si, BOOL* pfScroll, int* plres, BOOL bOldPos)
  391. {
  392. BOOL fChanged = FALSE;
  393. if( bOldPos )
  394. *plres = psi->nPos;
  395. if( si.fMask & SIF_RANGE ) {
  396. if( si.nMax < si.nMin )
  397. si.nMax = si.nMin;
  398. if( si.nMax != psi->nMax || si.nMin != psi->nMin ) {
  399. psi->nMax = si.nMax;
  400. psi->nMin = si.nMin;
  401. if( !(si.fMask & SIF_PAGE) ) {
  402. si.fMask |= SIF_PAGE;
  403. si.nPage = psi->nPage;
  404. }
  405. if( !(si.fMask & SIF_POS) ) {
  406. si.fMask |= SIF_POS;
  407. si.nPos = psi->nPos;
  408. }
  409. fChanged = TRUE;
  410. }
  411. }
  412. if( si.fMask & SIF_PAGE ) {
  413. UINT nMaxPage = abs( psi->nMax - psi->nMin ) + 1;
  414. if( si.nPage > nMaxPage )
  415. si.nPage = nMaxPage;
  416. if( psi->nPage != si.nPage ) {
  417. psi->nPage = si.nPage;
  418. if( !(si.fMask & SIF_POS) ) {
  419. si.fMask |= SIF_POS;
  420. si.nPos = psi->nPos;
  421. }
  422. fChanged = TRUE;
  423. }
  424. }
  425. if( si.fMask & SIF_POS ) {
  426. int nMaxPos = psi->nMax - ((psi->nPage) ? psi->nPage - 1 : 0);
  427. if( si.nPos < psi->nMin )
  428. si.nPos = psi->nMin;
  429. else if( si.nPos > nMaxPos )
  430. si.nPos = nMaxPos;
  431. if( psi->nPos != si.nPos ) {
  432. psi->nPos = si.nPos;
  433. fChanged = TRUE;
  434. }
  435. }
  436. if( si.fMask & SIF_TRACKPOS ) {
  437. if( psi->nTrackPos != si.nTrackPos ) {
  438. psi->nTrackPos = si.nTrackPos;
  439. fChanged = TRUE;
  440. }
  441. }
  442. if( !bOldPos )
  443. *plres = psi->nPos; // Return the new pos
  444. if( si.fMask & SIF_RANGE ) {
  445. if( *pfScroll = (psi->nMin != psi->nMax) )
  446. goto CheckPage;
  447. }
  448. else if( si.fMask & SIF_PAGE ) {
  449. CheckPage:
  450. *pfScroll = ( (int)psi->nPage <= (psi->nMax - psi->nMin) );
  451. }
  452. return fChanged;
  453. }
  454. //---------------------------------------------------------
  455. // Name : SkinSB_EnableArrows()
  456. // Desc : Enable or Disable scrollbar arrows
  457. //---------------------------------------------------------
  458. BOOL SkinSB_EnableArrows(LPSB psb, int nBar, UINT wArrows)
  459. {
  460. HDC hDC;
  461. UINT uOldFlags;
  462. BOOL bRetValue = FALSE;
  463. if( psb == NULL )
  464. return FALSE;
  465. uOldFlags = psb->flags;
  466. if( (hDC = GetWindowDC(psb->hwnd)) == NULL )
  467. return FALSE;
  468. // Enable or disable horizontal scrollbar
  469. if( nBar == SB_HORZ || nBar == SB_BOTH ) {
  470. if( wArrows == ESB_ENABLE_BOTH )
  471. psb->flags &= ~ESB_DISABLE_BOTH;
  472. else
  473. psb->flags |= wArrows;
  474. if( uOldFlags != psb->flags ) {
  475. bRetValue = TRUE;
  476. if( GetWindowLong(psb->hwnd, GWL_STYLE ) & WS_HSCROLL )
  477. SkinSB_DrawScrollBar(psb, hDC, FALSE);
  478. }
  479. }
  480. // Enable or disable vertical scrollbar
  481. if( nBar == SB_VERT || nBar == SB_BOTH ) {
  482. if( wArrows == ESB_ENABLE_BOTH )
  483. psb->flags &= ~(ESB_DISABLE_BOTH << 2);
  484. else
  485. psb->flags |= (wArrows << 2);
  486. if( uOldFlags != psb->flags ) {
  487. bRetValue = TRUE;
  488. if( GetWindowLong(psb->hwnd, GWL_STYLE ) & WS_VSCROLL )
  489. SkinSB_DrawScrollBar(psb, hDC, TRUE);
  490. }
  491. }
  492. ReleaseDC(psb->hwnd, hDC);
  493. return bRetValue;
  494. }
  495. //---------------------------------------------------------
  496. // Name : SkinSB_GetDisableFlags()
  497. // Desc : Get scrollbar disable flags
  498. //---------------------------------------------------------
  499. UINT SkinSB_GetDisableFlags(LPSB psb, BOOL fVert)
  500. {
  501. return (fVert ? (psb->flags & 0x000c) >> 2 : psb->flags & 0x0003);
  502. }
  503. //---------------------------------------------------------
  504. // Name : SkinSB_GetSB()
  505. // Desc : Get the skin scrollbar data structure
  506. //---------------------------------------------------------
  507. LPSB SkinSB_GetSB(HWND hwnd)
  508. {
  509. return (LPSB)GetProp(hwnd, g_szPropSB);
  510. }
  511. //---------------------------------------------------------
  512. // Name : SkinSB_IsScrollInfoActive()
  513. // Desc : Check scroll information whether active
  514. //---------------------------------------------------------
  515. BOOL SkinSB_IsScrollInfoActive(LPCSCROLLINFO lpsi)
  516. {
  517. if( lpsi->nPage > (UINT)lpsi->nMax || lpsi->nMax <= lpsi->nMin || lpsi->nMax == 0 )
  518. return FALSE;
  519. else
  520. return TRUE;
  521. }
  522. //----------------------------------------------------------
  523. // Name : SkinSB_GetSizeBoxRect()
  524. // Desc : Get the size box rect
  525. //----------------------------------------------------------
  526. BOOL SkinSB_GetSizeBoxRect(LPSB psb, LPRECT lprc)
  527. {
  528. DWORD dwStyle;
  529. RECT rect;
  530. SetRectEmpty(lprc);
  531. dwStyle = GetWindowLong(psb->hwnd, GWL_STYLE);
  532. if( (dwStyle & WS_HSCROLL) && (dwStyle & WS_VSCROLL) ) {
  533. GetClientRect(psb->hwnd, &rect);
  534. ClientToScreen(psb->hwnd, (LPPOINT)&rect);
  535. ClientToScreen(psb->hwnd, ((LPPOINT)&rect)+1);
  536. // calculate left scroll style of size box rect
  537. if( psb->fLeftScrollBar ) {
  538. lprc->left = rect.left - GetSystemMetrics(SM_CXVSCROLL);
  539. lprc->right = rect.left;
  540. }
  541. else {
  542. lprc->left = rect.right;
  543. lprc->right = rect.right + GetSystemMetrics(SM_CXVSCROLL);
  544. }
  545. lprc->top = rect.bottom;
  546. lprc->bottom= rect.bottom + GetSystemMetrics(SM_CYHSCROLL);
  547. return TRUE;
  548. }
  549. return FALSE;
  550. }
  551. //----------------------------------------------------------
  552. // Name : SkinSB_GetScrollBarRect()
  553. // Desc : Get the scrollbar rect
  554. //----------------------------------------------------------
  555. BOOL SkinSB_GetScrollBarRect(LPSB psb, BOOL fVert, LPRECT lprc)
  556. {
  557. RECT rect;
  558. DWORD dwStyle;
  559. if( !psb || !lprc )
  560. return FALSE;
  561. SetRectEmpty(lprc);
  562. GetClientRect(psb->hwnd, &rect);
  563. ClientToScreen(psb->hwnd, (LPPOINT)&rect);
  564. ClientToScreen(psb->hwnd, ((LPPOINT)&rect)+1);
  565. dwStyle = GetWindowLong(psb->hwnd, GWL_STYLE);
  566. if( fVert ) {
  567. if( psb->fLeftScrollBar ) {
  568. lprc->right = lprc->left = rect.left;
  569. if( dwStyle & WS_VSCROLL )
  570. lprc->left -= GetSystemMetrics(SM_CXVSCROLL);
  571. }
  572. else {
  573. lprc->right = lprc->left = rect.right;
  574. if( dwStyle & WS_VSCROLL )
  575. lprc->right += GetSystemMetrics(SM_CXVSCROLL);
  576. }
  577. lprc->top = rect.top;
  578. lprc->bottom = rect.bottom;
  579. }
  580. else {
  581. lprc->top = lprc->bottom = rect.bottom;
  582. if( dwStyle & WS_HSCROLL )
  583. lprc->bottom += GetSystemMetrics(SM_CYHSCROLL);
  584. lprc->left = rect.left;
  585. lprc->right = rect.right;
  586. }
  587. return TRUE;
  588. }
  589. //----------------------------------------------------------
  590. // Name : SkinSB_GetThumbRect()
  591. // Desc :
  592. //----------------------------------------------------------
  593. BOOL SkinSB_GetThumbRect(LPSB psb, LPRECT lprc, BOOL fVert)
  594. {
  595. if( !psb || !lprc )
  596. return FALSE;
  597. SBCALC sbc;
  598. RECT rect;
  599. SkinSB_SBCalc(psb, &sbc, fVert);
  600. GetWindowRect(psb->hwnd, &rect);
  601. if( fVert )
  602. SetRect(lprc, sbc.pxLeft, sbc.pxThumbTop, sbc.pxRight, sbc.pxThumbBottom);
  603. else
  604. SetRect(lprc, sbc.pxThumbLeft, sbc.pxTop, sbc.pxThumbRight, sbc.pxBottom);
  605. OffsetRect(lprc, -rect.left, -rect.top);
  606. return TRUE;
  607. }
  608. //----------------------------------------------------------
  609. // Name : SkinSB_GetGrooveRect()
  610. // Desc :
  611. //----------------------------------------------------------
  612. BOOL SkinSB_GetGrooveRect(LPSB psb, LPRECT lprc, BOOL fVert)
  613. {
  614. if( !psb || !lprc )
  615. return FALSE;
  616. SBCALC sbc;
  617. RECT rect;
  618. SkinSB_SBCalc(psb, &sbc, fVert);
  619. GetWindowRect(psb->hwnd, &rect);
  620. if( fVert )
  621. SetRect(lprc, sbc.pxLeft, sbc.pxUpArrow, sbc.pxRight, sbc.pxDownArrow);
  622. else
  623. SetRect(lprc, sbc.pxUpArrow, sbc.pxTop, sbc.pxDownArrow, sbc.pxBottom);
  624. OffsetRect(lprc, -rect.left, -rect.top);
  625. return TRUE;
  626. }
  627. //----------------------------------------------------------
  628. // Name : SkinSB_DrawSizeBox()
  629. // Desc : Draw the size box
  630. //----------------------------------------------------------
  631. BOOL SkinSB_DrawSizeBox(LPSB psb, HDC hDC)
  632. {
  633. HDC hMemDC;
  634. HBITMAP hOldBmp;
  635. RECT rect, rc;
  636. GetWindowRect(psb->hwnd, &rect);
  637. SkinSB_GetSizeBoxRect(psb, &rc);
  638. OffsetRect(&rc, -rect.left, -rect.top);
  639. hMemDC = CreateCompatibleDC(NULL);
  640. hOldBmp = (HBITMAP)SelectObject(hMemDC, psb->hBmp);
  641. BitBlt(hDC, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, hMemDC, 38, 57, SRCCOPY);
  642. // Release GDI object
  643. SelectObject(hMemDC, hOldBmp);
  644. DeleteDC(hMemDC);
  645. return TRUE;
  646. }
  647. //----------------------------------------------------------
  648. // Name : SkinSB_DrawGroove()
  649. // Desc :
  650. //----------------------------------------------------------
  651. BOOL SkinSB_DrawGroove(LPSB psb, HDC hdc, LPRECT lprc, BOOL fVert)
  652. {
  653. if( !hdc || !lprc || IsRectEmpty(lprc) )
  654. return FALSE;
  655. HDC hbmpDC;
  656. HBITMAP hOldBmp;
  657. POINT pt;
  658. int nMode;
  659. hbmpDC = CreateCompatibleDC(hdc);
  660. hOldBmp = (HBITMAP)SelectObject(hbmpDC, psb->hBmp);
  661. if( fVert )
  662. pt = ptArray[0][4];
  663. else
  664. pt = ptArray[3][4];
  665. nMode = SetStretchBltMode(hdc, HALFTONE);
  666. StretchBlt(hdc, lprc->left, lprc->top, lprc->right - lprc->left,
  667. lprc->bottom - lprc->top, hbmpDC, pt.x, pt.y, 16, 16, SRCCOPY);
  668. SetStretchBltMode(hdc, nMode);
  669. SelectObject(hbmpDC, hOldBmp);
  670. DeleteDC(hbmpDC);
  671. return TRUE;
  672. }
  673. //----------------------------------------------------------
  674. // Name : SkinSB_DrawScrollBar()
  675. // Desc : Draw the scrollbar
  676. //----------------------------------------------------------
  677. void SkinSB_DrawScrollBar(LPSB psb, HDC hDC, BOOL fVert)
  678. {
  679. RECT rcGroove;
  680. SkinSB_GetGrooveRect(psb, &rcGroove, fVert);
  681. SkinSB_DrawGroove(psb, hDC, &rcGroove, fVert);
  682. if( fVert ) {
  683. // Draw the arrow
  684. SkinSB_DrawArrow(psb, hDC, fVert, HTSCROLL_LINEUP, 0);
  685. SkinSB_DrawArrow(psb, hDC, fVert, HTSCROLL_LINEDOWN, 0);
  686. // Draw the thumb
  687. if( !SkinSB_IsScrollInfoActive(&psb->Vert) )
  688. return ;
  689. SkinSB_DrawThumb(psb, hDC, fVert);
  690. }
  691. else {
  692. SkinSB_DrawArrow(psb, hDC, fVert, HTSCROLL_LINEUP, 0);
  693. SkinSB_DrawArrow(psb, hDC, fVert, HTSCROLL_LINEDOWN, 0);
  694. if( !SkinSB_IsScrollInfoActive(&psb->Horz) )
  695. return ;
  696. SkinSB_DrawThumb(psb, hDC, fVert);
  697. }
  698. }
  699. //----------------------------------------------------------
  700. // Name : SkinSB_DrawArrow()
  701. // Desc : Draw the scrollbar arrow button
  702. //----------------------------------------------------------
  703. BOOL SkinSB_DrawArrow(LPSB psb, HDC hdc, BOOL fVert, int nArrow, UINT uState)
  704. {
  705. RECT rect, rc;
  706. SBCALC sbc;
  707. HDC hMemDC;
  708. HBITMAP hOldBmp;
  709. int x, y;
  710. POINT pt;
  711. GetWindowRect(psb->hwnd, &rect);
  712. SkinSB_SBCalc(psb, &sbc, fVert);
  713. if( uState == 0 )
  714. uState = SkinSB_GetState(psb, fVert, nArrow);
  715. hMemDC = CreateCompatibleDC(hdc);
  716. hOldBmp = (HBITMAP)SelectObject(hMemDC, psb->hBmp);
  717. switch( nArrow )
  718. {
  719. // up or left arrow
  720. case HTSCROLL_LINEUP:
  721. if( fVert ) {
  722. x = 0; y = 0;
  723. SetRect(&rc, sbc.pxLeft, sbc.pxTop, sbc.pxRight, sbc.pxUpArrow);
  724. }
  725. else {
  726. x = 3; y = 0;
  727. SetRect(&rc, sbc.pxLeft, sbc.pxTop, sbc.pxUpArrow, sbc.pxBottom);
  728. }
  729. break;
  730. // right or down arrow
  731. case HTSCROLL_LINEDOWN:
  732. if( fVert ) {
  733. x = 0; y = 1;
  734. SetRect(&rc, sbc.pxLeft, sbc.pxDownArrow, sbc.pxRight, sbc.pxBottom);
  735. }
  736. else {
  737. x = 3; y = 1;
  738. SetRect(&rc, sbc.pxDownArrow, sbc.pxTop, sbc.pxRight, sbc.pxBottom);
  739. }
  740. break;
  741. }
  742. OffsetRect(&rc, -rect.left, -rect.top);
  743. switch( uState )
  744. {
  745. case SB_STATE_NORMAL:
  746. break;
  747. case SB_STATE_HOTTRACKED:
  748. x += 1;
  749. break;
  750. case SB_STATE_PRESSED:
  751. x += 2;
  752. break;
  753. case SB_STATE_DISABLED:
  754. break;
  755. }
  756. pt = ptArray[x][y];
  757. BitBlt(hdc, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top,
  758. hMemDC, pt.x, pt.y, SRCCOPY);
  759. SelectObject(hMemDC, hOldBmp);
  760. DeleteDC(hMemDC);
  761. return TRUE;
  762. }
  763. //----------------------------------------------------------
  764. // Name : SkinSB_DrawThumb()
  765. // Desc : Draw the thumb
  766. //----------------------------------------------------------
  767. void SkinSB_DrawThumb(LPSB psb, HDC hdc, BOOL fVert)
  768. {
  769. SBCALC sbc;
  770. RECT rc, rect;
  771. HDC hSrcDC;
  772. HDC memDC;
  773. HBITMAP hOldBmp;
  774. HBITMAP hMemBitmap, hOldBitmap;
  775. UINT uState;
  776. int cx, cy;
  777. POINT pt, pt1;
  778. uState = SkinSB_GetState(psb, fVert, HTSCROLL_THUMB);
  779. SkinSB_SBCalc(psb, &sbc, fVert);
  780. GetWindowRect(psb->hwnd, &rect);
  781. // Chack scrollbar of size whether valid
  782. if( sbc.pxTop >= sbc.pxBottom || sbc.pxLeft >= sbc.pxRight )
  783. return ;
  784. // Draw the groove only
  785. if( (sbc.pxDownArrow - sbc.pxUpArrow) < (sbc.pxThumbBottom - sbc.pxThumbTop) ||
  786. uState == SB_STATE_DISABLED ) {
  787. SkinSB_GetGrooveRect(psb, &rc, fVert);
  788. SkinSB_DrawGroove(psb, hdc, &rc, fVert);
  789. return ;
  790. }
  791. // Padding the groove part
  792. if( sbc.pxUpArrow < sbc.pxThumbTop ) {
  793. if( fVert )
  794. SetRect(&rc, sbc.pxLeft, sbc.pxUpArrow, sbc.pxRight, sbc.pxThumbTop);
  795. else
  796. SetRect(&rc, sbc.pxUpArrow, sbc.pxTop, sbc.pxThumbTop, sbc.pxBottom);
  797. OffsetRect(&rc, -rect.left, -rect.top);
  798. SkinSB_DrawGroove(psb, hdc, &rc, fVert);
  799. }
  800. if( sbc.pxThumbBottom < sbc.pxDownArrow ) {
  801. if( fVert )
  802. SetRect(&rc, sbc.pxLeft, sbc.pxThumbBottom, sbc.pxRight, sbc.pxDownArrow);
  803. else
  804. SetRect(&rc, sbc.pxThumbBottom, sbc.pxTop, sbc.pxDownArrow, sbc.pxBottom);
  805. OffsetRect(&rc, -rect.left, -rect.top);
  806. SkinSB_DrawGroove(psb, hdc, &rc, fVert);
  807. }
  808. // Draw the thumb use memory dc
  809. // Select scrollbar bitmap resource to dc
  810. hSrcDC = CreateCompatibleDC(hdc);
  811. hOldBmp = (HBITMAP)SelectObject(hSrcDC, psb->hBmp);
  812. // get the thumb rectangle
  813. SkinSB_GetThumbRect(psb, &rc, fVert);
  814. cx = rc.right - rc.left;
  815. cy = rc.bottom - rc.top;
  816. RECT rcMemDC = {0, 0, rc.right-rc.left, rc.bottom-rc.top};
  817. // create the memory dc
  818. memDC = CreateCompatibleDC(hdc);
  819. hMemBitmap = CreateCompatibleBitmap(hdc, cx, cy);
  820. hOldBitmap = (HBITMAP)SelectObject(memDC, hMemBitmap);
  821. SetBkColor(memDC, 0xFFFFFF);
  822. ExtTextOut(memDC, 0, 0, ETO_OPAQUE, &rcMemDC, NULL, 0, NULL);
  823. // Select state bitmap part
  824. switch( uState )
  825. {
  826. case SB_STATE_NORMAL:
  827. pt = fVert ? ptArray[0][2] : ptArray[3][2];
  828. pt1 = fVert ? ptArray[0][3] : ptArray[3][3];
  829. break;
  830. case SB_STATE_HOTTRACKED:
  831. pt = fVert ? ptArray[1][2] : ptArray[4][2];
  832. pt1 = fVert ? ptArray[1][3] : ptArray[4][3];
  833. break;
  834. case SB_STATE_PRESSED:
  835. pt = fVert ? ptArray[2][2] : ptArray[5][2];
  836. pt1 = fVert ? ptArray[2][3] : ptArray[5][3];
  837. break;
  838. case SB_STATE_DISABLED:
  839. break;
  840. }
  841. if ( fVert ) {
  842. for( int i= 4; i < cy - 4; i += 8 )
  843. BitBlt(memDC, 0, i, cx, 8, hSrcDC, pt.x, pt.y + 4, SRCCOPY);
  844. BitBlt(memDC, 0, 0, cx, 4, hSrcDC, pt.x, pt.y, SRCCOPY);
  845. BitBlt(memDC, 0, cy - 4, cx, 4, hSrcDC, pt.x, (pt.y + 16) - 4, SRCCOPY);
  846. if( cy > 16 + 8 ) {
  847. int y = (cy - 16) / 2;
  848. BitBlt(memDC, 0, y, cx, 16, hSrcDC, pt1.x, pt1.y, SRCCOPY);
  849. }
  850. }
  851. else {
  852. for( int i=4; i < cx - 4; i += 8 )
  853. BitBlt(memDC, i, 0, 8, cy, hSrcDC, pt.x + 4, pt.y, SRCCOPY);
  854. BitBlt(memDC, 0, 0, 4, cy, hSrcDC, pt.x, pt.y, SRCCOPY);
  855. BitBlt(memDC, cx - 4, 0, 4, cy, hSrcDC, (pt.x + 16) - 4, pt.y, SRCCOPY);
  856. if( cx > 16 + 8 ) {
  857. int x = (cx - 16) / 2;
  858. BitBlt(memDC, x, 0, 16, cy, hSrcDC, pt1.x, pt1.y, SRCCOPY);
  859. }
  860. }
  861. BitBlt(hdc, rc.left, rc.top, rc.right-rc.left, rc.bottom-rc.top, memDC, 0,0, SRCCOPY);
  862. // Release the memory DC
  863. SelectObject(memDC, hOldBitmap);
  864. DeleteDC(memDC);
  865. DeleteObject(hMemBitmap);
  866. SelectObject(hSrcDC, hOldBmp);
  867. DeleteDC(hSrcDC);
  868. }
  869. //----------------------------------------------------------
  870. // Name : SkinSB_HitTest()
  871. // Desc : HitTest scrollbar items
  872. //----------------------------------------------------------
  873. UINT SkinSB_HitTest(LPSB psb, BOOL fVert, POINT pt)
  874. {
  875. SBCALC sbc;
  876. UINT disFlags;
  877. int x;
  878. SkinSB_SBCalc(psb, &sbc, fVert);
  879. if( pt.x < sbc.pxLeft || pt.y < sbc.pxTop ||
  880. pt.x > sbc.pxRight || pt.y > sbc.pxBottom )
  881. return HTSCROLL_NONE;
  882. disFlags = SkinSB_GetDisableFlags(psb, fVert);
  883. x = fVert ? pt.y : pt.x;
  884. if( x < sbc.pxUpArrow ) {
  885. if( disFlags & ESB_DISABLE_BOTH || disFlags & ESB_DISABLE_LTUP )
  886. return HTSCROLL_NONE;
  887. else
  888. return HTSCROLL_LINEUP;
  889. }
  890. else if( x > sbc.pxUpArrow && x < sbc.pxThumbTop ) {
  891. if( disFlags & ESB_DISABLE_BOTH || disFlags & ESB_DISABLE_LTUP )
  892. return HTSCROLL_NONE;
  893. else
  894. return HTSCROLL_PAGEUP;
  895. }
  896. else if( (x >= sbc.pxThumbTop && x <= sbc.pxThumbBottom) &&
  897. (sbc.pxThumbTop > 0 && sbc.pxThumbBottom > sbc.pxThumbTop) ) {
  898. return HTSCROLL_THUMB;
  899. }
  900. else if( x > sbc.pxThumbBottom && x < sbc.pxDownArrow ) {
  901. if( disFlags & ESB_DISABLE_BOTH || disFlags & ESB_DISABLE_RTDN )
  902. return HTSCROLL_NONE;
  903. else
  904. return HTSCROLL_PAGEDOWN;
  905. }
  906. else if( x >= sbc.pxDownArrow ) {
  907. if( disFlags & ESB_DISABLE_BOTH || disFlags & ESB_DISABLE_RTDN )
  908. return HTSCROLL_NONE;
  909. else
  910. return HTSCROLL_LINEDOWN;
  911. }
  912. return HTSCROLL_NONE;
  913. }
  914. //----------------------------------------------------------
  915. // Name : SkinSB_SBCalc()
  916. // Desc : Calc the scrollbar items position and size
  917. //----------------------------------------------------------
  918. void SkinSB_SBCalc(LPSB psb, LPSBCALC lpcalc, BOOL fVert)
  919. {
  920. RECT rcBar; // the scrollbar rect
  921. int nRange; // the scroll range
  922. int nWorkingsize; // the scroll working size
  923. int nArrowsize; // the arrow button of size
  924. int nThumbsize; // the thumb of size
  925. int nStart; // the thumb of start position
  926. int nThumbpos; // the thumn of current position
  927. SCROLLINFO* psi;
  928. SkinSB_GetScrollBarRect(psb, fVert, &rcBar);
  929. lpcalc->pxLeft = rcBar.left;
  930. lpcalc->pxTop = rcBar.top;
  931. lpcalc->pxRight = rcBar.right;
  932. lpcalc->pxBottom = rcBar.bottom;
  933. if( fVert ) {
  934. psi = &psb->Vert;
  935. nArrowsize = GetSystemMetrics(SM_CYVSCROLL);
  936. nWorkingsize = (rcBar.bottom - rcBar.top) - nArrowsize * 2;
  937. nStart = rcBar.top + nArrowsize;
  938. lpcalc->pxUpArrow = rcBar.top + nArrowsize;
  939. lpcalc->pxDownArrow = rcBar.bottom - nArrowsize;
  940. }
  941. else {
  942. psi = &psb->Horz;
  943. nArrowsize = GetSystemMetrics(SM_CXHSCROLL);
  944. nWorkingsize = (rcBar.right - rcBar.left) - nArrowsize * 2;
  945. nStart = rcBar.left + nArrowsize;
  946. lpcalc->pxUpArrow = rcBar.left + nArrowsize;
  947. lpcalc->pxDownArrow = rcBar.right - nArrowsize;
  948. }
  949. nRange = (psi->nMax - psi->nMin) + 1;
  950. if( nRange > 0 && SkinSB_IsScrollInfoActive(psi)) {
  951. nThumbsize = MulDiv(psi->nPage, nWorkingsize, nRange);
  952. if( nThumbsize < SB_MINTHUMB_SIZE )
  953. nThumbsize = SB_MINTHUMB_SIZE;
  954. int pagesize = max(1, psi->nPage);
  955. nThumbpos = MulDiv(psi->nPos - psi->nMin, nWorkingsize - nThumbsize, nRange - pagesize);
  956. if( nThumbpos < 0 )
  957. nThumbpos = 0;
  958. if( nThumbpos >= nWorkingsize - nThumbsize )
  959. nThumbpos = nWorkingsize - nThumbsize;
  960. nThumbpos += nStart;
  961. lpcalc->pxThumbTop = nThumbpos;
  962. lpcalc->pxThumbBottom = nThumbpos + nThumbsize;
  963. }
  964. else {
  965. lpcalc->pxThumbTop = 0;
  966. lpcalc->pxThumbBottom = 0;
  967. }
  968. }
  969. //---------------------------------------------------------
  970. // Name : SkinSB_TrackThumb()
  971. // Desc : Track the scroll thumb
  972. //---------------------------------------------------------
  973. BOOL SkinSB_TrackThumb(LPSB psb, BOOL fVert, POINT pt)
  974. {
  975. SBCALC sbclc;
  976. int nPos;
  977. int nThumbpos;
  978. int nRange;
  979. int nThumbsize;
  980. int nWorksize;
  981. LPSCROLLINFO psi;
  982. SkinSB_SBCalc(psb, &sbclc, fVert);
  983. if( fVert ) {
  984. psi = &psb->Vert;
  985. nThumbpos = pt.y - psb->nOffsetPoint;
  986. }
  987. else {
  988. psi = &psb->Horz;
  989. nThumbpos = pt.x - psb->nOffsetPoint;
  990. }
  991. nPos = 0;
  992. nThumbpos -= sbclc.pxUpArrow;
  993. nThumbsize = sbclc.pxThumbBottom - sbclc.pxThumbTop;
  994. nWorksize = sbclc.pxDownArrow - sbclc.pxUpArrow;
  995. nRange = (psi->nMax - psi->nMin) + 1;
  996. if( nThumbpos < 0 )
  997. nThumbpos = 0;
  998. if( nThumbpos > nWorksize - nThumbsize)
  999. nThumbpos = nWorksize - nThumbsize;
  1000. if( nRange > 0 )
  1001. nPos = MulDiv(nThumbpos, nRange - psi->nPage, nWorksize - nThumbsize);
  1002. // Send the scroll message to window !!!
  1003. if( psi->nPos != nPos ) {
  1004. psi->nTrackPos = nPos;
  1005. psb->nTrackPos = nPos;
  1006. // NOTE: 2008-12-24
  1007. // Not use the SB_THUMBTRACK flag that because of
  1008. // the RichEdit cannot receving the WM_MOUSEMOVE message
  1009. if( psb->fRichEdit ) {
  1010. DoScrollMsg(psb->hwnd, /*SB_THUMBTRACK*/SB_THUMBPOSITION, nPos, fVert);
  1011. }
  1012. else {
  1013. DoScrollMsg(psb->hwnd, SB_THUMBTRACK, nPos, fVert);
  1014. }
  1015. }
  1016. return TRUE;
  1017. }
  1018. //---------------------------------------------------------
  1019. // Name : SkinSB_GetState()
  1020. // Desc :
  1021. //---------------------------------------------------------
  1022. UINT SkinSB_GetState(LPSB psb, BOOL fVert, UINT nHit)
  1023. {
  1024. BOOL fHotTracked;
  1025. BOOL fPressed;
  1026. BOOL fDisabled;
  1027. UINT disFlags;
  1028. UINT state;
  1029. if( nHit == HTSCROLL_NONE )
  1030. return 0;
  1031. fHotTracked = fPressed = fDisabled = FALSE;
  1032. disFlags = SkinSB_GetDisableFlags(psb, fVert);
  1033. switch( nHit ) {
  1034. case HTSCROLL_LINEUP:
  1035. fDisabled = (disFlags & ESB_DISABLE_BOTH || disFlags & ESB_DISABLE_UP);
  1036. break;
  1037. case HTSCROLL_LINEDOWN:
  1038. fDisabled = (disFlags & ESB_DISABLE_BOTH || disFlags & ESB_DISABLE_DOWN);
  1039. break;
  1040. case HTSCROLL_THUMB:
  1041. fDisabled = (disFlags & ESB_DISABLE_BOTH);
  1042. break;
  1043. }
  1044. if( nHit == psb->nLastCode && fVert == psb->fLastVert )
  1045. fHotTracked = TRUE;
  1046. if( !fDisabled && psb->fTracking && fHotTracked )
  1047. fPressed = TRUE;
  1048. if( fDisabled )
  1049. state = SB_STATE_DISABLED;
  1050. else if( fPressed )
  1051. state = SB_STATE_PRESSED;
  1052. else if( fHotTracked )
  1053. state = SB_STATE_HOTTRACKED;
  1054. else
  1055. state = SB_STATE_NORMAL;
  1056. return state;
  1057. }
  1058. //----------------------------------------------------------
  1059. // Name : SkinSB_Track()
  1060. // Desc :
  1061. //----------------------------------------------------------
  1062. void SkinSB_Track(LPSB psb, BOOL fVert, UINT nHit, POINT pt)
  1063. {
  1064. UINT disFlags;
  1065. LPSCROLLINFO psi;
  1066. WORD wSBCode;
  1067. psi = fVert ? &psb->Vert : &psb->Horz;
  1068. disFlags = SkinSB_GetDisableFlags(psb, fVert);
  1069. switch( nHit )
  1070. {
  1071. case HTSCROLL_THUMB:
  1072. SBCALC sbclc;
  1073. SkinSB_SBCalc(psb, &sbclc, fVert);
  1074. psi->nTrackPos = psi->nPos;
  1075. psb->nOffsetPoint = (fVert ? pt.y : pt.x) - sbclc.pxThumbTop;
  1076. break;
  1077. case HTSCROLL_LINEUP:
  1078. wSBCode = SB_LINEUP;
  1079. goto DoScroll;
  1080. case HTSCROLL_LINEDOWN:
  1081. wSBCode = SB_LINEDOWN;
  1082. goto DoScroll;
  1083. case HTSCROLL_PAGEDOWN:
  1084. wSBCode = SB_PAGEDOWN;
  1085. goto DoScroll;
  1086. case HTSCROLL_PAGEUP:
  1087. wSBCode = SB_PAGEUP;
  1088. DoScroll:
  1089. psb->nScrollTimerMsg = MAKELONG(fVert ? WM_VSCROLL : WM_HSCROLL, wSBCode);
  1090. DoScrollMsg(psb->hwnd, wSBCode, 0, fVert);
  1091. SetTimer(psb->hwnd, SB_TIMER_DELAY, SB_SCROLL_DELAY, NULL);
  1092. break;
  1093. default:
  1094. return;
  1095. }
  1096. psb->nTrackCode = nHit;
  1097. psb->fTrackVert = fVert;
  1098. psb->fTracking = TRUE;
  1099. SkinSB_HotTrack(psb, nHit, fVert, TRUE);
  1100. SetCapture(psb->hwnd);
  1101. //SkinSB_TrackLoop(psb);
  1102. }
  1103. /*
  1104. //----------------------------------------------------------
  1105. // Name : SkinSB_TrackLoop()
  1106. // Desc :
  1107. //----------------------------------------------------------
  1108. void SkinSB_TrackLoop(LPSB psb)
  1109. {
  1110. HWND hwnd;
  1111. MSG msg;
  1112. int cmd;
  1113. //POINT pt;
  1114. if( !psb->fTracking )
  1115. return;
  1116. hwnd = psb->hwnd;
  1117. while( GetCapture() == hwnd )
  1118. {
  1119. if( !GetMessage(&msg, hwnd, 0, 0) )
  1120. break;
  1121. if( !CallMsgFilter(&msg, MSGF_SCROLLBAR) )
  1122. {
  1123. cmd = msg.message;
  1124. if( msg.hwnd == hwnd &&
  1125. ((cmd >= WM_MOUSEFIRST && cmd <= WM_MOUSELAST) ||
  1126. (cmd >= WM_KEYFIRST && cmd <= WM_KEYLAST))
  1127. )
  1128. {
  1129. TRACE("message loop\n");
  1130. SkinSB_Proc(hwnd, cmd, msg.wParam, msg.lParam);
  1131. }
  1132. }
  1133. else
  1134. {
  1135. TranslateMessage(&msg);
  1136. DispatchMessage(&msg);
  1137. }
  1138. }
  1139. }*/
  1140. //----------------------------------------------------------
  1141. // Name : SkinSB_HotTrack()
  1142. // Desc : Hot scrollbar arrow and thumb items
  1143. //----------------------------------------------------------
  1144. BOOL SkinSB_HotTrack(LPSB psb, int nHitCode, BOOL fVert, BOOL fMouseDown)
  1145. {
  1146. HDC hDC;
  1147. UINT oldHit;
  1148. BOOL oldVert;
  1149. // Save old hittest code
  1150. oldHit = psb->nLastCode;
  1151. oldVert = psb->fLastVert;
  1152. psb->nLastCode = nHitCode;
  1153. psb->fLastVert = fVert;
  1154. if( (hDC = GetWindowDC(psb->hwnd)) == NULL )
  1155. return FALSE;
  1156. if( nHitCode != oldHit || fVert != oldVert || fMouseDown) {
  1157. // Restore old hittest item state
  1158. if( oldHit == HTSCROLL_LINEUP )
  1159. SkinSB_DrawArrow(psb, hDC, oldVert, HTSCROLL_LINEUP, SB_STATE_NORMAL);
  1160. else if( oldHit == HTSCROLL_LINEDOWN )
  1161. SkinSB_DrawArrow(psb, hDC, oldVert, HTSCROLL_LINEDOWN, SB_STATE_NORMAL);
  1162. else if( oldHit == HTSCROLL_THUMB )
  1163. SkinSB_DrawThumb(psb, hDC, oldVert);
  1164. // Draw new hittest item state
  1165. if( nHitCode == HTSCROLL_LINEUP )
  1166. SkinSB_DrawArrow(psb, hDC, fVert, HTSCROLL_LINEUP, fMouseDown ? SB_STATE_PRESSED : SB_STATE_HOTTRACKED);
  1167. else if( nHitCode == HTSCROLL_LINEDOWN )
  1168. SkinSB_DrawArrow(psb, hDC, fVert, HTSCROLL_LINEDOWN, fMouseDown ? SB_STATE_PRESSED : SB_STATE_HOTTRACKED);
  1169. else if( nHitCode == HTSCROLL_THUMB)
  1170. SkinSB_DrawThumb(psb, hDC, fVert);
  1171. }
  1172. ReleaseDC(psb->hwnd, hDC);
  1173. return TRUE;
  1174. }
  1175. // Message handle
  1176. //----------------------------------------------------------
  1177. // Name : SkinSB_OnStyleChanged()
  1178. // Desc :
  1179. //----------------------------------------------------------
  1180. LRESULT SkinSB_OnStyleChanged(LPSB psb, int nStyleType, LPSTYLESTRUCT lpStyleStruct)
  1181. {
  1182. if( psb->fPreventStyleChange )
  1183. return 0;
  1184. if( nStyleType == GWL_EXSTYLE ) {
  1185. BOOL fOld = psb->fLeftScrollBar;
  1186. if( lpStyleStruct->styleNew & WS_EX_LEFTSCROLLBAR )
  1187. psb->fLeftScrollBar = TRUE;
  1188. else
  1189. psb->fLeftScrollBar = FALSE;
  1190. if( fOld != psb->fLeftScrollBar )
  1191. SetWindowPos(psb->hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER |
  1192. SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_DRAWFRAME);
  1193. }
  1194. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_STYLECHANGED, nStyleType, (LPARAM)lpStyleStruct);
  1195. }
  1196. //----------------------------------------------------------
  1197. // Name : SkinSB_OnNcCalcSize()
  1198. // Desc :
  1199. //----------------------------------------------------------
  1200. LRESULT SkinSB_OnNcCalcSize(LPSB psb, BOOL bCalcValidRects, NCCALCSIZE_PARAMS* lpncsp)
  1201. {
  1202. LRESULT lr;
  1203. DWORD dwStyle;
  1204. RECT* lprc, rect;
  1205. lprc = &lpncsp->rgrc[0];
  1206. rect = *lprc;
  1207. dwStyle = GetWindowLong(psb->hwnd, GWL_STYLE);
  1208. if( dwStyle & (WS_VSCROLL | WS_HSCROLL) ) {
  1209. psb->fPreventStyleChange = TRUE;
  1210. SetWindowLong(psb->hwnd, GWL_STYLE, dwStyle & ~(WS_VSCROLL|WS_HSCROLL));
  1211. }
  1212. lr = CallWindowProc(psb->pfnOldProc, psb->hwnd,
  1213. WM_NCCALCSIZE, (WPARAM)bCalcValidRects, (LPARAM)lpncsp);
  1214. if( dwStyle & (WS_VSCROLL | WS_HSCROLL) ) {
  1215. SetWindowLong(psb->hwnd, GWL_STYLE, dwStyle);
  1216. psb->fPreventStyleChange = FALSE;
  1217. }
  1218. if( (dwStyle & WS_HSCROLL) &&
  1219. (lprc->bottom - lprc->top) > GetSystemMetrics(SM_CYHSCROLL) ) {
  1220. lprc->bottom -= GetSystemMetrics(SM_CYHSCROLL);
  1221. }
  1222. if( (dwStyle & WS_VSCROLL) &&
  1223. (lprc->right - lprc->left) > GetSystemMetrics(SM_CXVSCROLL) ) {
  1224. if( psb->fLeftScrollBar )
  1225. lprc->left += GetSystemMetrics(SM_CXVSCROLL);
  1226. else
  1227. lprc->right -= GetSystemMetrics(SM_CXVSCROLL);
  1228. }
  1229. return lr;
  1230. }
  1231. //----------------------------------------------------------
  1232. // Name : SkinSB_OnNcPaint()
  1233. // Desc :
  1234. //----------------------------------------------------------
  1235. LRESULT SkinSB_OnNcPaint(LPSB psb, HRGN hRgn)
  1236. {
  1237. LRESULT lr;
  1238. DWORD dwStyle;
  1239. HDC hDC;
  1240. dwStyle = GetWindowLong(psb->hwnd, GWL_STYLE);
  1241. if( dwStyle & (WS_VSCROLL | WS_HSCROLL) ) {
  1242. psb->fPreventStyleChange = TRUE;
  1243. SetWindowLong(psb->hwnd, GWL_STYLE, dwStyle & ~(WS_VSCROLL|WS_HSCROLL));
  1244. }
  1245. // draw frame border by system
  1246. lr = CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_NCPAINT, (WPARAM)hRgn, 0);
  1247. if( dwStyle & (WS_VSCROLL | WS_HSCROLL) ) {
  1248. SetWindowLong(psb->hwnd, GWL_STYLE, dwStyle);
  1249. psb->fPreventStyleChange = FALSE;
  1250. }
  1251. //hDC = GetDCEx(psb->hwnd, hRgn, DCX_WINDOW|DCX_INTERSECTRGN|DCX_CACHE );
  1252. hDC = GetWindowDC(psb->hwnd);
  1253. // draw the size box
  1254. if( dwStyle & (WS_VSCROLL | WS_HSCROLL) )
  1255. SkinSB_DrawSizeBox(psb, hDC);
  1256. // Draw scrollbar
  1257. if( dwStyle & WS_VSCROLL )
  1258. SkinSB_DrawScrollBar(psb, hDC, TRUE);
  1259. if( dwStyle & WS_HSCROLL )
  1260. SkinSB_DrawScrollBar(psb, hDC, FALSE);
  1261. ReleaseDC(psb->hwnd, hDC);
  1262. return lr;
  1263. }
  1264. //----------------------------------------------------------
  1265. // Name : SkinSB_OnNcHitTest()
  1266. // Desc :
  1267. //----------------------------------------------------------
  1268. LRESULT SkinSB_OnNcHitTest(LPSB psb, WPARAM wParam, LPARAM lParam)
  1269. {
  1270. POINT pt;
  1271. RECT rcHorz, rcVert, rcSize;
  1272. pt.x = GET_X_LPARAM(lParam);
  1273. pt.y = GET_Y_LPARAM(lParam);
  1274. SkinSB_GetScrollBarRect(psb, TRUE, &rcVert);
  1275. SkinSB_GetScrollBarRect(psb, FALSE, &rcHorz);
  1276. SkinSB_GetSizeBoxRect(psb, &rcSize);
  1277. if( PtInRect(&rcVert, pt) )
  1278. return HTVSCROLL;
  1279. else if( PtInRect(&rcHorz, pt) )
  1280. return HTHSCROLL;
  1281. else if( PtInRect(&rcSize, pt) )
  1282. return HTSIZE;
  1283. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_NCHITTEST, wParam, lParam);
  1284. }
  1285. //----------------------------------------------------------
  1286. // Name : SkinSB_OnNcMouseMove()
  1287. // Desc :
  1288. //----------------------------------------------------------
  1289. LRESULT SkinSB_OnNcMouseMove(LPSB psb, WPARAM wParam, LPARAM lParam)
  1290. {
  1291. if( wParam == HTHSCROLL || wParam == HTVSCROLL ) {
  1292. POINT pt;
  1293. BOOL fVert;
  1294. int nHitCode;
  1295. pt.x = GET_X_LPARAM(lParam);
  1296. pt.y = GET_Y_LPARAM(lParam);
  1297. fVert = ( wParam == HTVSCROLL );
  1298. nHitCode = SkinSB_HitTest(psb, fVert, pt);
  1299. SkinSB_HotTrack(psb, nHitCode, fVert, FALSE);
  1300. }
  1301. else if( wParam == HTSIZE ) {
  1302. SkinSB_HotTrack(psb, HTSCROLL_NONE, FALSE, FALSE);
  1303. }
  1304. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_NCMOUSEMOVE, wParam, lParam);
  1305. }
  1306. //---------------------------------------------------------
  1307. // Name : SkinSB_OnNcMouseLeave()
  1308. // Desc :
  1309. //---------------------------------------------------------
  1310. LRESULT SkinSB_OnNcMouseLeave(LPSB psb, WPARAM wParam, LPARAM lParam)
  1311. {
  1312. psb->fMouseTracking = FALSE;
  1313. if( psb->fTracking )
  1314. return 0;
  1315. SkinSB_HotTrack(psb, HTSCROLL_NONE, FALSE, FALSE);
  1316. psb->nLastCode = HTSCROLL_NONE;
  1317. return 0;
  1318. }
  1319. //----------------------------------------------------------
  1320. // Name : SkinSB_OnNcLButtonDown()
  1321. // Desc :
  1322. //----------------------------------------------------------
  1323. LRESULT SkinSB_OnNcLButtonDown(LPSB psb, WPARAM wParam, LPARAM lParam)
  1324. {
  1325. if( wParam == HTHSCROLL || wParam == HTVSCROLL ) {
  1326. POINT pt;
  1327. BOOL fVert;
  1328. int nHitCode;
  1329. LPSCROLLINFO psi;
  1330. pt.x = GET_X_LPARAM(lParam);
  1331. pt.y = GET_Y_LPARAM(lParam);
  1332. fVert = (wParam == HTVSCROLL);
  1333. psi = (fVert ? &psb->Vert : &psb->Horz);
  1334. nHitCode = SkinSB_HitTest(psb, fVert, pt);
  1335. psb->nLastCode = nHitCode;
  1336. psb->fLastVert = fVert;
  1337. SkinSB_Track(psb, fVert, nHitCode, pt);
  1338. return 0;
  1339. }
  1340. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_NCLBUTTONDOWN, wParam, lParam);
  1341. }
  1342. //----------------------------------------------------------
  1343. // Name : SkinSB_OnMouseMove()
  1344. // Desc :
  1345. //----------------------------------------------------------
  1346. LRESULT SkinSB_OnMouseMove(LPSB psb, WPARAM wParam, LPARAM lParam)
  1347. {
  1348. DWORD dwPos;
  1349. POINT pt;
  1350. UINT nHitCode, nHitTest;
  1351. dwPos = GetMessagePos();
  1352. pt.x = GET_X_LPARAM(dwPos);
  1353. pt.y = GET_Y_LPARAM(dwPos);
  1354. // Mouse left button down
  1355. if( psb->fTracking && wParam & MK_LBUTTON ) {
  1356. if( psb->nTrackCode == HTSCROLL_THUMB ) {
  1357. SkinSB_TrackThumb(psb, psb->fTrackVert, pt);
  1358. return 0;
  1359. }
  1360. nHitTest = (UINT)SkinSB_OnNcHitTest(psb, 0, dwPos);
  1361. if( nHitTest == HTHSCROLL || nHitTest == HTVSCROLL ) {
  1362. BOOL fVert = (nHitTest == HTVSCROLL);
  1363. nHitCode = SkinSB_HitTest(psb, fVert, pt);
  1364. if( nHitCode != psb->nTrackCode || fVert != psb->fTrackVert) {
  1365. // Unallowed hot-track other hittest item
  1366. SkinSB_HotTrack(psb, HTSCROLL_NONE, FALSE, FALSE);
  1367. }
  1368. else {
  1369. SkinSB_HotTrack(psb, psb->nTrackCode, psb->fTrackVert, TRUE);
  1370. }
  1371. }
  1372. else {
  1373. SkinSB_HotTrack(psb, HTSCROLL_NONE, FALSE, FALSE);
  1374. }
  1375. return 0;
  1376. }
  1377. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_MOUSEMOVE, wParam, lParam);
  1378. }
  1379. //----------------------------------------------------------
  1380. // Name : SkinSB_OnLButtonUp()
  1381. // Desc :
  1382. //----------------------------------------------------------
  1383. LRESULT SkinSB_OnLButtonUp(LPSB psb, WPARAM wParam, LPARAM lParam)
  1384. {
  1385. if( psb->nTrackCode != HTSCROLL_NONE && psb->fTracking) {
  1386. // Release mouse capture
  1387. ReleaseCapture();
  1388. // End scroll
  1389. switch( psb->nTrackCode )
  1390. {
  1391. case HTSCROLL_LINEUP:
  1392. case HTSCROLL_LINEDOWN:
  1393. case HTSCROLL_PAGEUP:
  1394. case HTSCROLL_PAGEDOWN:
  1395. KillTimer(psb->hwnd, SB_TIMER_SCROLL);
  1396. DoScrollMsg(psb->hwnd, SB_ENDSCROLL, 0, psb->fTrackVert);
  1397. break;
  1398. case HTSCROLL_THUMB:
  1399. if( psb->fTracking ) {
  1400. DWORD dwPos;
  1401. POINT pt;
  1402. dwPos = GetMessagePos();
  1403. pt.x = GET_X_LPARAM(dwPos);
  1404. pt.y = GET_Y_LPARAM(dwPos);
  1405. DoScrollMsg(psb->hwnd, SB_THUMBPOSITION, psb->nTrackPos, psb->fTrackVert);
  1406. DoScrollMsg(psb->hwnd, SB_ENDSCROLL, 0, psb->fTrackVert);
  1407. psb->nLastCode = SkinSB_HitTest(psb, psb->fTrackVert, pt);
  1408. }
  1409. break;
  1410. }
  1411. // Clean the track parameters
  1412. psb->nOffsetPoint = 0;
  1413. psb->nScrollTimerMsg = MAKELONG(WM_NULL, 0);
  1414. psb->nTrackCode = HTSCROLL_NONE;
  1415. psb->fTracking = FALSE;
  1416. psb->nTrackPos = 0;
  1417. HDC hdc = GetWindowDC(psb->hwnd);
  1418. SkinSB_DrawThumb(psb, hdc, psb->fTrackVert);
  1419. ReleaseDC(psb->hwnd, hdc);
  1420. return 0;
  1421. }
  1422. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_LBUTTONUP, wParam, lParam);
  1423. }
  1424. //----------------------------------------------------------
  1425. // Name : SkinSB_OnTimer()
  1426. // Desc :
  1427. //----------------------------------------------------------
  1428. LRESULT SkinSB_OnTimer(LPSB psb, WPARAM wParam, LPARAM lParam)
  1429. {
  1430. if(wParam == SB_TIMER_SCROLL) {
  1431. // if mouse left button released then close scroll timer
  1432. if( psb->nTrackCode == HTSCROLL_NONE ) {
  1433. KillTimer(psb->hwnd, SB_TIMER_SCROLL);
  1434. return 0;
  1435. }
  1436. // Timer send scroll message
  1437. if( psb->nTrackCode == psb->nLastCode && psb->fTrackVert == psb->fLastVert )
  1438. DoScrollMsg(psb->hwnd, HIWORD(psb->nScrollTimerMsg), 0, psb->fTrackVert);
  1439. return 0;
  1440. }
  1441. else if(wParam == SB_TIMER_DELAY) {
  1442. KillTimer(psb->hwnd, SB_TIMER_DELAY);
  1443. SetTimer(psb->hwnd, SB_TIMER_SCROLL, SB_SCROLL_INTERVAL, 0);
  1444. return 0;
  1445. }
  1446. return CallWindowProc(psb->pfnOldProc, psb->hwnd, WM_TIMER, wParam, lParam);
  1447. }
  1448. //----------------------------------------------------------
  1449. // Name : SkinSB_Proc()
  1450. // Desc :
  1451. //----------------------------------------------------------
  1452. LRESULT CALLBACK SkinSB_Proc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
  1453. {
  1454. LPSB psb = SkinSB_GetSB(hwnd);
  1455. if( psb && uMsg == WM_NCMOUSEMOVE ) {
  1456. if(!psb->fMouseTracking && !psb->fTracking) {
  1457. TRACKMOUSEEVENT tme;
  1458. tme.cbSize = sizeof(tme);
  1459. tme.hwndTrack = psb->hwnd;
  1460. tme.dwFlags = TME_LEAVE|TME_NONCLIENT;
  1461. tme.dwHoverTime = 0;
  1462. psb->fMouseTracking = _TrackMouseEvent(&tme);
  1463. }
  1464. }
  1465. switch( uMsg )
  1466. {
  1467. case WM_NCPAINT:
  1468. return SkinSB_OnNcPaint(psb, (HRGN)wParam);
  1469. case WM_NCCALCSIZE:
  1470. return SkinSB_OnNcCalcSize(psb, (BOOL)wParam, (NCCALCSIZE_PARAMS*)lParam);
  1471. case WM_NCHITTEST:
  1472. return SkinSB_OnNcHitTest(psb, wParam, lParam);
  1473. case WM_STYLECHANGED:
  1474. return SkinSB_OnStyleChanged(psb, (int)wParam, (LPSTYLESTRUCT)lParam);
  1475. case WM_NCLBUTTONDOWN:
  1476. return SkinSB_OnNcLButtonDown(psb, wParam, lParam);
  1477. case WM_NCMOUSEMOVE:
  1478. return SkinSB_OnNcMouseMove(psb, wParam, lParam);
  1479. case WM_MOUSEMOVE:
  1480. return SkinSB_OnMouseMove(psb, wParam, lParam);
  1481. case WM_LBUTTONUP:
  1482. return SkinSB_OnLButtonUp(psb, wParam, lParam);
  1483. case WM_NCMOUSELEAVE:
  1484. return SkinSB_OnNcMouseLeave(psb, wParam, lParam);
  1485. case WM_NCLBUTTONDBLCLK:
  1486. case WM_NCRBUTTONDBLCLK:
  1487. return 0;
  1488. case WM_TIMER:
  1489. return SkinSB_OnTimer(psb, wParam, lParam);
  1490. case WM_NCDESTROY:
  1491. LRESULT lr = CallWindowProc(psb->pfnOldProc, hwnd, uMsg, wParam, lParam);
  1492. SkinSB_Uninit(hwnd);
  1493. return lr;
  1494. }
  1495. return CallWindowProc(psb->pfnOldProc, hwnd, uMsg, wParam, lParam);
  1496. }