2019-01-05 20:21:43 +08:00
|
|
|
|
// ScreenSpy.cpp: implementation of the CScreenSpy class.
|
|
|
|
|
|
//
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
#include "stdafx.h"
|
|
|
|
|
|
#include "ScreenSpy.h"
|
|
|
|
|
|
#include "Common.h"
|
2019-01-06 21:18:26 +08:00
|
|
|
|
#include <stdio.h>
|
2019-01-05 20:21:43 +08:00
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
// Construction/Destruction
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
CScreenSpy::CScreenSpy(ULONG ulbiBitCount)
|
|
|
|
|
|
{
|
|
|
|
|
|
m_bAlgorithm = ALGORITHM_DIFF;
|
|
|
|
|
|
m_dwBitBltRop = SRCCOPY;
|
|
|
|
|
|
m_BitmapInfor_Full = NULL;
|
|
|
|
|
|
switch (ulbiBitCount)
|
|
|
|
|
|
{
|
|
|
|
|
|
case 16:
|
|
|
|
|
|
case 32:
|
|
|
|
|
|
m_ulbiBitCount = ulbiBitCount;
|
|
|
|
|
|
break;
|
|
|
|
|
|
default:
|
|
|
|
|
|
m_ulbiBitCount = 16;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
m_hDeskTopWnd = GetDesktopWindow();
|
2019-01-06 21:18:26 +08:00
|
|
|
|
m_hFullDC = GetDC(m_hDeskTopWnd);
|
2019-01-05 20:21:43 +08:00
|
|
|
|
|
2019-01-06 21:18:26 +08:00
|
|
|
|
m_hFullMemDC = CreateCompatibleDC(m_hFullDC);
|
|
|
|
|
|
//::GetSystemMetrics(SM_CXSCREEN/SM_CYSCREEN)<29><>ȡ<EFBFBD><C8A1>Ļ<EFBFBD><C4BB>С<EFBFBD><D0A1>
|
|
|
|
|
|
//<2F><><EFBFBD>統<EFBFBD><E7B5B1>Ļ<EFBFBD><C4BB>ʾ<EFBFBD><CABE><EFBFBD><EFBFBD>Ϊ125%ʱ<><CAB1><EFBFBD><EFBFBD>ȡ<EFBFBD><C8A1><EFBFBD><EFBFBD><EFBFBD><EFBFBD>Ļ<EFBFBD><C4BB>С<EFBFBD><D0A1>Ҫ<EFBFBD><D2AA><EFBFBD><EFBFBD>1.25<EFBFBD>Ŷ<EFBFBD>
|
|
|
|
|
|
DEVMODE devmode;
|
|
|
|
|
|
memset(&devmode, 0, sizeof (devmode));
|
|
|
|
|
|
devmode.dmSize = sizeof(DEVMODE);
|
|
|
|
|
|
devmode.dmDriverExtra = 0;
|
|
|
|
|
|
BOOL Isgetdisplay = EnumDisplaySettings(NULL, ENUM_CURRENT_SETTINGS, &devmode);
|
|
|
|
|
|
m_ulFullWidth = devmode.dmPelsWidth;
|
|
|
|
|
|
m_ulFullHeight = devmode.dmPelsHeight;
|
2019-01-10 19:35:03 +08:00
|
|
|
|
int w = ::GetSystemMetrics(SM_CXSCREEN), h = ::GetSystemMetrics(SM_CYSCREEN);
|
|
|
|
|
|
m_bZoomed = (w != m_ulFullWidth) || (h != m_ulFullHeight);
|
|
|
|
|
|
m_wZoom = double(m_ulFullWidth) / w, m_hZoom = double(m_ulFullHeight) / h;
|
|
|
|
|
|
printf("=> <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ű<EFBFBD><C5B1><EFBFBD>: %.2f, %.2f\t<EFBFBD>ֱ<EFBFBD><EFBFBD>ʣ<EFBFBD>%d x %d\n", m_wZoom, m_hZoom, m_ulFullWidth, m_ulFullHeight);
|
|
|
|
|
|
m_wZoom = 1.0/m_wZoom, m_hZoom = 1.0/m_hZoom;
|
2019-01-05 20:21:43 +08:00
|
|
|
|
m_BitmapInfor_Full = ConstructBI(m_ulbiBitCount,m_ulFullWidth, m_ulFullHeight);
|
|
|
|
|
|
m_BitmapData_Full = NULL;
|
|
|
|
|
|
m_BitmapHandle = ::CreateDIBSection(m_hFullDC, m_BitmapInfor_Full,
|
|
|
|
|
|
DIB_RGB_COLORS, &m_BitmapData_Full, NULL, NULL);
|
|
|
|
|
|
::SelectObject(m_hFullMemDC, m_BitmapHandle);
|
|
|
|
|
|
|
|
|
|
|
|
m_RectBuffer = new BYTE[m_BitmapInfor_Full->bmiHeader.biSizeImage * 2];
|
|
|
|
|
|
|
|
|
|
|
|
m_RectBufferOffset = 0;
|
|
|
|
|
|
|
|
|
|
|
|
m_hDiffMemDC = CreateCompatibleDC(m_hFullDC);
|
|
|
|
|
|
m_DiffBitmapHandle = ::CreateDIBSection(m_hFullDC, m_BitmapInfor_Full,
|
|
|
|
|
|
DIB_RGB_COLORS, &m_DiffBitmapData_Full, NULL, NULL);
|
|
|
|
|
|
::SelectObject(m_hDiffMemDC, m_DiffBitmapHandle);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
CScreenSpy::~CScreenSpy()
|
|
|
|
|
|
{
|
|
|
|
|
|
ReleaseDC(m_hDeskTopWnd, m_hFullDC); //GetDC
|
|
|
|
|
|
if (m_hFullMemDC!=NULL)
|
|
|
|
|
|
{
|
|
|
|
|
|
DeleteDC(m_hFullMemDC); //Createƥ<65><C6A5><EFBFBD>ڴ<EFBFBD>DC
|
|
|
|
|
|
|
|
|
|
|
|
::DeleteObject(m_BitmapHandle);
|
|
|
|
|
|
if (m_BitmapData_Full!=NULL)
|
|
|
|
|
|
{
|
|
|
|
|
|
m_BitmapData_Full = NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
m_hFullMemDC = NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (m_hDiffMemDC!=NULL)
|
|
|
|
|
|
{
|
|
|
|
|
|
DeleteDC(m_hDiffMemDC); //Createƥ<65><C6A5><EFBFBD>ڴ<EFBFBD>DC
|
|
|
|
|
|
|
|
|
|
|
|
::DeleteObject(m_DiffBitmapHandle);
|
|
|
|
|
|
if (m_DiffBitmapData_Full!=NULL)
|
|
|
|
|
|
{
|
|
|
|
|
|
m_DiffBitmapData_Full = NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (m_BitmapInfor_Full!=NULL)
|
|
|
|
|
|
{
|
|
|
|
|
|
delete[] m_BitmapInfor_Full;
|
|
|
|
|
|
m_BitmapInfor_Full = NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (m_RectBuffer)
|
|
|
|
|
|
{
|
|
|
|
|
|
delete[] m_RectBuffer;
|
|
|
|
|
|
m_RectBuffer = NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
m_RectBufferOffset = 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ULONG CScreenSpy::GetBISize()
|
|
|
|
|
|
{
|
|
|
|
|
|
ULONG ColorNum = m_ulbiBitCount <= 8 ? 1 << m_ulbiBitCount : 0;
|
|
|
|
|
|
|
|
|
|
|
|
return sizeof(BITMAPINFOHEADER) + (ColorNum * sizeof(RGBQUAD));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LPBITMAPINFO CScreenSpy::GetBIData()
|
|
|
|
|
|
{
|
|
|
|
|
|
return m_BitmapInfor_Full;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LPBITMAPINFO CScreenSpy::ConstructBI(ULONG ulbiBitCount,
|
|
|
|
|
|
ULONG ulFullWidth, ULONG ulFullHeight)
|
|
|
|
|
|
{
|
|
|
|
|
|
int ColorNum = ulbiBitCount <= 8 ? 1 << ulbiBitCount : 0;
|
|
|
|
|
|
ULONG ulBitmapLength = sizeof(BITMAPINFOHEADER) + (ColorNum * sizeof(RGBQUAD)); //BITMAPINFOHEADER +<2B><><EFBFBD><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD>ĸ<EFBFBD><C4B8><EFBFBD>
|
|
|
|
|
|
BITMAPINFO *BitmapInfor = (BITMAPINFO *) new BYTE[ulBitmapLength]; //[][]
|
|
|
|
|
|
|
|
|
|
|
|
BITMAPINFOHEADER* BitmapInforHeader = &(BitmapInfor->bmiHeader);
|
|
|
|
|
|
|
|
|
|
|
|
BitmapInforHeader->biSize = sizeof(BITMAPINFOHEADER);//pi si
|
|
|
|
|
|
BitmapInforHeader->biWidth = ulFullWidth; //1080
|
|
|
|
|
|
BitmapInforHeader->biHeight = ulFullHeight; //1920
|
|
|
|
|
|
BitmapInforHeader->biPlanes = 1;
|
|
|
|
|
|
BitmapInforHeader->biBitCount = ulbiBitCount; //32
|
|
|
|
|
|
BitmapInforHeader->biCompression = BI_RGB;
|
|
|
|
|
|
BitmapInforHeader->biXPelsPerMeter = 0;
|
|
|
|
|
|
BitmapInforHeader->biYPelsPerMeter = 0;
|
|
|
|
|
|
BitmapInforHeader->biClrUsed = 0;
|
|
|
|
|
|
BitmapInforHeader->biClrImportant = 0;
|
|
|
|
|
|
BitmapInforHeader->biSizeImage =
|
|
|
|
|
|
((BitmapInforHeader->biWidth * BitmapInforHeader->biBitCount + 31)/32)*4* BitmapInforHeader->biHeight;
|
|
|
|
|
|
|
|
|
|
|
|
// 16λ<36><CEBB><EFBFBD>Ժ<EFBFBD><D4BA><EFBFBD>û<EFBFBD><C3BB><EFBFBD><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD><EFBFBD>ֱ<EFBFBD>ӷ<EFBFBD><D3B7><EFBFBD>
|
|
|
|
|
|
|
|
|
|
|
|
return BitmapInfor;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LPVOID CScreenSpy::GetFirstScreenData()
|
|
|
|
|
|
{
|
|
|
|
|
|
//<2F><><EFBFBD>ڴ<EFBFBD>ԭ<EFBFBD>豸<EFBFBD>и<EFBFBD><D0B8><EFBFBD>λͼ<CEBB><CDBC>Ŀ<EFBFBD><C4BF><EFBFBD>豸
|
|
|
|
|
|
::BitBlt(m_hFullMemDC, 0, 0,
|
|
|
|
|
|
m_ulFullWidth, m_ulFullHeight, m_hFullDC, 0, 0, m_dwBitBltRop);
|
|
|
|
|
|
|
|
|
|
|
|
return m_BitmapData_Full; //<2F>ڴ<EFBFBD>
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
ULONG CScreenSpy::GetFirstScreenLength()
|
|
|
|
|
|
{
|
|
|
|
|
|
return m_BitmapInfor_Full->bmiHeader.biSizeImage;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LPVOID CScreenSpy::GetNextScreenData(ULONG* ulNextSendLength)
|
|
|
|
|
|
{
|
2019-01-10 19:35:03 +08:00
|
|
|
|
if (m_RectBuffer == NULL)
|
2019-01-05 20:21:43 +08:00
|
|
|
|
{
|
|
|
|
|
|
return NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD>rect<63><74><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ָ<EFBFBD><D6B8>
|
|
|
|
|
|
m_RectBufferOffset = 0;
|
|
|
|
|
|
|
|
|
|
|
|
// д<><D0B4>ʹ<EFBFBD><CAB9><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>㷨
|
|
|
|
|
|
WriteRectBuffer((LPBYTE)&m_bAlgorithm, sizeof(m_bAlgorithm));
|
|
|
|
|
|
|
|
|
|
|
|
// д<><D0B4><EFBFBD><EFBFBD><EFBFBD><EFBFBD>λ<EFBFBD><CEBB>
|
|
|
|
|
|
POINT CursorPos;
|
|
|
|
|
|
GetCursorPos(&CursorPos);
|
|
|
|
|
|
WriteRectBuffer((LPBYTE)&CursorPos, sizeof(POINT));
|
|
|
|
|
|
|
|
|
|
|
|
// д<>뵱ǰ<EBB5B1><C7B0><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
|
BYTE bCursorIndex = m_CursorInfor.GetCurrentCursorIndex();
|
|
|
|
|
|
WriteRectBuffer(&bCursorIndex, sizeof(BYTE));
|
|
|
|
|
|
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD><EFBFBD>Ƚ<EFBFBD><C8BD>㷨
|
|
|
|
|
|
if (m_bAlgorithm == ALGORITHM_DIFF)
|
|
|
|
|
|
{
|
|
|
|
|
|
// <20>ֶ<EFBFBD>ɨ<EFBFBD><C9A8>ȫ<EFBFBD><C8AB>Ļ <20><><EFBFBD>µ<EFBFBD>λͼ<CEBB><CDBC><EFBFBD>뵽m_hDiffMemDC<44><43>
|
|
|
|
|
|
ScanScreen(m_hDiffMemDC, m_hFullDC, m_BitmapInfor_Full->bmiHeader.biWidth,
|
|
|
|
|
|
m_BitmapInfor_Full->bmiHeader.biHeight);
|
|
|
|
|
|
|
|
|
|
|
|
//<2F><><EFBFBD><EFBFBD>Bit<69><74><EFBFBD>бȽ<D0B1><C8BD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>һ<EFBFBD><D2BB><EFBFBD><EFBFBD>m_lpvFullBits<74>еķ<D0B5><C4B7><EFBFBD>
|
|
|
|
|
|
*ulNextSendLength = m_RectBufferOffset +
|
|
|
|
|
|
CompareBitmap((LPBYTE)m_DiffBitmapData_Full, (LPBYTE)m_BitmapData_Full,
|
2019-01-10 19:35:03 +08:00
|
|
|
|
m_RectBuffer + m_RectBufferOffset, m_BitmapInfor_Full->bmiHeader.biSizeImage);
|
|
|
|
|
|
|
2019-01-05 20:21:43 +08:00
|
|
|
|
return m_RectBuffer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
VOID CScreenSpy::WriteRectBuffer(LPBYTE szBuffer,ULONG ulLength)
|
|
|
|
|
|
{
|
|
|
|
|
|
memcpy(m_RectBuffer + m_RectBufferOffset, szBuffer, ulLength);
|
|
|
|
|
|
m_RectBufferOffset += ulLength;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2019-01-07 20:46:03 +08:00
|
|
|
|
|
2019-01-05 20:21:43 +08:00
|
|
|
|
VOID CScreenSpy::ScanScreen(HDC hdcDest, HDC hdcSour, ULONG ulWidth, ULONG ulHeight)
|
|
|
|
|
|
{
|
2019-01-10 19:35:03 +08:00
|
|
|
|
AUTO_TICK(1);
|
2019-01-07 20:46:03 +08:00
|
|
|
|
#ifdef COPY_ALL
|
|
|
|
|
|
BitBlt(hdcDest, 0, 0, ulWidth, ulHeight, hdcSour, 0, 0, m_dwBitBltRop);
|
|
|
|
|
|
#else
|
|
|
|
|
|
const ULONG ulJumpLine = 50;
|
|
|
|
|
|
const ULONG ulJumpSleep = ulJumpLine / 10;
|
2019-01-05 20:21:43 +08:00
|
|
|
|
|
|
|
|
|
|
for (int i = 0, ulToJump = 0; i < ulHeight; i += ulToJump)
|
|
|
|
|
|
{
|
|
|
|
|
|
ULONG ulv1 = ulHeight - i;
|
|
|
|
|
|
|
|
|
|
|
|
if (ulv1 > ulJumpLine)
|
|
|
|
|
|
ulToJump = ulJumpLine;
|
|
|
|
|
|
else
|
|
|
|
|
|
ulToJump = ulv1;
|
|
|
|
|
|
BitBlt(hdcDest, 0, i, ulWidth, ulToJump, hdcSour,0, i, m_dwBitBltRop);
|
|
|
|
|
|
Sleep(ulJumpSleep);
|
|
|
|
|
|
}
|
2019-01-07 20:46:03 +08:00
|
|
|
|
#endif
|
2019-01-05 20:21:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2019-01-10 19:35:03 +08:00
|
|
|
|
|
2019-01-05 20:21:43 +08:00
|
|
|
|
ULONG CScreenSpy::CompareBitmap(LPBYTE CompareSourData, LPBYTE CompareDestData,
|
|
|
|
|
|
LPBYTE szBuffer, DWORD ulCompareLength)
|
|
|
|
|
|
{
|
2019-01-10 19:35:03 +08:00
|
|
|
|
AUTO_TICK(1);
|
2019-01-05 20:21:43 +08:00
|
|
|
|
// Windows<77>涨һ<E6B6A8><D2BB>ɨ<EFBFBD><C9A8><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ռ<EFBFBD><D5BC><EFBFBD>ֽ<EFBFBD><D6BD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>4<EFBFBD>ı<EFBFBD><C4B1><EFBFBD>, <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>DWORD<52>Ƚ<EFBFBD>
|
2019-01-10 19:35:03 +08:00
|
|
|
|
LPDWORD p1 = (LPDWORD)CompareDestData, p2 = (LPDWORD)CompareSourData;
|
2019-01-05 20:21:43 +08:00
|
|
|
|
// ƫ<>Ƶ<EFBFBD>ƫ<EFBFBD>ƣ<EFBFBD><C6A3><EFBFBD>ͬ<EFBFBD><CDAC><EFBFBD>ȵ<EFBFBD>ƫ<EFBFBD><C6AB>
|
2019-01-10 19:35:03 +08:00
|
|
|
|
ULONG ulszBufferOffset = 0, ulv1 = 0, ulv2 = 0, ulCount = 0;
|
|
|
|
|
|
for (int i = 0; i < ulCompareLength; i += 4, ++p1, ++p2)
|
2019-01-05 20:21:43 +08:00
|
|
|
|
{
|
2019-01-10 19:35:03 +08:00
|
|
|
|
if (*p1 == *p2)
|
2019-01-05 20:21:43 +08:00
|
|
|
|
continue;
|
|
|
|
|
|
|
2019-01-10 19:35:03 +08:00
|
|
|
|
*(LPDWORD)(szBuffer + ulszBufferOffset) = i;
|
2019-01-05 20:21:43 +08:00
|
|
|
|
// <20><>¼<EFBFBD><C2BC><EFBFBD>ݴ<EFBFBD>С<EFBFBD>Ĵ<EFBFBD><C4B4><EFBFBD>λ<EFBFBD><CEBB>
|
2019-01-10 19:35:03 +08:00
|
|
|
|
ulv1 = ulszBufferOffset + sizeof(int);
|
|
|
|
|
|
ulv2 = ulv1 + sizeof(int);
|
2019-01-05 20:21:43 +08:00
|
|
|
|
ulCount = 0; // <20><><EFBFBD>ݼ<EFBFBD><DDBC><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
|
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD>Dest<73>е<EFBFBD><D0B5><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
|
*p1 = *p2;
|
|
|
|
|
|
*(LPDWORD)(szBuffer + ulv2 + ulCount) = *p2;
|
|
|
|
|
|
|
|
|
|
|
|
ulCount += 4;
|
|
|
|
|
|
i += 4, p1++, p2++;
|
2019-01-10 19:35:03 +08:00
|
|
|
|
for (int j = i; j < ulCompareLength; j += 4, i += 4, ++p1, ++p2)
|
2019-01-05 20:21:43 +08:00
|
|
|
|
{
|
|
|
|
|
|
if (*p1 == *p2)
|
|
|
|
|
|
break;
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD>Dest<73>е<EFBFBD><D0B5><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
|
*p1 = *p2;
|
|
|
|
|
|
*(LPDWORD)(szBuffer + ulv2 + ulCount) = *p2;
|
|
|
|
|
|
ulCount += 4;
|
|
|
|
|
|
}
|
|
|
|
|
|
// д<><D0B4><EFBFBD><EFBFBD><EFBFBD>ݳ<EFBFBD><DDB3><EFBFBD>
|
|
|
|
|
|
*(LPDWORD)(szBuffer + ulv1) = ulCount;
|
2019-01-10 19:35:03 +08:00
|
|
|
|
ulszBufferOffset = ulv2 + ulCount;
|
2019-01-05 20:21:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return ulszBufferOffset;
|
|
|
|
|
|
}
|