别在MFC了,先分析下,上图

我们以左上角为坐标原点,用position_width和position_height来保存当前显示坐标。

根据msdn说明,滚动条默认情况下的值在0~100之间。

根据图可以知道positon_width的活动范围是0到canvas_width-screen-width,另一边类似。

所以有恒等式1:position_width/(canvas_width-screen_width) = hb_pos/100,其中hb_pos是水平方向滚动条当前值。

滚动块长度公式2:screen_width/canvas_width = 滚动块长度/滚动条可滚动区域长度,滚动条可滚动区域长度大概是screen_width-40差不多,可以设置大写留余地。

下面直接上完整代码,可以运行的,只实现了拖动滚动块事件,其他事件自己补充吧

    1. #include <windows.h>
    2. #define  IDC_CANVAS                  200
    3. HWND hwnd_screen = NULL;//屏幕句柄,这里的屏幕既是我们创建的顶级窗口
    4. HWND hwnd_canvas = NULL;//画布句柄
    5. HINSTANCE   Ghinstance = NULL;//程序实例
    6. //注意:以下提到的“屏幕”指的都是我们创建的模拟屏幕,也就是顶级窗口,而不是我们计算机的屏幕
    7. int         canvas_width        = 2000;//画布长度
    8. int         canvas_height       = 1500;//画布宽度
    9. int         screen_width        = 0;//屏幕长度
    10. int         screen_height       = 0;//屏幕宽度
    11. int         position_width      = 0;//当前位置的横坐标
    12. int         position_height     = 0;//当前位置的纵坐标
    13. int         hb_pos              = 0;//竖直方向滚动条当前位置
    14. int         vb_pos              = 0;//水平方向滚动条当前位置
    15. LRESULT CALLBACK ScreenProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam);//屏幕事件处理函数
    16. LRESULT CALLBACK CanvasProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam);//画布事件处理函数
    17. //入口函数
    18. int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) {
    19. //========================================创建屏幕begin==================================================
    20. WNDCLASSEX wc;
    21. MSG Msg;
    22. memset(&wc,0,sizeof(wc));
    23. wc.cbSize        = sizeof(WNDCLASSEX);
    24. wc.lpfnWndProc   = ScreenProc;
    25. wc.hInstance     = hInstance;
    26. wc.hCursor       = LoadCursor(NULL, IDC_ARROW);
    27. wc.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
    28. wc.lpszClassName = "WindowClass";
    29. wc.hIcon         = LoadIcon(NULL, IDI_APPLICATION);
    30. if(!RegisterClassEx(&wc)) {
    31. MessageBox(NULL, "Window Registration Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
    32. return 0;
    33. }
    34. //程序实例和屏幕句柄放到全局变量里
    35. Ghinstance = hInstance;
    36. hwnd_screen = CreateWindowEx(WS_EX_CLIENTEDGE,"WindowClass","Caption",WS_VISIBLE|WS_OVERLAPPEDWINDOW | WS_HSCROLL | WS_VSCROLL,
    37. CW_USEDEFAULT,
    38. CW_USEDEFAULT,
    39. 800,
    40. 600,
    41. NULL,NULL,hInstance,NULL);
    42. if(hwnd_screen == NULL) {
    43. MessageBox(NULL, "Screen Creation Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
    44. return 0;
    45. }
    46. //========================================创建屏幕end==================================================
    47. //你也可以把创建画布的过程放到屏幕的WM_CREATE事件中,放这里是使读者思路能清晰些
    48. //========================================创建画布begin==================================================
    49. wc;
    50. memset(&wc,0,sizeof(wc));
    51. wc.cbSize        = sizeof(WNDCLASSEX);
    52. wc.lpszClassName = "Canvas";
    53. wc.lpfnWndProc   = CanvasProc;
    54. wc.hInstance     = Ghinstance;//这里可以直接使用全局变量了
    55. wc.hCursor       = LoadCursor(NULL, IDC_ARROW);
    56. wc.hbrBackground = (HBRUSH)(COLOR_BACKGROUND);
    57. if(!RegisterClassEx(&wc)) {
    58. MessageBox(NULL, "Canvas Registration Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
    59. return 0;
    60. }
    61. hwnd_canvas= CreateWindow(
    62. "Canvas",
    63. "",
    64. WS_CHILD | WS_VISIBLE | WS_BORDER,
    65. 0, 0, canvas_width, canvas_height,
    66. hwnd_screen,//这里可以直接使用全局变量了,注意,如果是放屏幕的WM_CREATE里面,这时候是还不能使用这个全局变量的,WM_CREATE事件结束后CreateWindow方法才会返回创建窗口的句柄
    67. (HMENU)IDC_CANVAS,
    68. Ghinstance,//这里可以直接使用全局变量了
    69. 0
    70. );
    71. if(hwnd_canvas == NULL) {
    72. MessageBox(NULL, "Canvas Creation Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
    73. return 0;
    74. }
    75. //========================================创建画布end==================================================
    76. //该显示的显示
    77. ShowWindow(hwnd_screen, nCmdShow);
    78. UpdateWindow(hwnd_screen);
    79. //该显示的显示
    80. ShowWindow(hwnd_canvas, SW_SHOW);
    81. UpdateWindow(hwnd_canvas);
    82. //消息循环
    83. while(GetMessage(&Msg, NULL, 0, 0) > 0) {
    84. TranslateMessage(&Msg);
    85. DispatchMessage(&Msg);
    86. }
    87. return Msg.wParam;
    88. }
    89. //屏幕的事件
    90. LRESULT CALLBACK ScreenProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam) {
    91. //只需要处理WM_SIZE、WM_HSCROLL、WM_VSCROLL三个消息
    92. switch(Message) {
    93. //窗口大小改变时,更新全局变量中的屏幕大小,更新滚动条上滚动块的位置
    94. case WM_SIZE: {
    95. //更新屏幕大小begin-----------------------------
    96. screen_width = LOWORD (lParam);
    97. screen_height = HIWORD (lParam);
    98. //更新屏幕大小end-----------------------------
    99. //更新滚动条上滚动块的位置begin----------------------
    100. hb_pos = position_width * 100 / (canvas_width - screen_width);//根据恒等式1
    101. vb_pos = position_height * 100 / (canvas_height - screen_height);
    102. SCROLLINFO  si;
    103. si.cbSize=sizeof(SCROLLINFO);
    104. si.fMask=SIF_POS;
    105. si.nPos = vb_pos;
    106. SetScrollInfo(hwnd_screen,SB_VERT,&si,true);
    107. si.nPos = hb_pos;
    108. SetScrollInfo(hwnd_screen,SB_HORZ,&si,true);
    109. //更新滚动条上滚动块的位置end----------------------
    110. //其实还应该更新滚动条上滚动块的长度,这里先忽略吧
    111. //int hb_length = GValue::s_width * (GValue::s_width - 40) / GValue::c_width;//根据恒等式2
    112. //int vb_length = GValue::s_height * (GValue::s_height - 40) / GValue::c_height;
    113. break;
    114. }
    115. //水平方向滚动条事件
    116. case WM_HSCROLL : {
    117. SCROLLINFO  si;
    118. si.cbSize=sizeof(SCROLLINFO);
    119. si.fMask=SIF_ALL;
    120. GetScrollInfo(hwnd_screen,SB_HORZ,&si);//先拿滚动条信息
    121. switch(LOWORD(wParam)){//这里只处理按下滚动条拖动的事件,其他滚动条事件自己实现吧
    122. //分析可知按住滚动条拖动过程中,需要修改当前位置、然后基于当前位置移动画布,最后修改滚动条位置(你不修改的话视觉效果上会弹回去的)、
    123. case SB_THUMBTRACK: {
    124. position_width = si.nTrackPos * (canvas_width - screen_width) / 100;//更改当前位置
    125. MoveWindow(hwnd_canvas, 0 - position_width, 0 - position_height, canvas_width, canvas_height, true);//移动画布
    126. si.nPos=si.nTrackPos;
    127. break;
    128. }
    129. //TODO 滚动条的其他事件
    130. default: {
    131. break;
    132. }
    133. }
    134. //回写滚动条滚动块的位置,时视觉上正常
    135. si.fMask=SIF_POS;
    136. SetScrollInfo(hwnd_screen, SB_HORZ, &si, true);
    137. break;
    138. }
    139. //竖直方向滚动条事件,与上面相似不解释了
    140. case WM_VSCROLL : {
    141. SCROLLINFO  si;
    142. si.cbSize=sizeof(SCROLLINFO);
    143. si.fMask=SIF_ALL;
    144. GetScrollInfo(hwnd_screen, SB_VERT, &si);
    145. switch(LOWORD(wParam)){
    146. case SB_THUMBTRACK: {
    147. position_height = si.nTrackPos * (canvas_height - screen_height) / 100;
    148. MoveWindow(hwnd_canvas, 0 - position_width, 0 - position_height, canvas_width, canvas_height, true);
    149. si.nPos=si.nTrackPos;
    150. break;
    151. }
    152. default: {
    153. break;
    154. }
    155. }
    156. si.fMask=SIF_POS;
    157. SetScrollInfo(hwnd_screen, SB_VERT, &si, true);
    158. break;
    159. }
    160. //鼠标滚轮
    161. /*      case WM_MOUSEWHEEL : {
    162. //向下
    163. if(HIWORD(wParam) < 0) {
    164. vb_pos = vb_pos + 10;
    165. if(vb_pos > 100) {
    166. vb_pos = 0;
    167. }
    168. } else {
    169. vb_pos = vb_pos - 10;
    170. if(vb_pos < 0) {
    171. vb_pos = 0;
    172. }
    173. }
    174. break;
    175. }
    176. */
    177. case WM_CLOSE: {
    178. DestroyWindow(hwnd);
    179. break;
    180. }
    181. case WM_DESTROY: {
    182. PostQuitMessage(0);
    183. break;
    184. }
    185. default:
    186. return DefWindowProc(hwnd, Message, wParam, lParam);
    187. }
    188. return 0;
    189. }
    190. //窗口的事件
    191. LRESULT CALLBACK CanvasProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam) {
    192. switch(Message) {
    193. //从画布左上角到右下角画一条线,以便观察滚动效果
    194. case WM_PAINT: {
    195. PAINTSTRUCT ps;
    196. HDC hdc;
    197. RECT rc;
    198. GetClientRect(hwnd_canvas, &rc);
    199. hdc = BeginPaint(hwnd_canvas, &ps);
    200. MoveToEx(hdc, 0 , 0 , NULL);
    201. LineTo(hdc, rc.right, rc.bottom);
    202. EndPaint(hwnd_canvas, &ps);
    203. break;
    204. }
    205. case WM_CLOSE: {
    206. DestroyWindow(hwnd);
    207. break;
    208. }
    209. case WM_DESTROY: {
    210. PostQuitMessage(0);
    211. break;
    212. }
    213. default:
    214. return DefWindowProc(hwnd, Message, wParam, lParam);
    215. }
    216. }

纯c++实现之滚动窗口的更多相关文章

  1. 李洪强iOS开发之后使用纯代码实现横向滚动的UIScrollView

    李洪强iOS开发之后使用纯代码实现横向滚动的UIScrollView (VTmagic是一个实现左右滚动的控制器的框架,也可以实现此功能) 实现的效果:  01 - 创建四个控制器 02 - 定义需要 ...

  2. Delphi TScrollBar 用于滚动窗口、组件内容

    滚动条组件(TScrollBar)此组件是一个Windows滚动条,用于滚动窗口.组件内容.许多控制有滚动条属性,它们把滚动条作为自己的一部分,对于没有完整滚动条的控制,TScrollBar组件提供了 ...

  3. Flink Streaming基于滚动窗口的事件时间分析

    使用flink-1.9.0进行的测试,在不同的并行度下,Flink对事件时间的处理逻辑不同.包括1.1在并行度为1的本地模式分析和1.2在多并行度的本地模式分析两部分.通过理论结合源码进行验证,得到具 ...

  4. Qt 滚动窗口类

    { QScrollArea *scrollArea = new QScrollArea(this); scrollArea->setFrameStyle(); scrollArea->se ...

  5. 纯Js ——文字上下左右滚动

    ScrollBaseJs.js var $$ = function (id) { return typeof id == 'string' ? document.getElementById(id) ...

  6. 使用纯 CSS 实现滚动阴影效果

    开门见山,有这样一种非常常见的情况,对于一些可滚动的元素而言.通常在滚动的时候会给垂直于滚动的一侧添加一个阴影,用于表明当前有元素被滚动给该滚出了可视区域,类似这样: 可以看到,在滚动的过程中,会出现 ...

  7. JS平滑无缝滚动实现———实现首页广告自动滚动效果(附实例)

    本文我们实现纯JS方式的滚动广告效果. 先show一下成品: 首先是网页样式: 1. #demo { 2. background: #FFF; 3. overflow:hidden; 4. borde ...

  8. Storm Windowing storm滑动窗口简介

    Storm Windowing 简介 Storm可同时处理窗口内的所有tuple.窗口可以从时间或数量上来划分,由如下两个因素决定: 窗口的长度,可以是时间间隔或Tuple数量: 滑动间隔(slidi ...

  9. 【jQuery】scroll 滚动到顶部

    Jquery 实现页面滚动到顶端 $(document).ready(function () { // 滚动窗口来判断按钮显示或隐藏 $(window).scroll(function () { // ...

随机推荐

  1. c 函数及指针学习 8

    联合体 1 2 3 4 5 6 7 8 9 10 11 12 13 #include <stdio.h>   union sa     {     double a;     int b; ...

  2. php部分--操作MySQL 增删改查

    1.建立主页 并进行信息的显示和查询 <table width="95%" border="1" cellpadding="0" ce ...

  3. UVa 272 Tex Quotes --- 水题

    题目大意:在TeX中,左引号是 ``,右引号是 ''.输入一篇包含双引号的文章,你的任务是把他转成TeX的格式 解题思路:水题,定义一个变量标记是左引号还是右引号即可 /* UVa 272 Tex Q ...

  4. Linux系统编程@终端IO

    Linux系统中终端设备种类  终端是一种字符型设备,有多种类型,通常使用tty 来简称各种类型的终端设备.终端特殊设备文件一般有以下几种: 串行端口终端(/dev/ttySn ) ,伪终端(/dev ...

  5. 使用substring和split方法从字符串中抽取一组清单

    提取前: '这是一个句子.这是一个句子,包含了一列清单: 樱桃, 橙子, 桔子, 苹果, 香蕉.这是清单内容.'; 提取后: ["樱桃", " 橙子", &qu ...

  6. C#实现图片文件到数据流再到图片文件的转换 --转

    /----引入必要的命名空间 using System.IO; using System.Drawing.Imaging; //----代码部分----// private byte[] photo; ...

  7. Sklearn库例子1:Sklearn库中AdaBoost和Decision Tree运行结果的比较

    DisCrete Versus Real AdaBoost 关于Discrete 和Real AdaBoost 可以参考博客:http://www.cnblogs.com/jcchen1987/p/4 ...

  8. for循环小题

    已知数列1,1,2,3,5,8,…….,N.输出前N项的和: 出1到100之间所有偶数之和 国际象棋问题 已知数列1,1,2,3,5,8,…….,N.输出前N项的和: int a = 1, b = 1 ...

  9. Facade模式

    Facade模式要求一个子系统的外部与其内部的通信必须通过一个统一的Facade对象进行.Facade模式提供一个高层次的接口,使得子系统更易于使用.  就如同医院的接待员一样,Facade模式的Fa ...

  10. 如何在Windows 下安装Python

    1.  安装源程序的选择 官网:https://www.python.org/downloads/ 选择版本下载,根据实际计算机位数(64位),下载的为: Python-2.7.10.amd64.ms ...