Not
Bu sayfaya erişim yetkilendirme gerektiriyor. Oturum açmayı veya dizinleri değiştirmeyi deneyebilirsiniz.
Bu sayfaya erişim yetkilendirme gerektiriyor. Dizinleri değiştirmeyi deneyebilirsiniz.
Bu bölüm, fare girişiyle ilişkili görevleri kapsar.
- Fare İmlecini İzleme
- Fare ile Çizgi Çizmeyi
- Çift Tıklama Mesajını İşleme
- Metin Satırı Seçme
- Bir Belgede Eklenmiş Nesnelerle Fare Tekerleği Kullanma
- Fare Tekerleği Kaydırma Çizgisi Sayısını Alma
Fare İmlecini İzleme
Uygulamalar genellikle fare imlecinin konumunu izlemeyi içeren görevleri gerçekleştirir. Örneğin çoğu çizim uygulaması, çizim işlemleri sırasında fare imlecinin konumunu izleyerek kullanıcının fareyi sürükleyerek pencerenin istemci alanında çizim yapmasına olanak tanır. Sözcük işleme uygulamaları da imleci izleyerek kullanıcının fareye tıklayıp sürükleyerek bir sözcük veya metin bloğu seçmesini sağlar.
İmleci izlemek genellikle WM_LBUTTONDOWN, WM_MOUSEMOVEve WM_LBUTTONUP iletilerinin işlenmesini içerir. Pencere, WM_LBUTTONDOWN iletisinin lParam parametresinde sağlanan imleç konumunu denetleyerek imleci izlemeye ne zaman başlayacağını belirler. Örneğin, bir kelime işleme uygulaması, imleç bir metin satırındayken WM_LBUTTONDOWN iletisi oluştuğunda imleci izlemeye başlar, ancak imleç belgenin sonundan ötesine geçtiyse izlemeye başlamaz.
Pencere, fare hareket ettikçe pencereye gönderilen WM_MOUSEMOVE iletilerinin akışını işleyerek imlecin konumunu izler. WM_MOUSEMOVE iletisinin işlenmesi genellikle istemci alanında yinelenen bir boyama veya çizim işlemi içerir. Örneğin, bir çizim uygulaması fare hareket ettikçe bir çizgiyi art arda yeniden çizebilir. Pencere, imleci izlemeyi durdurmak için sinyal olarak WM_LBUTTONUP iletisini kullanır.
Ayrıca bir uygulama, sistemin imleci izlemek için yararlı olan diğer iletileri göndermesini sağlamak için TrackMouseEvent işlevini çağırabilir. İmleç belirli bir süre boyunca istemci alanının üzerine geldiğinde sistem WM_MOUSEHOVER iletisini postalar. İmleç istemci alanından ayrıldığında WM_MOUSELEAVE iletisini postalar. WM_NCMOUSEHOVER ve WM_NCMOUSELEAVE iletileri, istemci olmayan alanlar için karşılık gelen iletilerdir.
Fare ile Çizim Yapma
Bu bölümdeki örnekte fare imlecini izleme gösterilmektedir. Kullanıcının fareyi sürükleyerek pencerenin istemci alanına çizgiler çizmesini sağlayan pencere yordamının bölümlerini içerir.
Pencere yordamı bir WM_LBUTTONDOWN iletisi aldığında, fareyi yakalar ve çizginin başlangıç noktası olarak koordinatları kullanarak imlecin koordinatlarını kaydeder. Ayrıca, çizgi çizme işlemi sırasında imleci istemci alanıyla sınırlamak için ClipCursor işlevini kullanır.
İlk WM_MOUSEMOVE iletisi sırasında, pencere yordamı başlangıç noktasından imlecin geçerli konumuna bir çizgi çizer. Sonraki WM_MOUSEMOVE iletilerde pencere yordamı, önceki satırı ters kalem rengiyle çizerek siler. Ardından başlangıç noktasından imlecin yeni konumuna yeni bir çizgi çizer.
WM_LBUTTONUP iletisi, çizim işleminin sonuna işaret eder. Pencere yordamı fare yakalamayı serbest bırakır ve fareyi istemci alanından serbest bırakır.
LRESULT APIENTRY MainWndProc(HWND hwndMain, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
HDC hdc; // handle to device context
RECT rcClient; // client area rectangle
POINT ptClientUL; // client upper left corner
POINT ptClientLR; // client lower right corner
static POINTS ptsBegin; // beginning point
static POINTS ptsEnd; // new endpoint
static POINTS ptsPrevEnd; // previous endpoint
static BOOL fPrevLine = FALSE; // previous line flag
switch (uMsg)
{
case WM_LBUTTONDOWN:
// Capture mouse input.
SetCapture(hwndMain);
// Retrieve the screen coordinates of the client area,
// and convert them into client coordinates.
GetClientRect(hwndMain, &rcClient);
ptClientUL.x = rcClient.left;
ptClientUL.y = rcClient.top;
// Add one to the right and bottom sides, because the
// coordinates retrieved by GetClientRect do not
// include the far left and lowermost pixels.
ptClientLR.x = rcClient.right + 1;
ptClientLR.y = rcClient.bottom + 1;
ClientToScreen(hwndMain, &ptClientUL);
ClientToScreen(hwndMain, &ptClientLR);
// Copy the client coordinates of the client area
// to the rcClient structure. Confine the mouse cursor
// to the client area by passing the rcClient structure
// to the ClipCursor function.
SetRect(&rcClient, ptClientUL.x, ptClientUL.y,
ptClientLR.x, ptClientLR.y);
ClipCursor(&rcClient);
// Convert the cursor coordinates into a POINTS
// structure, which defines the beginning point of the
// line drawn during a WM_MOUSEMOVE message.
ptsBegin = MAKEPOINTS(lParam);
return 0;
case WM_MOUSEMOVE:
// When moving the mouse, the user must hold down
// the left mouse button to draw lines.
if (wParam & MK_LBUTTON)
{
// Retrieve a device context (DC) for the client area.
hdc = GetDC(hwndMain);
// The following function ensures that pixels of
// the previously drawn line are set to white and
// those of the new line are set to black.
SetROP2(hdc, R2_NOTXORPEN);
// If a line was drawn during an earlier WM_MOUSEMOVE
// message, draw over it. This erases the line by
// setting the color of its pixels to white.
if (fPrevLine)
{
MoveToEx(hdc, ptsBegin.x, ptsBegin.y,
(LPPOINT) NULL);
LineTo(hdc, ptsPrevEnd.x, ptsPrevEnd.y);
}
// Convert the current cursor coordinates to a
// POINTS structure, and then draw a new line.
ptsEnd = MAKEPOINTS(lParam);
MoveToEx(hdc, ptsBegin.x, ptsBegin.y, (LPPOINT) NULL);
LineTo(hdc, ptsEnd.x, ptsEnd.y);
// Set the previous line flag, save the ending
// point of the new line, and then release the DC.
fPrevLine = TRUE;
ptsPrevEnd = ptsEnd;
ReleaseDC(hwndMain, hdc);
}
break;
case WM_LBUTTONUP:
// The user has finished drawing the line. Reset the
// previous line flag, release the mouse cursor, and
// release the mouse capture.
fPrevLine = FALSE;
ClipCursor(NULL);
ReleaseCapture();
return 0;
case WM_DESTROY:
PostQuitMessage(0);
break;
// Process other messages.
}
}
Çift Tıklama İletisini İşleme
Çift tıklama iletileri almak için, pencerenin CS_DBLCLKS sınıf stiline sahip bir pencere sınıfına ait olması gerekir. Aşağıdaki örnekte gösterildiği gibi, pencere sınıfını kaydederken bu stili ayarlarsınız.
BOOL InitApplication(HINSTANCE hInstance)
{
WNDCLASS wc;
wc.style = CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW;
wc.lpfnWndProc = (WNDPROC) MainWndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hInstance;
wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
wc.hCursor = LoadCursor(NULL, IDC_IBEAM);
wc.hbrBackground = GetStockObject(WHITE_BRUSH);
wc.lpszMenuName = "MainMenu";
wc.lpszClassName = "MainWClass";
return RegisterClass(&wc);
}
Çift tıklama iletisinden önce her zaman bir düğmeye basma iletisi gelir. Bu nedenle, uygulamalar genellikle düğme aşağı ileti sırasında başladığı görevi genişletmek için çift tıklama iletisi kullanır.
Metin Satırı Seçme
Bu bölümdeki örnek, basit bir sözcük işleme uygulamasından alınmıştır. Kullanıcının metin satırında herhangi bir yere tıklayarak şapka işaretinin konumunu ayarlamasını ve satırda herhangi bir yere çift tıklayarak metin satırını seçmesini (vurgulamasını) sağlayan kod içerir.
LRESULT APIENTRY MainWndProc(HWND hwndMain, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
HDC hdc; // handle to device context
TEXTMETRIC tm; // font size data
int i, j; // loop counters
int cCR = 0; // count of carriage returns
char ch; // character from input buffer
static int nBegLine; // beginning of selected line
static int nCurrentLine = 0; // currently selected line
static int nLastLine = 0; // last text line
static int nCaretPosX = 0; // x-coordinate of caret
static int cch = 0; // number of characters entered
static int nCharWidth = 0; // exact width of a character
static char szHilite[128]; // text string to highlight
static DWORD dwCharX; // average width of characters
static DWORD dwLineHeight; // line height
static POINTS ptsCursor; // coordinates of mouse cursor
static COLORREF crPrevText; // previous text color
static COLORREF crPrevBk; // previous background color
static PTCHAR pchInputBuf; // pointer to input buffer
static BOOL fTextSelected = FALSE; // text-selection flag
size_t * pcch;
HRESULT hResult;
switch (uMsg)
{
case WM_CREATE:
// Get the metrics of the current font.
hdc = GetDC(hwndMain);
GetTextMetrics(hdc, &tm);
ReleaseDC(hwndMain, hdc);
// Save the average character width and height.
dwCharX = tm.tmAveCharWidth;
dwLineHeight = tm.tmHeight;
// Allocate a buffer to store keyboard input.
pchInputBuf = (LPSTR) GlobalAlloc(GPTR,
BUFSIZE * sizeof(TCHAR));
return 0;
case WM_CHAR:
switch (wParam)
{
case 0x08: // backspace
case 0x0A: // linefeed
case 0x1B: // escape
MessageBeep( (UINT) -1);
return 0;
case 0x09: // tab
// Convert tabs to four consecutive spaces.
for (i = 0; i < 4; i++)
SendMessage(hwndMain, WM_CHAR, 0x20, 0);
return 0;
case 0x0D: // carriage return
// Record the carriage return, and position the
// caret at the beginning of the new line.
pchInputBuf[cch++] = 0x0D;
nCaretPosX = 0;
nCurrentLine += 1;
break;
default: / displayable character
ch = (char) wParam;
HideCaret(hwndMain);
// Retrieve the character's width, and display the
// character.
hdc = GetDC(hwndMain);
GetCharWidth32(hdc, (UINT) wParam, (UINT) wParam,
&nCharWidth);
TextOut(hdc, nCaretPosX,
nCurrentLine * dwLineHeight, &ch, 1);
ReleaseDC(hwndMain, hdc);
// Store the character in the buffer.
pchInputBuf[cch++] = ch;
// Calculate the new horizontal position of the
// caret. If the new position exceeds the maximum,
// insert a carriage return and reposition the
// caret at the beginning of the next line.
nCaretPosX += nCharWidth;
if ((DWORD) nCaretPosX > dwMaxCharX)
{
nCaretPosX = 0;
pchInputBuf[cch++] = 0x0D;
++nCurrentLine;
}
ShowCaret(hwndMain);
break;
}
SetCaretPos(nCaretPosX, nCurrentLine * dwLineHeight);
nLastLine = max(nLastLine, nCurrentLine);
break;
// Process other messages.
case WM_LBUTTONDOWN:
// If a line of text is currently highlighted, redraw
// the text to remove the highlighting.
if (fTextSelected)
{
hdc = GetDC(hwndMain);
SetTextColor(hdc, crPrevText);
SetBkColor(hdc, crPrevBk);
hResult = StringCchLength(szHilite, 128/sizeof(TCHAR), pcch);
if (FAILED(hResult))
{
// TODO: write error handler
}
TextOut(hdc, 0, nCurrentLine * dwLineHeight,
szHilite, *pcch);
ReleaseDC(hwndMain, hdc);
ShowCaret(hwndMain);
fTextSelected = FALSE;
}
// Save the current mouse-cursor coordinates.
ptsCursor = MAKEPOINTS(lParam);
// Determine which line the cursor is on, and save
// the line number. Do not allow line numbers greater
// than the number of the last line of text. The
// line number is later multiplied by the average height
// of the current font. The result is used to set the
// y-coordinate of the caret.
nCurrentLine = min((int)(ptsCursor.y / dwLineHeight),
nLastLine);
// Parse the text input buffer to find the first
// character in the selected line of text. Each
// line ends with a carriage return, so it is possible
// to count the carriage returns to find the selected
// line.
cCR = 0;
nBegLine = 0;
if (nCurrentLine != 0)
{
for (i = 0; (i < cch) &&
(cCR < nCurrentLine); i++)
{
if (pchInputBuf[i] == 0x0D)
++cCR;
}
nBegLine = i;
}
// Starting at the beginning of the selected line,
// measure the width of each character, summing the
// width with each character measured. Stop when the
// sum is greater than the x-coordinate of the cursor.
// The sum is used to set the x-coordinate of the caret.
hdc = GetDC(hwndMain);
nCaretPosX = 0;
for (i = nBegLine;
(pchInputBuf[i] != 0x0D) && (i < cch); i++)
{
ch = pchInputBuf[i];
GetCharWidth32(hdc, (int) ch, (int) ch, &nCharWidth);
if ((nCaretPosX + nCharWidth) > ptsCursor.x) break;
else nCaretPosX += nCharWidth;
}
ReleaseDC(hwndMain, hdc);
// Set the caret to the user-selected position.
SetCaretPos(nCaretPosX, nCurrentLine * dwLineHeight);
break;
case WM_LBUTTONDBLCLK:
// Copy the selected line of text to a buffer.
for (i = nBegLine, j = 0; (pchInputBuf[i] != 0x0D) &&
(i < cch); i++)
{
szHilite[j++] = pchInputBuf[i];
}
szHilite[j] = '\0';
// Hide the caret, invert the background and foreground
// colors, and then redraw the selected line.
HideCaret(hwndMain);
hdc = GetDC(hwndMain);
crPrevText = SetTextColor(hdc, RGB(255, 255, 255));
crPrevBk = SetBkColor(hdc, RGB(0, 0, 0));
hResult = StringCchLength(szHilite, 128/sizeof(TCHAR), pcch);
if (FAILED(hResult))
{
// TODO: write error handler
}
TextOut(hdc, 0, nCurrentLine * dwLineHeight, szHilite, *pcch);
SetTextColor(hdc, crPrevText);
SetBkColor(hdc, crPrevBk);
ReleaseDC(hwndMain, hdc);
fTextSelected = TRUE;
break;
// Process other messages.
default:
return DefWindowProc(hwndMain, uMsg, wParam, lParam);
}
return NULL;
}
Eklenmiş Nesnelerle Belgede Fare Tekerleği Kullanma
Bu örnekte, çeşitli eklenmiş nesneler içeren bir Microsoft Word belgesi varsayılır:
- Microsoft Excel elektronik tablosu
- Tekerleğe yanıt olarak kaydırılan gömülü liste kutusu denetimi
- Tekerleğe yanıt vermeyen gömülü metin kutusu denetimi
MSH_MOUSEWHEEL iletisi her zaman Microsoft Word'deki ana pencereye gönderilir. Katıştırılmış elektronik tablo etkin olsa bile bu durum geçerlidir. Aşağıdaki tabloda, MSH_MOUSEWHEEL iletisinin odak durumuna göre nasıl işleneceği açıklanmıştır.
| Odak açık | İşleme aşağıdaki gibidir |
|---|---|
| Word belgesi | Word belge penceresini kaydırıyor. |
| Eklenmiş Excel elektronik tablosu | Word iletiyi Excel'e postalar. Katıştırılmış uygulamanın iletiye yanıt verip vermeyeceğine karar vermelisiniz. |
| Eklenmiş denetim | İletiyi odağı olan ekli bir denetime göndermek ve denetimin bunu işleyip işlemediğini görmek için dönüş kodunu denetlemek uygulamaya bağlı. Denetim bunu işlemediyse, uygulamanın belge penceresini kaydırması gerekir. Örneğin, kullanıcı bir liste kutusuna tıklar ve ardından tekerleği yuvarlarsa, bu denetim tekerleği döndürmeye yanıt olarak kaydırılır. Kullanıcı bir metin kutusuna tıklayıp tekerleği döndürdüyse, belgenin tamamı kaydırılır. |
Aşağıdaki örnekte, bir uygulamanın iki tekerlek iletisini nasıl işleyebilecekleri gösterilmektedir.
/************************************************
* this code deals with MSH_MOUSEWHEEL
*************************************************/
#include "zmouse.h"
//
// Mouse Wheel rotation stuff, only define if we are
// on a version of the OS that does not support
// WM_MOUSEWHEEL messages.
//
#ifndef WM_MOUSEWHEEL
#define WM_MOUSEWHEEL WM_MOUSELAST+1
// Message ID for IntelliMouse wheel
#endif
UINT uMSH_MOUSEWHEEL = 0; // Value returned from
// RegisterWindowMessage()
/**************************************************
INT WINAPI WinMain(
HINSTANCE hInst,
HINSTANCE hPrevInst,
LPSTR lpCmdLine,
INT nCmdShow)
{
MSG msg;
BOOL bRet;
if (!InitInstance(hInst, nCmdShow))
return FALSE;
//
// The new IntelliMouse uses a Registered message to transmit
// wheel rotation info. So register for it!
uMSH_MOUSEWHEEL =
RegisterWindowMessage(MSH_MOUSEWHEEL);
if ( !uMSH_MOUSEWHEEL )
{
MessageBox(NULL,"
RegisterWindowMessag Failed!",
"Error",MB_OK);
return msg.wParam;
}
while (( bRet = GetMessage(&msg, NULL, 0, 0)) != 0)
{
if (bRet == -1)
{
// handle the error and possibly exit
}
else
{
if (!TranslateAccelerator(ghwndApp,
ghaccelTable,
&msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
}
return msg.wParam;
}
/************************************************
* this code deals with WM_MOUSEWHEEL
*************************************************/
LONG APIENTRY MainWndProc(
HWND hwnd,
UINT msg,
WPARAM wParam,
LPARAM lParam)
{
static int nZoom = 0;
switch (msg)
{
//
// Handle Mouse Wheel messages generated
// by the operating systems that have built-in
// support for the WM_MOUSEWHEEL message.
//
case WM_MOUSEWHEEL:
((short) HIWORD(wParam)< 0) ? nZoom-- : nZoom++;
//
// Do other wheel stuff...
//
break;
default:
//
// uMSH_MOUSEWHEEL is a message registered by
// the mswheel dll on versions of Windows that
// do not support the new message in the OS.
if( msg == uMSH_MOUSEWHEEL )
{
((int)wParam < 0) ? nZoom-- : nZoom++;
//
// Do other wheel stuff...
//
break;
}
return DefWindowProc(hwnd,
msg,
wParam,
lParam);
}
return 0L;
}
Fare Tekerleği Kaydırma Çizgilerinin Sayısını Alma
Aşağıdaki kod, bir uygulamanın SystemParametersInfo işlevini kullanarak kaydırma çizgilerinin sayısını almasını sağlar.
#ifndef SPI_GETWHEELSCROLLLINES
#define SPI_GETWHEELSCROLLLINES 104
#endif
#include "zmouse.h"
/*********************************************************
* FUNCTION: GetNumScrollLines
* Purpose : An OS independent method to retrieve the
* number of wheel scroll lines
* Params : none
* Returns : UINT: Number of scroll lines where WHEEL_PAGESCROLL
* indicates to scroll a page at a time.
*********************************************************/
UINT GetNumScrollLines(void)
{
HWND hdlMsWheel;
UINT ucNumLines=3; // 3 is the default
OSVERSIONINFO osversion;
UINT uiMsh_MsgScrollLines;
memset(&osversion, 0, sizeof(osversion));
osversion.dwOSVersionInfoSize =sizeof(osversion);
GetVersionEx(&osversion);
if ((osversion.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS) ||
( (osversion.dwPlatformId == VER_PLATFORM_WIN32_NT) &&
(osversion.dwMajorVersion < 4) ) )
{
hdlMsWheel = FindWindow(MSH_WHEELMODULE_CLASS,
MSH_WHEELMODULE_TITLE);
if (hdlMsWheel)
{
uiMsh_MsgScrollLines = RegisterWindowMessage
(MSH_SCROLL_LINES);
if (uiMsh_MsgScrollLines)
ucNumLines = (int)SendMessage(hdlMsWheel,
uiMsh_MsgScrollLines,
0,
0);
}
}
else if ( (osversion.dwPlatformId ==
VER_PLATFORM_WIN32_NT) &&
(osversion.dwMajorVersion >= 4) )
{
SystemParametersInfo(SPI_GETWHEELSCROLLLINES,
0,
&ucNumLines, 0);
}
return(ucNumLines);
}