-
Доброе время суток. Пробую переписать с VCL Экранную лупу. Вот пока что имеется. program Project1;
uses
windows,messages,
kol;
const
Diametre = 300;
var
form,Loupe:PControl;
hRegion: HRgn;
Width:integer;
DeskDC : HDC;
HDeskBmp : HBitmap;
procedure MouseDown(Sender: PControl;var mouse:TMouseEventData);
begin
releasecapture();
form.Perform(wm_syscommand,$F012,0);
end;
procedure ScreenRelease;
begin
if HDeskBmp<> 0 then
begin
DeleteObject(HDeskBmp);
HDeskBmp:= 0;
end;
if DeskDC<> 0 then
begin
ReleaseDC(GetDesktopWindow, DeskDC);
DeskDC:= 0;
end;
end;
procedure ScreenCapture;
var
MemDC: HDC;
W, H: Integer;
begin
if (HDeskBmp<> 0)or (DeskDC<> 0) then
Exit;
W:= GetSystemMetrics(SM_CXSCREEN);
H:= GetSystemMetrics(SM_CYSCREEN);
DeskDC:= GetDC(GetDesktopWindow);
MemDC:= CreateCompatibleDC(DeskDC);
HDeskBmp:= CreateCompatibleBitmap(DeskDC, W, H);
SelectObject(MemDC, HDeskBmp);
BitBlt(MemDC, 0, 0, W, H, DeskDC, 0, 0, SRCCOPY);
DeleteDC(MemDC);
end;
procedure DrawFilter;
var
DrawToDC: HDC;
MemDC: HDC;
OldBmp: HBitMap;
OffsetX, OffsetY: Integer;
begin
if Loupe= nil then Exit;
DrawToDC:= Loupe.Canvas.Handle;
// Копируем фоновое изображение
MemDC:= CreateCompatibleDC(DeskDC);
OldBmp:= SelectObject(MemDC, HDeskBmp);
OffsetX:= Width div 4;
OffsetY:= Width div 4;
StretchBlt(DrawToDC, 0, 0, Width, Width, MemDC, form.Left + OffsetX, form.Top + OffsetY, Width div 2, Width div 2, SRCCOPY);
SelectObject(MemDC, OldBmp);
DeleteObject(MemDC);
with Loupe.Canvas^ do
begin
Brush.Color:= clYellow;
Brush.BrushStyle:= bsClear;
Pen.PenWidth:=2;
Pen.Color:= clYellow;
Pen.PenStyle:= psSolid;
Ellipse(1, 1, Width- 2, Width- 2);
Pen.Color:= clOlive;
Pen.PenStyle:= psSolid;
Ellipse(2, 2, Width- 1, Width- 1);
end;
end;
function FormMessage(Dummy : Pointer;var Msg: tagMSG; var Rslt: Integer): Boolean;
begin
Result := False;
if Msg.message = WM_NCACTIVATE then
ScreenCapture else ScreenRelease;
if Msg.message = WM_ERASEBKGND then
begin
Rslt := 1;
Result := TRUE;
end;
if Msg.message = WM_WINDOWPOSCHANGED then //or WM_WINDOWPOSCHANGING
begin
Rslt := 1;
Result := TRUE;
DrawFilter;
Form.Invalidate;
form.Update;
end;
//if Msg.message = WM_PAINT then form.Canvas.draw(0,0,Loupe);
end;
procedure FormPaint(Sender: TObject);
begin
//?.Draw(Loupe.Canvas.Handle,0,0); //???
end;//в VCL было form.Canvas.draw(0,0,Loupe);
procedure FormDestroy(Sender: TObject);
begin
ScreenRelease;
Loupe.Free;
end;
Begin
Applet := NewApplet('');
form:=NewForm(Applet,'Лупа').centeronparent.Tabulate;
Width := Diametre;
Loupe := NewPaintBox(form);
Loupe.Width:= Width;
Loupe.Height:=Width;
hRegion:= CreateEllipticRgn(0, 0, Width, Width);
if hRegion<> 0 then SetWindowRgn(form.GetWindowHandle , hRegion, TRUE);
HDeskBmp:= 0;
DeskDc:= 0;
ScreenCapture;
DrawFilter;
Loupe.OnMessage:=TOnMessage(MakeMethod(nil,@FormMessage));
Loupe.OnMouseDown:=TOnMouse(MakeMethod(nil,@MouseDown));
Loupe.OnPaint:=TOnPaint( MakeMethod( nil, @DrawFilter ));
Run(form);
end.
1 Рисование так и не выходит. 2 Виден заголовок.
-
Внимание - вопрос:
-
I once wrote a simple screen magnifier with KOL - maybe my code will help you:
program Magnify;
uses
Windows, Messages, KOL;
const
ZoomRatio: Integer = 2;
var
PaintBox: PControl;
Image: PBitmap;
procedure PaintPreview(Dummy: Pointer; Sender: PControl; DC: HDC);
begin
Image.Draw(Sender.Canvas.Handle, 0,0);
end;
procedure TimerTick(Dummy: Pointer; Sender: PObj);
var
ZoomRect: TRect;
DC: HDC;
Cursor: TPoint;
begin
GetCursorPos(Cursor);
ZoomRect := MakeRect(Cursor.X, Cursor.Y, Cursor.X, Cursor.Y);
InflateRect(ZoomRect, Round(Image.Width / (2 * ZoomRatio)), Round(Image.Height / (2 * ZoomRatio)));
if ZoomRect.Left < 0 then
OffsetRect(ZoomRect, -ZoomRect.Left, 0);
if ZoomRect.Top < 0 then
OffsetRect(ZoomRect, 0, -ZoomRect.Top);
if ZoomRect.Right > ScreenWidth then
OffsetRect(ZoomRect, -(ZoomRect.Right - ScreenWidth), 0);
if ZoomRect.Bottom > ScreenHeight then
OffsetRect(ZoomRect, 0, -(ZoomRect.Bottom - ScreenHeight));
DC := GetDC(GetDesktopWindow);
StretchBlt(Image.Canvas.Handle, 0, 0, Image.Width, Image.Height, DC, ZoomRect.Left, ZoomRect.Top, ZoomRect.Right - ZoomRect.Left, ZoomRect.Bottom - ZoomRect.Top, SRCCOPY);
PaintBox.Invalidate;
ReleaseDC(GetDesktopWindow, DC);
end;
procedure MouseDown(Dummy: Pointer; Sender: PControl; var Mouse: TMouseEventData);
begin
case Mouse.Button of
mbLeft: Sender.WindowedParent.DragStart;
mbRight: Applet.Close;
end;
end;
procedure AppResize(Dummy: Pointer; Sender: PObj);
begin
Image.Width := PaintBox.Width;
Image.Height := PaintBox.Height;
end;
begin
Applet := NewForm(Applet, 'KOL Zoomer');
with Applet^ do
begin
StayOnTop := True;
MinWidth := 120;
MinHeight := 120;
MaxWidth := ScreenWidth div 2;
MaxHeight := ScreenHeight div 2;
Style := Style and not WS_MAXIMIZEBOX;
Perform(WM_INITMENU, 0, 0);
SetSize(MinWidth, MinHeight);
HasCaption := False;
end;
PaintBox := NewPaintbox(Applet);
with PaintBox^ do
begin
Align := caClient;
OnPaint := TOnPaint(MakeMethod(nil, @PaintPreview));
OnMouseDown := TOnMouse(MakeMethod(nil, @MouseDown));
OnResize := TOnEvent(MakeMethod(nil, @AppResize));
end;
Image := NewBitmap(Applet.ClientWidth, Applet.ClientHeight);
with NewTimer(20)^ do
begin
OnTimer := TOnEvent(MakeMethod(nil, @TimerTick));
Enabled := True;
end;
SetProcessWorkingSetSize(GetCurrentProcess, Cardinal(-1), Cardinal(-1));
Run(Applet);
end.
-
Thank you very much. You really helped me.
-
I'm glad that it was of help. If you improve the code then please share it.
-
Теряется оконтовка при рисовании, пока сделал ещё один регион. StretchBlt(Img.Canvas.Handle, 0, 0, Width, Width, MemDC, form.Left + OffsetX, form.Top + OffsetY, Width div 2, Width div 2, SRCCOPY);
Loupe.Invalidate; SelectObject(MemDC, OldBmp);
DeleteObject(MemDC);
with Loupe.Canvas^ do begin
Brush.Color:= clYellow;
Brush.BrushStyle:= bsClear;
Pen.PenWidth:=2;
Pen.Color:= clYellow;
Pen.PenStyle:= psSolid;
Ellipse(1, 1, Width- 2, Width- 2);
Pen.Color:= clOlive;
Pen.PenStyle:= psSolid;
Ellipse(2, 2, Width- 1, Width- 1);
end;
end; Недостатки: 1 если запустить при открытой папке, то свернув папку будет видно что в ней 2 не сделал пока увеличение и уменьшение хочу сделать его в сообытии кручения колеса мыши. Вот весь код: program Project1;
uses
windows,messages,
kol;
const
Diametre = 300;
var
form,Loupe:PControl;
hRegion,hRegP: HRgn;
Width:integer;
DeskDC : HDC;
HDeskBmp : HBitmap;
Img: PBitmap;
procedure MouseDown(Sender: PControl;var mouse:TMouseEventData);
begin
releasecapture();
form.Perform(wm_syscommand,$F012,0);
end;
procedure ScreenRelease;
begin
if HDeskBmp<> 0 then
begin
DeleteObject(HDeskBmp);
HDeskBmp:= 0;
end;
if DeskDC<> 0 then
begin
ReleaseDC(GetDesktopWindow, DeskDC);
DeskDC:= 0;
end;
end;
procedure ScreenCapture;
var
MemDC: HDC;
W, H: Integer;
begin
if (HDeskBmp<> 0)or (DeskDC<> 0) then
Exit;
W:= GetSystemMetrics(SM_CXSCREEN);
H:= GetSystemMetrics(SM_CYSCREEN);
DeskDC:= GetDC(GetDesktopWindow);
MemDC:= CreateCompatibleDC(DeskDC);
HDeskBmp:= CreateCompatibleBitmap(DeskDC, W, H);
SelectObject(MemDC, HDeskBmp);
BitBlt(MemDC, 0, 0, W, H, DeskDC, 0, 0, SRCCOPY);
DeleteDC(MemDC);
end;
procedure DrawFilter;
var
MemDC: HDC;
OldBmp: HBitMap;
OffsetX, OffsetY: Integer;
begin
if Loupe= nil then Exit;
// Копируем фоновое изображение
MemDC:= CreateCompatibleDC(DeskDC);
OldBmp:= SelectObject(MemDC, HDeskBmp);
OffsetX:= Width div 4;
OffsetY:= Width div 4;
StretchBlt(Img.Canvas.Handle, 0, 0, Width, Width, MemDC, form.Left + OffsetX, form.Top + OffsetY, Width div 2, Width div 2, SRCCOPY);
Loupe.Invalidate;//Заставляем перерисовываться чтобы избавится от мелькания
//но тогда теряется оконтовка лупы
SelectObject(MemDC, OldBmp);
DeleteObject(MemDC);
end;
procedure TimerTick(Dummy: Pointer; Sender: PObj);
begin
ScreenCapture;
DrawFilter;
end;
procedure FormPaint(Dummy: Pointer; Sender: PControl; DC: HDC);
begin
Img.Draw(Sender.Canvas.Handle,0,0);
end;
function AppletMessage(Dummy : Pointer;var Msg: tagMSG; var Rslt: Integer): Boolean;
begin
Result := False;
if (Msg.message = WM_KEYDOWN) and ( Msg.wParam = VK_ESCAPE) then form.close;
end;
procedure FormDestroy(Sender: TObject);
begin
ScreenRelease;
Loupe.Free;
end;
Begin
Applet := NewApplet('');
form:=NewForm(Applet,'Лупа').centeronparent.Tabulate;
Width := Diametre;
Img := NewBitmap(form.ClientWidth,form.ClientHeight);
hRegP:=CreateEllipticRgn(4, 4, Width-4, Width-4);
Loupe := NewPaintBox(form);
Loupe.Width:= Width;
Loupe.Height:=Width;
if hRegP<> 0 then SetWindowRgn(Loupe.GetWindowHandle , hRegP, TRUE);
hRegion:= CreateEllipticRgn(0, 0, Width+3, Width+3);
if hRegion<> 0 then SetWindowRgn(form.GetWindowHandle , hRegion, TRUE);
HDeskBmp:= 0;
DeskDc:= 0;
ScreenCapture;
DrawFilter;
with NewTimer(100)^ do
begin
OnTimer := TOnEvent(MakeMethod(nil, @TimerTick));
Enabled := True;
end;
Loupe.OnMouseDown:=TOnMouse(MakeMethod(nil,@MouseDown));
Loupe.OnPaint:=TOnPaint( MakeMethod( nil, @FormPaint ));
form.Style := WS_POPUP or WS_VISIBLE;
form.StayOnTop := True;
Applet.OnMessage:=TOnMessage(MakeMethod(nil,@AppletMessage));
SetProcessWorkingSetSize(GetCurrentProcess, Cardinal(-1), Cardinal(-1));
Run(Applet);
end.
-
I hope that I understand you correctly: > Brush.Color:= clYellow;
Applet.Color := clYellow; > 1.
function AppletMessage(Dummy: Pointer; var Msg: tagMSG; var Rslt: Integer): Boolean;
begin
Result := False;
if (Msg.message = WM_KEYDOWN) and (Msg.wParam = VK_ESCAPE) then
form.close;
if (Msg.message = WM_KEYDOWN) and (Msg.wParam = VK_F5) then
begin
Result := True;
form.Hide;
Sleep(100);
DeskDC := 0;
HDeskBmp := 0;
ScreenCapture;
DrawFilter;
form.Show;
end;
end;
> 2.
Use .OnMouseWheel or hook the mouse messages - SetWindowsHookEx(WH_MOUSE, @HookProc, HInstance,0) ;
-
Yes, you understood correctly! I found a small bug in my code.
Applet.Color := clYellow;
Width := Diametre;
form:=NewForm(Applet,'Лупа').setsize(324,324).centeronparent.Tabulate; Again, thank you:) About > 2. I'll try.)
-
About > 2. I'll try.)
var ...z:real;
...
procedure MMouseWheel(Dummy:pointer;Sender: PControl;
var Mouse: TMouseEventData);
begin
if Integer( Mouse.Shift ) > 0 then
i:=i+10
else
i:=i-10;
Mouse.StopHandling := TRUE;
case i of
-10:i:=0;
0:begin z:=1;
ScreenCapture;
DrawFilter;
form.Show;
end;
10:;
20:begin z:=1.5;
ScreenCapture;
DrawFilter;
form.Show;
end;
30:;
40:begin z:=3;
ScreenCapture;
DrawFilter;
form.Show;
end;
50:;
60: i:=50;
end;
end;
...
form.OnMouseWheel:=TOnMouse(MakeMethod(nil,@MMouseWheel));
Вот. Хочу к нему сделать визуальный эффект. Это "веер" из RoundRectRgn'ов который раскрывается при усилении увеличения лупы. Вопрос: Как сделать область в которой развернутся регионы невидимой чтоб видно было только веер (при развёртке)? [URL= http://www.radikal.ru][IMG]http://s011.radikal.ru/i315/1104/ae/49f0da395296.jpg[/IMG][/URL]
-
-
-
to Jon Thank you just is not quite like. But it gave some idea. True, the idea does not work either.
In such a case is cut area which in turn gives invisibility. In my case, the region should be and remain invisible so that the torsional regions which will become visible to the user.
//В таком случае происходит вырезание области что в свою очередь даёт невидимость. В моём случае область должна быть и оставаться невидимой чтоб при кручении регионы которые появятся стали видны пользователю.
-
-
фигасе ты упорный, 4 года лупу ваяешь грабли наверно в том, что ты хочешь как там, где слоистое окно, а делаешь при этом региональное?
-
По-моему лучше будет не через таймер, а рисовать по событию WM_MOUSEMOVE. Т.е. по WM_LBUTTONDOWN запоминаем первоначальные координаты окна, по WM_MOUSEMOVE вычисляем новые координаты, и отрисовываемся. Что бы не потерять окно, я ставлю SetCapture. А сверху основного окна можно таскать одновременно окно с AlphaBlend и текстурой линзы. Ну я бы так сделал )
-
> фигасе ты упорный, 4 года лупу ваяешь
Всё руки никак не доходили. Потом появилось время. Сел. Посмотрел. Поковырял.
> По-моему лучше будет не через таймер, а рисовать по событию > WM_MOUSEMOVE.
Если можно кинь небольшой пример так нагляднее и быстрее реализовать идею.
Спасибо за советы.
-
аяяяяй спалился, 2 форумных ника - бан тебе вечный по правилам :)
-
Не обратил внимания. Ник один. Сорри.
-
Уважаемый модератор Удалите сообщения с ником famrus. Спасибо.
-
Вот немного накидал для демонстрации производительности.
program Lupe;
uses Windows, KOL, messages;
var LupeForm: PControl; Diametre: Integer; StartPos: TPoint; Moving: Boolean; DC:hDC; BgScrBt: PBitmap;
procedure ScreenCopy(); var W,H: integer; begin W:= GetSystemMetrics(SM_CXSCREEN); H:= GetSystemMetrics(SM_CYSCREEN); BgScrBt := NewBitmap( W, H ); BitBlt(BgScrBt.canvas.Handle, 0,0, W, H, GetDC(0), 0, 0, SRCCOPY); end;
function LupeWndProc(Sender: PControl; var Msg: TMsg; var Rslt: Integer ): Boolean; var pt: TPoint; R: TRect; begin Result:=False; case Msg.message of WM_CREATE: begin ScreenCopy; DC:=GetDC(LupeForm.Handle); end; WM_LBUTTONDOWN: begin GetCursorPos(pt); Moving:=True; Sender.Caption:='Moving'; StartPos.X:=pt.x-Sender.Left; StartPos.Y:=pt.Y-Sender.Top; SetCapture(Sender.Handle); end; WM_LBUTTONUP: begin Moving:=False; Sender.Caption:='Lupe'; ReleaseCapture(); end; WM_MOUSEMOVE: if Moving then begin GetCursorPos(pt); SetWindowPos(Sender.Handle,0, pt.X-StartPos.X, pt.Y-StartPos.Y, 0, 0, SWP_NOSIZE {or SWP_NOREDRAW}); end; WM_ERASEBKGND: begin StretchBlt(DC, 0, 0, Sender.Width, Sender.Height, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, Sender.Width div 2, Sender.Width div 2, SRCCOPY); Result:=True; end; WM_PAINT: begin Result:=True; end; end; end;
begin Diametre:=300; LupeForm:=NewForm(Applet, 'Lupe').Size(Diametre, Diametre); LupeForm.AttachProc(@LupeWndProc); Run(LupeForm); end.
Вот так получается значительно быстрее отрисовывать. Но это на простой форме.
-
Если в конце перед Run(LupeForm) добавить
hReg1:= CreateEllipticRgn(0, 0, Diametre, Diametre); LupeForm.HasBorder := False; SetWindowRgn(LupeForm.GetWindowHandle , hReg1, TRUE); То окно становиться круглым, и ничего не тормозит.
-
program Lupe;
uses Windows, KOL, messages;
var LupeForm: PControl; Diametre: Integer; StartPos: TPoint; Moving: Boolean; DC:hDC; BgScrBt: PBitmap; hReg1: HRgn;
procedure ScreenCopy(); var W,H: integer; begin W:= GetSystemMetrics(SM_CXSCREEN); H:= GetSystemMetrics(SM_CYSCREEN); BgScrBt := NewBitmap( W, H ); BitBlt(BgScrBt.canvas.Handle, 0,0, W, H, GetDC(0), 0, 0, SRCCOPY); end;
function LupeWndProc(Sender: PControl; var Msg: TMsg; var Rslt: Integer ): Boolean; var pt: TPoint; begin Result:=False; case Msg.message of WM_CREATE: begin ScreenCopy; DC:=GetDC(LupeForm.Handle); end; WM_LBUTTONDOWN: begin GetCursorPos(pt); Moving:=True; Sender.Caption:='Moving'; StartPos.X:=pt.x-Sender.Left; StartPos.Y:=pt.Y-Sender.Top; SetCapture(Sender.Handle); end; WM_LBUTTONUP: begin Moving:=False; Sender.Caption:='Lupe'; ReleaseCapture(); end; WM_MOUSEMOVE: if Moving then begin GetCursorPos(pt); SetWindowPos(Sender.Handle,0, pt.X-StartPos.X, pt.Y-StartPos.Y, 0, 0, SWP_NOSIZE {or SWP_NOREDRAW}); end; WM_ERASEBKGND: begin StretchBlt(DC, 0, 0, Sender.Width, Sender.Height, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, Sender.Width div 2, Sender.Width div 2, SRCCOPY); Result:=True; end; WM_PAINT: begin Result:=True; end; end; end;
begin Diametre:=300; LupeForm:=NewForm(Applet, 'Lupe').Size(Diametre, Diametre); LupeForm.AttachProc(@LupeWndProc); hReg1:= CreateEllipticRgn(0, 0, Diametre, Diametre); LupeForm.HasBorder := False; // Скрываем бордюр SetWindowRgn(LupeForm.GetWindowHandle , hReg1, TRUE); Run(LupeForm); end.
-
Браузер глючит, не обновлял страницу. Удалить бы 2 поста перед предыдущим
-
Neat, but it doesn't support multimonitors. try this instead ;)
procedure ScreenCopy();
var W,H: integer;
begin
W:= GetSystemMetrics(SM_CXVIRTUALSCREEN);
H:= GetSystemMetrics(SM_CYVIRTUALSCREEN);
BgScrBt := NewBitmap( W, H );
BitBlt(BgScrBt.canvas.Handle, 0,0, W, H, GetDC(0), 0, 0, SRCCOPY);
end;
-
You may also want to do something like this:
WM_ERASEBKGND:
begin
If (sender.left < 0) or (sender.Top < 0) then
FillRect(DC,MakeRect(0,0,Sender.Width+sender.left, Sender.Height+Sender.Top), GetStockObject(BLACK_BRUSH));
StretchBlt(DC, 0, 0, Sender.Width, Sender.Height, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, Sender.Width div 2, Sender.Width div 2, SRCCOPY);
Result:=True;
I will look into a neat way to solve minimizing/maximizing applications and have proper restore. (combinergn stuff)
-
На счет черных границ, получается двойной вывод, и проглядываются тормоза. На самом деле это не обязательно, ведь я не посчитал offset и увеличиваемый фрагмент у меня берется от угла формы, а надо высчитать, что бы он был по центру лупы.
-
А вот так можно реализовать прозрачность софтварно, без использования средств операционной системы.
program Lupe;
uses Windows, KOL, messages;
type TBig = array[0..0] of Integer;
var LupeForm: PControl; Diametre: Integer; StartPos: TPoint; Moving: Boolean; DC, BmpDC:hDC; BgScrBt, BgBmp: PBitmap; hReg1: HRgn; BgBmpPointer: Pointer;
function GetDIBPointer(BitMap: PBitmap):Pointer; var BInfo: TBitmapInfo; begin BInfo.bmiHeader.biSize := sizeof(BitmapInfo); BInfo.bmiHeader.biWidth := BitMap.Width; BInfo.bmiHeader.biHeight := -BitMap.Height; BInfo.bmiHeader.biPlanes := 1; BInfo.bmiHeader.biBitCount := 32; BInfo.bmiHeader.biCompression := BI_RGB; BitMap.Handle:=CreateDIBSection(BitMap.Handle, BInfo, DIB_RGB_COLORS, Result, 0, 0); end;
procedure ScreenCopy(); var W,H: integer; begin W:= GetSystemMetrics(SM_CXSCREEN); H:= GetSystemMetrics(SM_CYSCREEN); BgScrBt := NewBitmap( W, H ); BitBlt(BgScrBt.canvas.Handle, 0,0, W, H, GetDC(0), 0, 0, SRCCOPY); BgBmp:= NewBitmap(Diametre, Diametre); BgBmpPointer := GetDIBPointer(BgBmp); end;
function LupeWndProc(Sender: PControl; var Msg: TMsg; var Rslt: Integer ): Boolean; var pt: TPoint; R: TRect; x,y: integer; begin Result:=False; case Msg.message of WM_CREATE: begin ScreenCopy; DC:=GetDC(LupeForm.Handle); end; WM_LBUTTONDOWN: begin GetCursorPos(pt); Moving:=True; Sender.Caption:='Moving'; StartPos.X:=pt.x-Sender.Left; StartPos.Y:=pt.Y-Sender.Top; SetCapture(Sender.Handle); end; WM_LBUTTONUP: begin Moving:=False; Sender.Caption:='Lupe'; ReleaseCapture(); end; WM_MOUSEMOVE: if Moving then begin GetCursorPos(pt); SetWindowPos(Sender.Handle,0, pt.X-StartPos.X, pt.Y-StartPos.Y, 0, 0, SWP_NOSIZE {or SWP_NOREDRAW}); end; WM_ERASEBKGND: begin //StretchBlt(BgBmp.Canvas.Handle, 0, 0, Diametre, Diametre, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, Diametre div 2, Diametre div 2, SRCCOPY); BitBlt(BgBmp.Canvas.Handle, 0,0, Sender.Width, Sender.Height, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, SRCCOPY); for y:=0 to Diametre -1 do for x:=0 to Diametre -1 do begin TBig(BgBmpPointer^)[x + y * Diametre] := RGB(GetRValue(TBig(BgBmpPointer^)[x + y * Diametre]) div 2,GetGValue(TBig(BgBmpPointer^)[x + y * Diametre]) div 2,GetBValue(TBig(BgBmpPointer^)[x + y * Diametre]) div 2); end; BitBlt(DC, 0,0, Sender.Width, Sender.Height, BgBmp.Canvas.Handle, 0, 0, SRCCOPY); Result:=True; end; WM_PAINT: begin Result:=True; end; end; end;
begin Diametre:=300; LupeForm:=NewForm(Applet, 'Lupe').Size(Diametre, Diametre); LupeForm.AttachProc(@LupeWndProc); hReg1:= CreateEllipticRgn(0, 0, Diametre, Diametre); LupeForm.HasBorder := False; // Скрываем бордюр SetWindowRgn(LupeForm.GetWindowHandle , hReg1, TRUE);
Run(LupeForm); end.
-
Размытие
program Lupe;
uses Windows, KOL, messages;
type TBig = array[0..0] of Integer;
var LupeForm: PControl; Diametre: Integer; StartPos: TPoint; Moving: Boolean; DC, BmpDC:hDC; BgScrBt, BgBmp: PBitmap; hReg1: HRgn; BgBmpPointer: Pointer;
function GetDIBPointer(BitMap: PBitmap):Pointer; var BInfo: TBitmapInfo; begin BInfo.bmiHeader.biSize := sizeof(BitmapInfo); BInfo.bmiHeader.biWidth := BitMap.Width; BInfo.bmiHeader.biHeight := -BitMap.Height; BInfo.bmiHeader.biPlanes := 1; BInfo.bmiHeader.biBitCount := 32; BInfo.bmiHeader.biCompression := BI_RGB; BitMap.Handle:=CreateDIBSection(BitMap.Handle, BInfo, DIB_RGB_COLORS, Result, 0, 0); end;
procedure ScreenCopy(); var W,H: integer; begin W:= GetSystemMetrics(SM_CXSCREEN); H:= GetSystemMetrics(SM_CYSCREEN); BgScrBt := NewBitmap( W, H ); BitBlt(BgScrBt.canvas.Handle, 0,0, W, H, GetDC(0), 0, 0, SRCCOPY); BgBmp:= NewBitmap(Diametre, Diametre); BgBmpPointer := GetDIBPointer(BgBmp); end;
function LupeWndProc(Sender: PControl; var Msg: TMsg; var Rslt: Integer ): Boolean; var pt: TPoint;
x,y,R,G,B: integer; begin Result:=False; case Msg.message of WM_CREATE: begin ScreenCopy; DC:=GetDC(LupeForm.Handle); end; WM_LBUTTONDOWN: begin GetCursorPos(pt); Moving:=True; Sender.Caption:='Moving'; StartPos.X:=pt.x-Sender.Left; StartPos.Y:=pt.Y-Sender.Top; SetCapture(Sender.Handle); end; WM_LBUTTONUP: begin Moving:=False; Sender.Caption:='Lupe'; ReleaseCapture(); end; WM_MOUSEMOVE: if Moving then begin GetCursorPos(pt); SetWindowPos(Sender.Handle,0, pt.X-StartPos.X, pt.Y-StartPos.Y, 0, 0, SWP_NOSIZE {or SWP_NOREDRAW}); end; WM_ERASEBKGND: begin //StretchBlt(BgBmp.Canvas.Handle, 0, 0, Diametre, Diametre, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, Diametre div 2, Diametre div 2, SRCCOPY); BitBlt(BgBmp.Canvas.Handle, 0,0, Sender.Width, Sender.Height, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, SRCCOPY); for y:=0 to Diametre -2 do for x:=0 to Diametre -2 do begin R:=(GetRValue(TBig(BgBmpPointer^)[x + y * Diametre]) + GetRValue(TBig(BgBmpPointer^)[x+1 + (y+1) * Diametre]) + GetRValue(TBig(BgBmpPointer^)[x+ (y+1) * Diametre]) + GetRValue(TBig(BgBmpPointer^)[x+1 +y * Diametre])) div 4; G:=(GetGValue(TBig(BgBmpPointer^)[x + y * Diametre]) + GetGValue(TBig(BgBmpPointer^)[x+1 + (y+1) * Diametre]) + GetGValue(TBig(BgBmpPointer^)[x+ (y+1) * Diametre]) + GetGValue(TBig(BgBmpPointer^)[x+1 +y * Diametre])) div 4; B:=(GetBValue(TBig(BgBmpPointer^)[x + y * Diametre]) + GetBValue(TBig(BgBmpPointer^)[x+1 + (y+1) * Diametre]) + GetBValue(TBig(BgBmpPointer^)[x+ (y+1) * Diametre]) + GetBValue(TBig(BgBmpPointer^)[x+1 +y * Diametre])) div 4; if R>254 then R:=254; if G>254 then G:=254; if B>254 then B:=254; if R<0 then R:=0; if G<0 then B:=0; if B<0 then B:=0; TBig(BgBmpPointer^)[x + y * Diametre] := RGB(R,G,B); end; BitBlt(DC, 0,0, Sender.Width, Sender.Height, BgBmp.Canvas.Handle, 0, 0, SRCCOPY); Result:=True; end; WM_PAINT: begin Result:=True; end; end; end;
begin Diametre:=300; LupeForm:=NewForm(Applet, 'Lupe').Size(Diametre, Diametre); LupeForm.AttachProc(@LupeWndProc); hReg1:= CreateEllipticRgn(0, 0, Diametre, Diametre); LupeForm.HasBorder := False; // Скрываем бордюр SetWindowRgn(LupeForm.GetWindowHandle , hReg1, TRUE);
Run(LupeForm); end.
-
Глубокое размытие
program Lupe;
uses Windows, KOL, messages;
type TBig = array[0..0] of Integer;
var LupeForm: PControl; Diametre: Integer; StartPos: TPoint; Moving: Boolean; DC, BmpDC:hDC; BgScrBt, BgBmp: PBitmap; hReg1: HRgn; BgBmpPointer: Pointer;
function GetDIBPointer(BitMap: PBitmap):Pointer; var BInfo: TBitmapInfo; begin BInfo.bmiHeader.biSize := sizeof(BitmapInfo); BInfo.bmiHeader.biWidth := BitMap.Width; BInfo.bmiHeader.biHeight := -BitMap.Height; BInfo.bmiHeader.biPlanes := 1; BInfo.bmiHeader.biBitCount := 32; BInfo.bmiHeader.biCompression := BI_RGB; BitMap.Handle:=CreateDIBSection(BitMap.Handle, BInfo, DIB_RGB_COLORS, Result, 0, 0); end;
procedure ScreenCopy(); var W,H: integer; begin W:= GetSystemMetrics(SM_CXSCREEN); H:= GetSystemMetrics(SM_CYSCREEN); BgScrBt := NewBitmap( W, H ); BitBlt(BgScrBt.canvas.Handle, 0,0, W, H, GetDC(0), 0, 0, SRCCOPY); BgBmp:= NewBitmap(Diametre, Diametre); BgBmpPointer := GetDIBPointer(BgBmp); end;
function LupeWndProc(Sender: PControl; var Msg: TMsg; var Rslt: Integer ): Boolean; var pt: TPoint;
i, x,y,R,G,B: integer; begin Result:=False; case Msg.message of WM_CREATE: begin ScreenCopy; DC:=GetDC(LupeForm.Handle); end; WM_LBUTTONDOWN: begin GetCursorPos(pt); Moving:=True; Sender.Caption:='Moving'; StartPos.X:=pt.x-Sender.Left; StartPos.Y:=pt.Y-Sender.Top; SetCapture(Sender.Handle); end; WM_LBUTTONUP: begin Moving:=False; Sender.Caption:='Lupe'; ReleaseCapture(); end; WM_MOUSEMOVE: if Moving then begin GetCursorPos(pt); SetWindowPos(Sender.Handle,0, pt.X-StartPos.X, pt.Y-StartPos.Y, 0, 0, SWP_NOSIZE {or SWP_NOREDRAW}); end; WM_ERASEBKGND: begin //StretchBlt(BgBmp.Canvas.Handle, 0, 0, Diametre, Diametre, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, Diametre div 2, Diametre div 2, SRCCOPY); BitBlt(BgBmp.Canvas.Handle, 0,0, Sender.Width, Sender.Height, BgScrBt.Canvas.Handle, Sender.Left, Sender.Top, SRCCOPY); for i:=1 to 5 do //Глубина размытия. Повторяем усреднение соседних пикселей несколько раз. Чем больше тем глубже. Но тормознее for y:=0 to Diametre -2 do for x:=0 to Diametre -2 do begin R:=(GetRValue(TBig(BgBmpPointer^)[x + y * Diametre]) + GetRValue(TBig(BgBmpPointer^)[x+1 + (y+1) * Diametre]) + GetRValue(TBig(BgBmpPointer^)[x+ (y+1) * Diametre]) + GetRValue(TBig(BgBmpPointer^)[x+1 +y * Diametre])) div 4; G:=(GetGValue(TBig(BgBmpPointer^)[x + y * Diametre]) + GetGValue(TBig(BgBmpPointer^)[x+1 + (y+1) * Diametre]) + GetGValue(TBig(BgBmpPointer^)[x+ (y+1) * Diametre]) + GetGValue(TBig(BgBmpPointer^)[x+1 +y * Diametre])) div 4; B:=(GetBValue(TBig(BgBmpPointer^)[x + y * Diametre]) + GetBValue(TBig(BgBmpPointer^)[x+1 + (y+1) * Diametre]) + GetBValue(TBig(BgBmpPointer^)[x+ (y+1) * Diametre]) + GetBValue(TBig(BgBmpPointer^)[x+1 +y * Diametre])) div 4; if R>254 then R:=254; if G>254 then G:=254; if B>254 then B:=254; if R<0 then R:=0; if G<0 then B:=0; if B<0 then B:=0; TBig(BgBmpPointer^)[x + y * Diametre] := RGB(R,G,B); end;
BitBlt(DC, 0,0, Sender.Width, Sender.Height, BgBmp.Canvas.Handle, 0, 0, SRCCOPY); Result:=True; end; WM_PAINT: begin Result:=True; end; end; end;
begin Diametre:=300; LupeForm:=NewForm(Applet, 'Lupe').Size(Diametre, Diametre); LupeForm.AttachProc(@LupeWndProc); hReg1:= CreateEllipticRgn(0, 0, Diametre, Diametre); LupeForm.HasBorder := False; // Скрываем бордюр SetWindowRgn(LupeForm.GetWindowHandle , hReg1, TRUE);
Run(LupeForm); end.
-
Удалено модератором
|