最强大脑--51度灰挑战项目

择善教育公开课



//{{NO_DEPENDENCIES}}
// Microsoft Visual C++ 生成的包含文件。
// 供 BrainMaster.rc 使用
//
#define IDD_DIALOG_MAIN                 101
#define IDC_BTN_START                   1001
#define IDC_BUTTON_CHANGE_COLOR         1002
#define IDC_SLIDER_DIF                  1003
#define IDC_STATIC_DIF                  1004
#define IDC_COLORNOW                    1005
#define IDC_GAME_AREA                   1006

// Next default values for new objects
// 
#ifdef APSTUDIO_INVOKED
#ifndef APSTUDIO_READONLY_SYMBOLS
#define _APS_NEXT_RESOURCE_VALUE        102
#define _APS_NEXT_COMMAND_VALUE         40001
#define _APS_NEXT_CONTROL_VALUE         1007
#define _APS_NEXT_SYMED_VALUE           101
#endif
#endif

#include <stdio.h>
#include <windows.h>
#include <CommCtrl.h>
#include <list>
using namespace  std;
#include "resource.h"

typedef struct _gameRect{
	RECT rect; //保存一个色块的位置信息
	COLORREF rgb;// 保存一个色块的颜色信息
	bool Win;// 保存这个色块是否唯一的颜色,即正确色块
}GameRect;
typedef list<GameRect> GameList; //游戏色块链表数据类型

GameList g_GameList; // 链表
HINSTANCE g_hInstance;
COLORREF g_ColorNow; // 全局变量保存当前颜色
unsigned int g_GameDiff;//当前游戏难度
unsigned int g_Stage; // 当前第几关?
// 生成随机数 ...0~base
DWORD GetRandrom(DWORD base)
{
	DWORD result;
	__asm{
		rdtsc
		mov result,eax
	}
	return result%base;
}
// 生成本次游戏的数据
void OnInitStageData(HWND hWnd)
{
	g_GameList.clear(); // 清空原来的链表信息
	DWORD nItemNum = g_Stage + 3; // 当前关卡需要显示多少个色块?
	if (nItemNum>8)
	{
		nItemNum = 8;
	}
	HWND GameWnd = GetDlgItem(hWnd, IDC_GAME_AREA); // 获取游戏区域的句柄
	RECT WndRect;
	GetWindowRect(GameWnd, &WndRect); //获取游戏区域的大小
	DWORD nWidth = WndRect.right - WndRect.left;
	DWORD nHeight = WndRect.bottom - WndRect.top;

	DWORD WinItemIndex = GetRandrom(nItemNum); // 生成一个随机数,指示哪一个色块是正确滴
	DWORD nItemCntRow = 3;
	DWORD nItemCntColumn = (nItemNum % 3) ? nItemNum / 3 + 1 : nItemNum / 3;//计算需要多少行色块
	DWORD nItemWidth = (nWidth - 10) / nItemCntRow; // 每个色块的宽度
	DWORD nItemHeight = (nHeight - 10) / nItemCntColumn;// 每个色块的高度

	for (DWORD i = 0; i <= nItemNum;i++)
	{
		GameRect gRect;
		gRect.rect.left = 10 + (i % 3)*nItemWidth;
		gRect.rect.right = gRect.rect.left + nItemWidth - 10;

		gRect.rect.top = 10 + (i / 3)*nItemHeight;
		gRect.rect.bottom = gRect.rect.top + nItemHeight - 10;

		gRect.rgb = g_ColorNow; // 正常的颜色
		gRect.Win = false; //大部分的色块不是正确答案
		if (i == WinItemIndex)
		{
			byte r = GetRValue(g_ColorNow) > 125 ? GetRValue(g_ColorNow) - g_GameDiff : GetRValue(g_ColorNow) + g_GameDiff;
			byte g = GetGValue(g_ColorNow) > 125 ? GetGValue(g_ColorNow) - g_GameDiff : GetGValue(g_ColorNow) + g_GameDiff;
			byte b = GetBValue(g_ColorNow) > 125 ? GetBValue(g_ColorNow) - g_GameDiff : GetBValue(g_ColorNow) + g_GameDiff;
			gRect.rgb = RGB(r, g, b);

			gRect.Win = TRUE;
		}
		g_GameList.push_back(gRect);
	}
	return;
}
void OnShowStage(HWND hWnd)
{
	HWND GameWnd = GetDlgItem(hWnd, IDC_GAME_AREA);
	HDC hdc = GetDC(GameWnd);
	GameList::iterator iter = g_GameList.begin();
	// 遍历链表,取出每个色块的信息,并按信息去进行绘图
	for (; iter != g_GameList.end();iter++)
	{
		GameRect gameRect = *iter;
		HBRUSH hBrush = CreateSolidBrush(gameRect.rgb);
		HBRUSH hOldBrush = (HBRUSH)SelectObject(hdc, hBrush);

		Rectangle(hdc, gameRect.rect.left, gameRect.rect.top, gameRect.rect.right, gameRect.rect.bottom);
		SelectObject(hdc, hOldBrush);
	}

	ReleaseDC(GameWnd, hdc);
	return;
}

// 功能:更新当前颜色
bool OnChangeColor(HWND hWnd)
{
	HWND hColorWnd = GetDlgItem(hWnd, IDC_COLORNOW);//获取当前颜色指示器控件的句柄
	// 初始化一个结构体...
	CHOOSECOLOR chs;
	memset(&chs, 0, sizeof(CHOOSECOLOR));
	chs.lStructSize = sizeof(CHOOSECOLOR);
	chs.hwndOwner = hColorWnd;
	chs.rgbResult = g_ColorNow;
	chs.lpCustColors = &g_ColorNow;
	chs.Flags = CC_ANYCOLOR | CC_RGBINIT;
	ChooseColor(&chs);
	g_ColorNow = chs.rgbResult; //获取到选择的颜色

	HDC hdc = GetDC(hColorWnd);
	RECT rect;
	GetWindowRect(hColorWnd, &rect);//获取的是指示器控件的大小
	// 创建一个笔刷
	HBRUSH hBrush = CreateSolidBrush(g_ColorNow);
	SelectObject(hdc, hBrush);
	Rectangle(hdc, 0, 0, rect.right - rect.left, rect.bottom - rect.top);
	ReleaseDC(hColorWnd, hdc);
	return true;
}
bool OnStartGame(HWND hWnd)
{
	if (g_ColorNow == 0) // 如果点击开始游戏的时候,没有选择颜色,先要求选择一次
	{
		OnChangeColor(hWnd);
	}
	OnInitStageData(hWnd);
	OnShowStage(hWnd);
	return false;
}
// 初始化的函数
bool OnInitDialog(HWND hWnd)
{
	g_Stage = 1;
	return false;
}
// 检查一个点是否在规定矩形内
bool PointInRect(HWND hWnd,POINT pt, RECT rect)
{
	return pt.x < (rect.right - rect.left) && pt.y < (rect.bottom - rect.top);
}
// 检查一个点是否是全局链表中的正确色块信息
bool PointIsRight(HWND hWnd, POINT pt)
{
	//遍历链表,对比每个色块的位置信息与鼠标点
	HWND gameWnd = GetDlgItem(hWnd, IDC_GAME_AREA);
	RECT WndRect;
	GetWindowRect(gameWnd, &WndRect);
	// 将屏幕坐标转换成客户区坐标并减去
	POINT lefttop;
	lefttop.x = WndRect.left;
	lefttop.y = WndRect.top;
	ScreenToClient(hWnd, &lefttop);
	//570427370
	//
	pt.x -= lefttop.x;
	pt.y -= lefttop.y;

	bool bIsRight = false;
	GameList::iterator iter = g_GameList.begin();
	for (; iter != g_GameList.end();iter++)
	{
		GameRect gameRect = *iter;
		// 如果鼠标点在矩形区域内,且矩形区域是不同颜色滴...找到了正确答案
		if (PointInRect(hWnd,pt,gameRect.rect) && gameRect.Win)
		{
			bIsRight = true;
			break;
		}
	}

	return bIsRight;
}
// Windows通过消息来驱动,
INT_PTR CALLBACK GameProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
	RECT gameArea;
	switch (uMsg)
	{
	case WM_INITDIALOG://首次创建窗口时产生
		OnInitDialog(hWnd);
		break;
	case WM_LBUTTONUP:
		POINT pt;
		GetCursorPos(&pt);//获取当前鼠标的位置
		ScreenToClient(hWnd, &pt);//鼠标点也转成客户区坐标系
		GetWindowRect(GetDlgItem(hWnd, IDC_GAME_AREA), &gameArea);
		if (PointInRect(hWnd,pt,gameArea)) // 检查鼠标点是否在游戏区域内
		{
			if (PointIsRight(hWnd, pt))
			{
				g_Stage++; // 当前关卡加1
				SendMessage(hWnd, WM_COMMAND, IDC_BTN_START, 0);//模拟鼠标点击窗口中开始游戏按钮

			}
			else
				MessageBox(hWnd, L"选择错误", L"Wrong", MB_OK | MB_ICONINFORMATION);
		}
		break;
	case WM_NOTIFY://获取通知消息
		if (wParam == IDC_SLIDER_DIF)
		{
			g_GameDiff = (unsigned int)SendMessage(GetDlgItem(hWnd, IDC_SLIDER_DIF), TBM_GETPOS, 0, 0);
			if (g_GameDiff <1)
			{
				g_GameDiff = 1;
			}
			char szTmp[MAX_PATH];
			sprintf_s(szTmp, MAX_PATH, "当前游戏难度:%d", 100-g_GameDiff);
			SetDlgItemTextA(hWnd, IDC_STATIC_DIF, szTmp);
		}
		break;
	case WM_COMMAND:
		switch (LOWORD(wParam))
		{
		case IDC_BUTTON_CHANGE_COLOR:
			OnChangeColor(hWnd);
			break;
		case IDC_BTN_START:
			OnStartGame(hWnd);
			break;
		default:
			break;
		}
		break;
	case WM_CLOSE:
	case WM_DESTROY:
		PostQuitMessage(0);
		return TRUE;
	default:
		break;
	}
	return FALSE;
}
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nShowCmd)
{
	g_hInstance = hInstance;
	HWND hWnd = CreateDialog(hInstance, MAKEINTRESOURCE(IDD_DIALOG_MAIN), NULL, GameProc);
	if (hWnd == 0)
	{
		MessageBox(0, L"创建窗口失败", 0, 0);
		return -1;
	}
	// 获取窗口大小,将窗口设置到整个屏幕的中心
	RECT rect;
	GetWindowRect(hWnd, &rect);
	int nScreenX = GetSystemMetrics(SM_CXSCREEN); //获取屏幕的宽度
	int nScreenY = GetSystemMetrics(SM_CYSCREEN); //获取屏幕的高度

	SetWindowPos(hWnd, HWND_TOP, nScreenX / 2 - (rect.right - rect.left) / 2,
		nScreenY / 2 - (rect.bottom - rect.top) / 2, 0, 0, SWP_NOSIZE | SWP_SHOWWINDOW);//不改变窗口大小,只将窗口置顶显示,并移动到屏幕中心

	MSG msg;
	while (GetMessage(&msg,0,0,0))
	{
		TranslateMessage(&msg);
		DispatchMessage(&msg);
	}
	return 0;
}

鼠标点击色块区域的算法,还需进一步的修改。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 《史上最强-kettle-培训教程.pdf》是一本关于Kettle软件的培训教程,旨在帮助读者了解和掌握Kettle工具的使用。Kettle(Kitchen Extract,Transform,Transport and Load Environment)是一款强大的ETL工具,用于数据仓库建设、数据抽取、转换和加载等工作。 这本培训教程通过详细的步骤和案例介绍了Kettle工具的各个功能和应用场景。首先,它从Kettle的基本介绍开始,包括Kettle的功能特点、安装和环境配置等方面。接着,介绍了Kettle的三个核心模块:抽取(Extract)、转换(Transform)和加载(Load),并通过实际案例演示了如何进行数据抽取、清洗、转换和加载。 此外,该培训教程还介绍了Kettle常用的数据处理技术和工具,如数据清洗、数据合并、数据拆分、数据过滤等,以及Kettle与其他数据库软件(如MySQL、Oracle)的集成方法。同时,还详细介绍了Kettle的作业调、性能优化、运行监控等高级功能,使读者能够更好地使用Kettle进行大规模数据处理和ETL工作。 总的来说,《史上最强-kettle-培训教程.pdf》是一本全面而实用的Kettle培训教程,无论是初学者还是有一定Kettle使用经验的人都能从中受益。通过学习这本教程,读者能够系统地掌握Kettle的使用方法和技巧,提高数据处理的效率和质量,为企业的数据分析和决策提供有力支持。 ### 回答2: 《史上最强-kettle-培训教程.pdf》是一本介绍数据集成工具Kettle的培训教程。Kettle,全称“Kettle Extraction, Transformation, Loading”,是一款开源的ETL(Extract, Transform, Load)工具,可以帮助用户高效地进行数据集成和数据处理。 这本教程由作者经过精心编写,旨在帮助读者快速掌握Kettle的使用方法和技巧。教程的内容分为多个章节,涵盖了Kettle的基本概念、安装配置、基本操作、数据抽取与转换、数据加载等方面的知识点,旨在帮助读者系统地了解Kettle的功能和应用场景。 教程的特点是内容详尽、通俗易懂。作者在讲解每个知识点时,都采用了简单明了的语言和图例,使得读者能够轻松理解和掌握。此外,教程还提供了实例演示和练习题,帮助读者巩固所学的知识并能够独立运用Kettle进行数据集成和处理。 《史上最强-kettle-培训教程.pdf》毫无疑问是一本对于想要学习和应用Kettle的人来说非常有价值的资料。无论是对于初学者还是有一定经验的用户,都能够从中获得新的知识和技能。通过学习这本教程,读者将能够充分利用Kettle的强大功能,提高工作效率,并能够更好地应对数据集成和处理的需求。 ### 回答3: 《史上最强-kettle-培训教程.pdf》是一本关于使用 Kettle 软件进行培训的教程资料。Kettle 是一款强大的开源的ETL工具,用于数据抽取、转换和加载。这个教程资料以其详细、全面和易懂的内容,被誉为史上最强的培训教程之一。 首先,这本教程在内容上非常丰富,涵盖了Kettle软件的各个方面。无论是初学者还是有一定经验的从业者,都能从中找到适合自己的学习内容。教程按照逻辑顺序,从介绍Kettle的基本概念开始,逐步深入讲解了数据源连接、数据转换、数据处理和数据加载等主要功能。此外,还有许多实际案例和示例,帮助读者更好地理解和应用所学知识。 其次,这本教程的讲解方式简洁明了,语言通俗易懂。即使没有相关的IT专业知识背景,读者也能够轻松理解和掌握Kettle的使用方法。教程中使用了大量的图表、示意图和代码示例,通过直观的形式展示Kettle的操作步骤和实际效果,使学习过程更加直观、生动。 最重要的是,这本教程注重实践的指导,强调学以致用。教程中不仅讲解了理论知识,还提供了许多实践操作的机会,使读者能够通过实际操作来巩固所学的知识。同时,教程还介绍了一些常见的问题和解决方案,帮助读者解决在实际应用中遇到的困难。 总之,《史上最强-kettle-培训教程.pdf》是一本内容丰富、讲解清晰、实践导向的教程资料。无论是初学者还是有经验的用户,都能从中受益匪浅。通过学习这本教程,读者可以更好地掌握Kettle软件的使用,提高数据处理和分析的能力。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值