纯c++实现之滚动窗口
别在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差不多,可以设置大写留余地。
下面直接上完整代码,可以运行的,只实现了拖动滚动块事件,其他事件自己补充吧
- #include <windows.h>
- #define IDC_CANVAS 200
- HWND hwnd_screen = NULL;//屏幕句柄,这里的屏幕既是我们创建的顶级窗口
- HWND hwnd_canvas = NULL;//画布句柄
- HINSTANCE Ghinstance = NULL;//程序实例
- //注意:以下提到的“屏幕”指的都是我们创建的模拟屏幕,也就是顶级窗口,而不是我们计算机的屏幕
- int canvas_width = 2000;//画布长度
- int canvas_height = 1500;//画布宽度
- int screen_width = 0;//屏幕长度
- int screen_height = 0;//屏幕宽度
- int position_width = 0;//当前位置的横坐标
- int position_height = 0;//当前位置的纵坐标
- int hb_pos = 0;//竖直方向滚动条当前位置
- int vb_pos = 0;//水平方向滚动条当前位置
- LRESULT CALLBACK ScreenProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam);//屏幕事件处理函数
- LRESULT CALLBACK CanvasProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam);//画布事件处理函数
- //入口函数
- int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) {
- //========================================创建屏幕begin==================================================
- WNDCLASSEX wc;
- MSG Msg;
- memset(&wc,0,sizeof(wc));
- wc.cbSize = sizeof(WNDCLASSEX);
- wc.lpfnWndProc = ScreenProc;
- wc.hInstance = hInstance;
- wc.hCursor = LoadCursor(NULL, IDC_ARROW);
- wc.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
- wc.lpszClassName = "WindowClass";
- wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
- if(!RegisterClassEx(&wc)) {
- MessageBox(NULL, "Window Registration Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
- return 0;
- }
- //程序实例和屏幕句柄放到全局变量里
- Ghinstance = hInstance;
- hwnd_screen = CreateWindowEx(WS_EX_CLIENTEDGE,"WindowClass","Caption",WS_VISIBLE|WS_OVERLAPPEDWINDOW | WS_HSCROLL | WS_VSCROLL,
- CW_USEDEFAULT,
- CW_USEDEFAULT,
- 800,
- 600,
- NULL,NULL,hInstance,NULL);
- if(hwnd_screen == NULL) {
- MessageBox(NULL, "Screen Creation Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
- return 0;
- }
- //========================================创建屏幕end==================================================
- //你也可以把创建画布的过程放到屏幕的WM_CREATE事件中,放这里是使读者思路能清晰些
- //========================================创建画布begin==================================================
- wc;
- memset(&wc,0,sizeof(wc));
- wc.cbSize = sizeof(WNDCLASSEX);
- wc.lpszClassName = "Canvas";
- wc.lpfnWndProc = CanvasProc;
- wc.hInstance = Ghinstance;//这里可以直接使用全局变量了
- wc.hCursor = LoadCursor(NULL, IDC_ARROW);
- wc.hbrBackground = (HBRUSH)(COLOR_BACKGROUND);
- if(!RegisterClassEx(&wc)) {
- MessageBox(NULL, "Canvas Registration Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
- return 0;
- }
- hwnd_canvas= CreateWindow(
- "Canvas",
- "",
- WS_CHILD | WS_VISIBLE | WS_BORDER,
- 0, 0, canvas_width, canvas_height,
- hwnd_screen,//这里可以直接使用全局变量了,注意,如果是放屏幕的WM_CREATE里面,这时候是还不能使用这个全局变量的,WM_CREATE事件结束后CreateWindow方法才会返回创建窗口的句柄
- (HMENU)IDC_CANVAS,
- Ghinstance,//这里可以直接使用全局变量了
- 0
- );
- if(hwnd_canvas == NULL) {
- MessageBox(NULL, "Canvas Creation Failed!","Error!",MB_ICONEXCLAMATION|MB_OK);
- return 0;
- }
- //========================================创建画布end==================================================
- //该显示的显示
- ShowWindow(hwnd_screen, nCmdShow);
- UpdateWindow(hwnd_screen);
- //该显示的显示
- ShowWindow(hwnd_canvas, SW_SHOW);
- UpdateWindow(hwnd_canvas);
- //消息循环
- while(GetMessage(&Msg, NULL, 0, 0) > 0) {
- TranslateMessage(&Msg);
- DispatchMessage(&Msg);
- }
- return Msg.wParam;
- }
- //屏幕的事件
- LRESULT CALLBACK ScreenProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam) {
- //只需要处理WM_SIZE、WM_HSCROLL、WM_VSCROLL三个消息
- switch(Message) {
- //窗口大小改变时,更新全局变量中的屏幕大小,更新滚动条上滚动块的位置
- case WM_SIZE: {
- //更新屏幕大小begin-----------------------------
- screen_width = LOWORD (lParam);
- screen_height = HIWORD (lParam);
- //更新屏幕大小end-----------------------------
- //更新滚动条上滚动块的位置begin----------------------
- hb_pos = position_width * 100 / (canvas_width - screen_width);//根据恒等式1
- vb_pos = position_height * 100 / (canvas_height - screen_height);
- SCROLLINFO si;
- si.cbSize=sizeof(SCROLLINFO);
- si.fMask=SIF_POS;
- si.nPos = vb_pos;
- SetScrollInfo(hwnd_screen,SB_VERT,&si,true);
- si.nPos = hb_pos;
- SetScrollInfo(hwnd_screen,SB_HORZ,&si,true);
- //更新滚动条上滚动块的位置end----------------------
- //其实还应该更新滚动条上滚动块的长度,这里先忽略吧
- //int hb_length = GValue::s_width * (GValue::s_width - 40) / GValue::c_width;//根据恒等式2
- //int vb_length = GValue::s_height * (GValue::s_height - 40) / GValue::c_height;
- break;
- }
- //水平方向滚动条事件
- case WM_HSCROLL : {
- SCROLLINFO si;
- si.cbSize=sizeof(SCROLLINFO);
- si.fMask=SIF_ALL;
- GetScrollInfo(hwnd_screen,SB_HORZ,&si);//先拿滚动条信息
- switch(LOWORD(wParam)){//这里只处理按下滚动条拖动的事件,其他滚动条事件自己实现吧
- //分析可知按住滚动条拖动过程中,需要修改当前位置、然后基于当前位置移动画布,最后修改滚动条位置(你不修改的话视觉效果上会弹回去的)、
- case SB_THUMBTRACK: {
- position_width = si.nTrackPos * (canvas_width - screen_width) / 100;//更改当前位置
- MoveWindow(hwnd_canvas, 0 - position_width, 0 - position_height, canvas_width, canvas_height, true);//移动画布
- si.nPos=si.nTrackPos;
- break;
- }
- //TODO 滚动条的其他事件
- default: {
- break;
- }
- }
- //回写滚动条滚动块的位置,时视觉上正常
- si.fMask=SIF_POS;
- SetScrollInfo(hwnd_screen, SB_HORZ, &si, true);
- break;
- }
- //竖直方向滚动条事件,与上面相似不解释了
- case WM_VSCROLL : {
- SCROLLINFO si;
- si.cbSize=sizeof(SCROLLINFO);
- si.fMask=SIF_ALL;
- GetScrollInfo(hwnd_screen, SB_VERT, &si);
- switch(LOWORD(wParam)){
- case SB_THUMBTRACK: {
- position_height = si.nTrackPos * (canvas_height - screen_height) / 100;
- MoveWindow(hwnd_canvas, 0 - position_width, 0 - position_height, canvas_width, canvas_height, true);
- si.nPos=si.nTrackPos;
- break;
- }
- default: {
- break;
- }
- }
- si.fMask=SIF_POS;
- SetScrollInfo(hwnd_screen, SB_VERT, &si, true);
- break;
- }
- //鼠标滚轮
- /* case WM_MOUSEWHEEL : {
- //向下
- if(HIWORD(wParam) < 0) {
- vb_pos = vb_pos + 10;
- if(vb_pos > 100) {
- vb_pos = 0;
- }
- } else {
- vb_pos = vb_pos - 10;
- if(vb_pos < 0) {
- vb_pos = 0;
- }
- }
- break;
- }
- */
- case WM_CLOSE: {
- DestroyWindow(hwnd);
- break;
- }
- case WM_DESTROY: {
- PostQuitMessage(0);
- break;
- }
- default:
- return DefWindowProc(hwnd, Message, wParam, lParam);
- }
- return 0;
- }
- //窗口的事件
- LRESULT CALLBACK CanvasProc(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam) {
- switch(Message) {
- //从画布左上角到右下角画一条线,以便观察滚动效果
- case WM_PAINT: {
- PAINTSTRUCT ps;
- HDC hdc;
- RECT rc;
- GetClientRect(hwnd_canvas, &rc);
- hdc = BeginPaint(hwnd_canvas, &ps);
- MoveToEx(hdc, 0 , 0 , NULL);
- LineTo(hdc, rc.right, rc.bottom);
- EndPaint(hwnd_canvas, &ps);
- break;
- }
- case WM_CLOSE: {
- DestroyWindow(hwnd);
- break;
- }
- case WM_DESTROY: {
- PostQuitMessage(0);
- break;
- }
- default:
- return DefWindowProc(hwnd, Message, wParam, lParam);
- }
- }
纯c++实现之滚动窗口的更多相关文章
- 李洪强iOS开发之后使用纯代码实现横向滚动的UIScrollView
李洪强iOS开发之后使用纯代码实现横向滚动的UIScrollView (VTmagic是一个实现左右滚动的控制器的框架,也可以实现此功能) 实现的效果: 01 - 创建四个控制器 02 - 定义需要 ...
- Delphi TScrollBar 用于滚动窗口、组件内容
滚动条组件(TScrollBar)此组件是一个Windows滚动条,用于滚动窗口.组件内容.许多控制有滚动条属性,它们把滚动条作为自己的一部分,对于没有完整滚动条的控制,TScrollBar组件提供了 ...
- Flink Streaming基于滚动窗口的事件时间分析
使用flink-1.9.0进行的测试,在不同的并行度下,Flink对事件时间的处理逻辑不同.包括1.1在并行度为1的本地模式分析和1.2在多并行度的本地模式分析两部分.通过理论结合源码进行验证,得到具 ...
- Qt 滚动窗口类
{ QScrollArea *scrollArea = new QScrollArea(this); scrollArea->setFrameStyle(); scrollArea->se ...
- 纯Js ——文字上下左右滚动
ScrollBaseJs.js var $$ = function (id) { return typeof id == 'string' ? document.getElementById(id) ...
- 使用纯 CSS 实现滚动阴影效果
开门见山,有这样一种非常常见的情况,对于一些可滚动的元素而言.通常在滚动的时候会给垂直于滚动的一侧添加一个阴影,用于表明当前有元素被滚动给该滚出了可视区域,类似这样: 可以看到,在滚动的过程中,会出现 ...
- JS平滑无缝滚动实现———实现首页广告自动滚动效果(附实例)
本文我们实现纯JS方式的滚动广告效果. 先show一下成品: 首先是网页样式: 1. #demo { 2. background: #FFF; 3. overflow:hidden; 4. borde ...
- Storm Windowing storm滑动窗口简介
Storm Windowing 简介 Storm可同时处理窗口内的所有tuple.窗口可以从时间或数量上来划分,由如下两个因素决定: 窗口的长度,可以是时间间隔或Tuple数量: 滑动间隔(slidi ...
- 【jQuery】scroll 滚动到顶部
Jquery 实现页面滚动到顶端 $(document).ready(function () { // 滚动窗口来判断按钮显示或隐藏 $(window).scroll(function () { // ...
随机推荐
- JS初学之-选项卡(图片切换类)
初学选项卡,主要问题卡在了索引值上面,花了较长的时间学习. 索引值其实很好理解,就是为每一个元素用JS的方法添加一个属性,即自定义属性. 在for循环里的函数里用i,会直接弹出这个数组的length, ...
- 那些盒模型在IE6中的BUG们,工程狮的你可曾遇到过?
HTML5学堂 那些盒模型在IE6中的BUG们,工程狮的你可曾遇到过? IE6已经渐渐的开始退出浏览器的历史舞台.虽然当年IE6作为微软的一款利器击败网景,但之后也因为版本的持续不更新而被火狐和谷歌三 ...
- 不小心误删@‘local’操作恢复
今天在测试用户权限的时候不小心把User: ''@'localhost';用户删除了 导致任何用户登录都无权限操作 恢复过程 停止mysql服务:在mysql安装目录下找到my.ini(linux下是 ...
- 安装了iis之后,打开默认网站http://localhost/要求输入用户名和密码解决办法
开始-运行gpedit.msc回车. 计算机配置--管理模板-windows 组件-Internet Exporer-Internet控制面板-安全页-Internet区域:双击登陆选 ...
- Python如何规定对方输入的数字必须是整数?
可以使用字符串str的isdigit方法判断字符串是否是一个仅有数字组成,也就是整数.如果是整数退出while循环,否则继续请求输入. 1 2 3 4 5 6 while True: x = ...
- (转)The Road to TensorFlow
Stephen Smith's Blog All things Sage 300… The Road to TensorFlow – Part 7: Finally Some Code leave a ...
- java打包压缩文件
package com.it.simple.util; import java.io.BufferedOutputStream;import java.io.ByteArrayOutputStream ...
- [C] zintrin.h : 智能引入intrinsic函数。支持VC、GCC,兼容Windows、Linux、Mac OS X
博客来源:http://blog.csdn.net/zyl910/article/details/8100744 现在很多编译器支持intrinsic函数,这给编写SSE等SIMD代码带来了方便.但是 ...
- C#综合揭秘——深入分析委托与事件
http://www.cnblogs.com/leslies2/archive/2012/03/22/2389318.html 引言 本篇文章将为你介绍一下 Delegate 的使用方式,逐渐揭开 C ...
- 怎么保护PDF文档和扫描文件里的机密信息
从事商务工作的人,必然要处理带有机密信息的文档,需要分享这些文档的时候,如何谨慎小心地对待那些机密信息,说到底还是取决于自己.分享文档的目的不同,对文档的保护类型和级别也不一样.例如,只有授权的读者才 ...