যখন কোনও সীমানা ছিল ঠিক তেমনভাবে ফর্মটিতে মাউসটি ক্লিক করা হবে তখন কোনও সীমানা নেই এমন কোনও ফর্ম তৈরি করার কোনও উপায় রয়েছে (ফর্মবোর্ডার স্টাইলটি "কিছুই নয়" তে সেট করা আছে)?
যখন কোনও সীমানা ছিল ঠিক তেমনভাবে ফর্মটিতে মাউসটি ক্লিক করা হবে তখন কোনও সীমানা নেই এমন কোনও ফর্ম তৈরি করার কোনও উপায় রয়েছে (ফর্মবোর্ডার স্টাইলটি "কিছুই নয়" তে সেট করা আছে)?
উত্তর:
কোডপ্রজেক্টের এই নিবন্ধটিতে একটি কৌশল বিশদ রয়েছে। মূলত এগুলি ফোটায়:
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HT_CAPTION = 0x2;
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
[System.Runtime.InteropServices.DllImport("user32.dll")]
public static extern bool ReleaseCapture();
private void Form1_MouseDown(object sender, System.Windows.Forms.MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
}
}
এই মূলত করে ঠিক একটি উইন্ডো শিরোনাম বার দখল দৃশ্য জানালা ম্যানেজারের বিন্দু থেকে হিসাবে একই।
Form1_MouseDown
সত্যিকারের MouseDown
ইভেন্টে নির্ধারিত না হয়ে কাজ করবে না Form1
।
this.MouseDown += ...
আপনাকে Main()
যেকোন জিনিসগুলির প্রয়োজনের তুলনায় আর কোনও সমস্যা না করা যাক। আমি কোডের এতগুলি স্নিপেট নিয়ে এসেছি যা আপনাকে ফর্মটি (বা অন্য কোনও নিয়ন্ত্রণ) চারপাশে টেনে আনতে দেয়। এবং তাদের অনেকের নিজস্ব ত্রুটি / পার্শ্ব প্রতিক্রিয়া রয়েছে। বিশেষত সেইগুলি যেখানে তারা উইন্ডোজকে এমন ভাবনায় প্ররোচিত করে যে কোনও ফর্মের একটি নিয়ন্ত্রণই আসল রূপ।
বলা হচ্ছে, এখানে আমার স্নিপেট। আমি সব সময় এটি ব্যবহার। আমি এটিও নোট করতে চাই যে আপনার এটি ব্যবহার করা উচিত নয় nঅনুপাতিক (); অন্যরা যেমন করতে পছন্দ করে কারণ এটি ফর্মটিকে কিছু ক্ষেত্রে ঝাঁকুনির কারণ করে। এবং কিছু ক্ষেত্রে এটি করে। রিফ্রেশ। এটি ব্যবহার করে p আপডেট, আমার কোনও ঝাঁকুনির সমস্যা নেই:
private bool mouseDown;
private Point lastLocation;
private void Form1_MouseDown(object sender, MouseEventArgs e)
{
mouseDown = true;
lastLocation = e.Location;
}
private void Form1_MouseMove(object sender, MouseEventArgs e)
{
if(mouseDown)
{
this.Location = new Point(
(this.Location.X - lastLocation.X) + e.X, (this.Location.Y - lastLocation.Y) + e.Y);
this.Update();
}
}
private void Form1_MouseUp(object sender, MouseEventArgs e)
{
mouseDown = false;
}
একই জিনিসটি করার আরও একটি সহজ উপায়।
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
// set this.FormBorderStyle to None here if needed
// if set to none, make sure you have a way to close the form!
}
protected override void WndProc(ref Message m)
{
base.WndProc(ref m);
if (m.Msg == WM_NCHITTEST)
m.Result = (IntPtr)(HT_CAPTION);
}
private const int WM_NCHITTEST = 0x84;
private const int HT_CLIENT = 0x1;
private const int HT_CAPTION = 0x2;
}
মাউসডাউন, মাউসমোভ এবং মাউসআপ ব্যবহার করুন। আপনি তার জন্য একটি পরিবর্তনশীল পতাকা সেট করতে পারেন। আমার একটি নমুনা রয়েছে তবে আমি মনে করি আপনাকে সংশোধন করা দরকার।
আমি একটি প্যানেলে মাউস অ্যাকশন কোডিং করছি। আপনি একবার প্যানেলে ক্লিক করলে, আপনার ফর্মটি এটির সাথে সরে যাবে।
//Global variables;
private bool _dragging = false;
private Point _offset;
private Point _start_point=new Point(0,0);
private void panel1_MouseDown(object sender, MouseEventArgs e)
{
_dragging = true; // _dragging is your variable flag
_start_point = new Point(e.X, e.Y);
}
private void panel1_MouseUp(object sender, MouseEventArgs e)
{
_dragging = false;
}
private void panel1_MouseMove(object sender, MouseEventArgs e)
{
if(_dragging)
{
Point p = PointToScreen(e.Location);
Location = new Point(p.X - this._start_point.X,p.Y - this._start_point.Y);
}
}
শুধুমাত্র ডাব্লুপিএফ
হস্তান্তর করার জন্য সঠিক কোড নেই, তবে একটি সাম্প্রতিক প্রকল্পে আমি মনে করি যে আমি মাউসডাউন ইভেন্টটি ব্যবহার করেছি এবং কেবল এটি রেখেছি:
frmBorderless.DragMove();
এটি পরীক্ষিত এবং বোঝা সহজ।
protected override void WndProc(ref Message m)
{
switch (m.Msg)
{
case 0x84:
base.WndProc(ref m);
if((int)m.Result == 0x1)
m.Result = (IntPtr)0x2;
return;
}
base.WndProc(ref m);
}
WM_NCHITTEST
ছদ্মবেশে আছে।
এটিকে কেবল ম্যাজিকভাবেই ঘটানোর জন্য আপনি কোনও ফ্লিপ করতে পারবেন এমন কোনও সম্পত্তি নেই। ফর্মটির জন্য ইভেন্টগুলি দেখুন এবং এটি সেট করে this.Top
এবং প্রয়োগ করে এটি প্রয়োগ করা মোটামুটি তুচ্ছ হয়ে ওঠেthis.Left
। বিশেষত আপনি দেখতে চান MouseDown
, MouseUp
এবং MouseMove
।
public Point mouseLocation;
private void frmInstallDevice_MouseDown(object sender, MouseEventArgs e)
{
mouseLocation = new Point(-e.X, -e.Y);
}
private void frmInstallDevice_MouseMove(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
Point mousePos = Control.MousePosition;
mousePos.Offset(mouseLocation.X, mouseLocation.Y);
Location = mousePos;
}
}
এটি আপনার সমস্যার সমাধান করতে পারে ....
উপরের লিঙ্কটির এই বিট কোডটি আমার ক্ষেত্রে কৌশলটি করেছে:
protected override void OnMouseDown(MouseEventArgs e)
{
base.OnMouseDown(e);
if (e.Button == MouseButtons.Left)
{
this.Capture = false;
Message msg = Message.Create(this.Handle, 0XA1, new IntPtr(2), IntPtr.Zero);
this.WndProc(ref msg);
}
}
আমি খুঁজে পাওয়ার সেরা উপায় (অবশ্যই সংশোধিত)
// This adds the event handler for the control
private void AddDrag(Control Control) { Control.MouseDown += new System.Windows.Forms.MouseEventHandler(this.DragForm_MouseDown); }
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HT_CAPTION = 0x2;
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern bool ReleaseCapture();
private void DragForm_MouseDown(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
// Checks if Y = 0, if so maximize the form
if (this.Location.Y == 0) { this.WindowState = FormWindowState.Maximized; }
}
}
কোনও নিয়ন্ত্রণে টেনে আনার জন্য এটিকে প্রাথমিকভাবে কম্পোনেন্ট () এর পরে সন্নিবেশ করান
AddDrag(NameOfControl);
এটা আমার জন্য কাজ করেছে।
private void Form1_MouseDown(object sender, MouseEventArgs e)
{
_mouseLoc = e.Location;
}
private void Form1_MouseMove(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
int dx = e.Location.X - _mouseLoc.X;
int dy = e.Location.Y - _mouseLoc.Y;
this.Location = new Point(this.Location.X + dx, this.Location.Y + dy);
}
}
.NET ফ্রেমওয়ার্ক 4 এর জন্য,
আপনি ব্যবহার করতে পারেন this.DragMove()
জন্য MouseDown
উপাদান (এই উদাহরণে mainLayout) আপনি টেনে আনার ব্যবহার করছেন ইভেন্ট।
private void mainLayout_MouseDown(object sender, MouseButtonEventArgs e)
{
this.DragMove();
}
সবচেয়ে সহজ উপায়:
প্রথমে লেবেল 1 নামের একটি লেবেল তৈরি করুন। লেবেল 1 এর ইভেন্টগুলি> মাউস ইভেন্টস> লেবেল 1_মাউস এ সরান এবং এগুলি লিখুন:
if (e.Button == MouseButtons.Left){
Left += e.X;
Top += e.Y;`
}
আমি সীমান্তহীন উইন্ডোজ ফর্মটিকে স্থাবর করার চেষ্টা করছিলাম যাতে একটি ডাব্লুপিএফ এলিমেন্ট হোস্ট নিয়ন্ত্রণ এবং একটি ডাব্লুপিএফ ব্যবহারকারী নিয়ন্ত্রণ রয়েছে।
আমি আমার ডাব্লুপিএফ ব্যবহারকারী নিয়ন্ত্রণে স্ট্যাকপ্যানেল নামে একটি স্ট্যাক প্যানেলটি দিয়ে শেষ করেছি যা সরানোর জন্য ক্লিক করার চেষ্টা করার পক্ষে যুক্তিযুক্ত জিনিস বলে মনে হয়েছিল। জুনম্যাটসের কোডটি চেষ্টা করার সময় আমি মাউসটি আস্তে আস্তে কাজ করেছি, তবে আমি যদি মাউসটি আরও দ্রুত সরিয়ে ফেলতাম তবে মাউস ফর্মটি সরিয়ে ফেলবে এবং ফর্মটি কোথাও মাঝের সরানোতে আটকে থাকবে।
ক্যাপচারমাউস এবং রিলিজক্যাপচারমাউসটি ব্যবহার করে আমার অবস্থার জন্য তার উত্তরে এটির উন্নতি হয়েছে এবং এখনই আমি তাড়াতাড়ি সরিয়ে নিলেও মাউস ফর্মটি সরিয়ে নিয়ে যায় না।
private void StackPanel_MouseDown(object sender, MouseButtonEventArgs e)
{
_start_point = e.GetPosition(this);
StackPanel.CaptureMouse();
}
private void StackPanel_MouseUp(object sender, MouseButtonEventArgs e)
{
StackPanel.ReleaseMouseCapture();
}
private void StackPanel_MouseMove(object sender, MouseEventArgs e)
{
if (StackPanel.IsMouseCaptured)
{
var p = _form.GetMousePositionWindowsForms();
_form.Location = new System.Drawing.Point((int)(p.X - this._start_point.X), (int)(p.Y - this._start_point.Y));
}
}
//Global variables;
private Point _start_point = new Point(0, 0);
যেহেতু কিছু উত্তর শিশু নিয়ন্ত্রণ নিয়ন্ত্রণের জন্য টেনে নেওয়ার অনুমতি দেয় না, তাই আমি একটি সামান্য সহায়ক শ্রেণি তৈরি করেছি। এটি শীর্ষ স্তরের ফর্মটি পাস করা উচিত। চাইলে আরও জেনেরিক করা যায়।
class MouseDragger
{
private readonly Form _form;
private Point _mouseDown;
protected void OnMouseDown(object sender, MouseEventArgs e)
{
_mouseDown = e.Location;
}
protected void OnMouseMove(object sender, MouseEventArgs e)
{
if (e.Button == MouseButtons.Left)
{
int dx = e.Location.X - _mouseDown.X;
int dy = e.Location.Y - _mouseDown.Y;
_form.Location = new Point(_form.Location.X + dx, _form.Location.Y + dy);
}
}
public MouseDragger(Form form)
{
_form = form;
MakeDraggable(_form);
}
private void MakeDraggable(Control control)
{
var type = control.GetType();
if (typeof(Button).IsAssignableFrom(type))
{
return;
}
control.MouseDown += OnMouseDown;
control.MouseMove += OnMouseMove;
foreach (Control child in control.Controls)
{
MakeDraggable(child);
}
}
}
এছাড়াও আপনার যদি ডাবলক্লিক করতে এবং আপনার ফর্মটি আরও বড় / ছোট করার প্রয়োজন হয় তবে আপনি প্রথম উত্তরটি ব্যবহার করতে পারেন, একটি গ্লোবাল ইনট ভেরিয়েবল তৈরি করতে পারেন, আপনি যখন টেনে আনার জন্য ব্যবহার করেন সেই উপাদানটিতে প্রতিবার ক্লিক করুন 1 তাহলে variable == 2
আপনার ফর্মটি আরও বড় / আরও ছোট করুন। এছাড়াও আপনার তৈরি করতে প্রতি আধ সেকেন্ড বা এক সেকেন্ডের জন্য একটি টাইমার ব্যবহার করুন variable = 0
;
সংযোজন ক MouseLeftButtonDown
উইণ্ডো আমার জন্য কাজ করতে ইভেন্ট হ্যান্ডলার।
স্বয়ংক্রিয়ভাবে উত্পন্ন হওয়ার ইভেন্ট ইভেন্টে, নীচের কোডটি যুক্ত করুন:
base.OnMouseLeftButtonDown(e);
this.DragMove();
আমি jay_t55 থেকে আরও একটি পদ্ধতির সাথে সমাধানটি প্রসারিত করছি ToolStrip1_MouseLeave
যা মাউসটি দ্রুত চলে যাওয়ার এবং অঞ্চলটি ছেড়ে যাওয়ার ঘটনা পরিচালনা করে।
private bool mouseDown;
private Point lastLocation;
private void ToolStrip1_MouseDown(object sender, MouseEventArgs e) {
mouseDown = true;
lastLocation = e.Location;
}
private void ToolStrip1_MouseMove(object sender, MouseEventArgs e) {
if (mouseDown) {
this.Location = new Point(
(this.Location.X - lastLocation.X) + e.X, (this.Location.Y - lastLocation.Y) + e.Y);
this.Update();
}
}
private void ToolStrip1_MouseUp(object sender, MouseEventArgs e) {
mouseDown = false;
}
private void ToolStrip1_MouseLeave(object sender, EventArgs e) {
mouseDown = false;
}
আমি নিম্নলিখিত এবং প্রেস্টো চেঞ্জ চেষ্টা করেছিলাম, আমার স্বচ্ছ উইন্ডোটি আর স্থির ছিল না তবে সরানো যেতে পারে !! (উপরের সমস্ত জটিল সমাধানগুলি ফেলে দিন ...)
private void Window_MouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
base.OnMouseLeftButtonDown(e);
// Begin dragging the window
this.DragMove();
}
আবেদনপত্র 1(): new Moveable(control1, control2, control3);
ক্লাস:
using System;
using System.Windows.Forms;
class Moveable
{
public const int WM_NCLBUTTONDOWN = 0xA1;
public const int HT_CAPTION = 0x2;
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern int SendMessage(IntPtr hWnd, int Msg, int wParam, int lParam);
[System.Runtime.InteropServices.DllImportAttribute("user32.dll")]
public static extern bool ReleaseCapture();
public Moveable(params Control[] controls)
{
foreach (var ctrl in controls)
{
ctrl.MouseDown += (s, e) =>
{
if (e.Button == MouseButtons.Left)
{
ReleaseCapture();
SendMessage(ctrl.FindForm().Handle, WM_NCLBUTTONDOWN, HT_CAPTION, 0);
// Checks if Y = 0, if so maximize the form
if (ctrl.FindForm().Location.Y == 0) { ctrl.FindForm().WindowState = FormWindowState.Maximized; }
}
};
}
}
}
[DllImport("user32.DLL", EntryPoint = "ReleaseCapture")]
private extern static void ReleaseCapture();
[DllImport("user32.DLL", EntryPoint = "SendMessage")]
private extern static void SendMessage(System.IntPtr hWnd, int Msg, int wParam, int lParam);
private void panelTitleBar_MouseDown(object sender, MouseEventArgs e)
{
ReleaseCapture();
SendMessage(this.Handle, 0x112, 0xf012, 0);
}