Solved

MDI: Change CTRL-Tab behaviour

Posted on 2008-10-21
6
1,577 Views
Last Modified: 2013-12-17
Hi experts

How can I change the ctrl-tab behaviour of a MDI-Container.
Normally it just "loops" through all the mdi-children, but need to exclude some of them.
Where and how can I change this behaviour?

I'm using .NET 2.0, VS2008, c# and WinForms
0
Comment
Question by:Arikael
[X]
Welcome to Experts Exchange

Add your voice to the tech community where 5M+ people just like you are talking about what matters.

  • Help others & share knowledge
  • Earn cash & points
  • Learn & ask questions
  • 3
  • 3
6 Comments
 
LVL 86

Expert Comment

by:Mike Tomlinson
ID: 22768537
Hi Arikael,

In your MdiParent Form, the gray area where the MdiChild forms appear is known as the MdiClient window.  Whenever you press Ctrl-Tab, this window will get a WM_MDINEXT message which causes it to select the next MdiChild.

See WM_MDINEXT:
http://msdn.microsoft.com/en-us/library/ms644918(VS.85).aspx

In my example below, whenever an MdiChild of Type Form2 is selected as a result of a Ctrl-Tab, I send another WM_MDINEXT message causing the next MdiChild to be selected.  This technically doesn't SKIP Form2 since it is actually selected for a split second before the next MdiChild gets selected.

Sorry that it is in VB.Net...I'm still coming to grips with coding in C#.  =\

Hopefully you can make sense of what is going on...
Public Class Form4
 
    Private client As MyMdiClient
 
    Private Sub Form4_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles MyBase.Load
        For Each ctl As Control In Me.Controls
            If TypeOf ctl Is MdiClient Then
                client = New MyMdiClient(ctl, Me)
                Exit For
            End If
        Next
 
        Dim f1 As New Form1
        f1.MdiParent = Me
        f1.Show()
 
        Dim f2 As New Form2
        f2.MdiParent = Me
        f2.Show()
 
        Dim f3 As New Form3
        f3.MdiParent = Me
        f3.Show()
    End Sub
 
    Private Class MyMdiClient
        Inherits NativeWindow
 
        Private MyMdiParent As Form
        Private Const WM_MDINEXT As Integer = &H224
 
        Private Declare Function SendMessage Lib "user32" Alias "SendMessageW" _
            (ByVal handle As IntPtr, ByVal wMsg As Integer, _
            ByVal wParam As Integer, ByVal lParam As Integer) As Integer
 
        Public Sub New(ByVal client As MdiClient, ByVal parent As Form)
            Me.AssignHandle(client.Handle)
            Me.MyMdiParent = parent
        End Sub
 
        Protected Overrides Sub WndProc(ByRef m As System.Windows.Forms.Message)
            Select Case m.Msg
                Case WM_MDINEXT
                    Debug.Print("WM_MDINEXT")
                    Debug.Print("Before: " & Me.MyMdiParent.ActiveMdiChild.Name)
 
                    MyBase.WndProc(m)
                    Application.DoEvents()
 
                    Debug.Print("After: " & Me.MyMdiParent.ActiveMdiChild.Name)
 
                    ' Skip over to the next MdiChild if the current one is of Type Form2
                    If TypeOf Me.MyMdiParent.ActiveMdiChild Is Form2 Then
 
                        ' *** BE CAREFUL! ***
 
                        ' If all the MdiChildren are of Type Form2 then this
                        ' will cause an infinite loop...
 
                        SendMessage(Me.Handle, WM_MDINEXT, 0, 0)
                    End If
                    Exit Sub
 
            End Select
            MyBase.WndProc(m)
        End Sub
 
    End Class
 
 
End Class

Open in new window

0
 
LVL 3

Author Comment

by:Arikael
ID: 22773677
thanks, I will have a look at it today
0
 
LVL 3

Author Comment

by:Arikael
ID: 22774580
here is the code I used (c#)
0x0224 is 548 in the decimal system.
And I put the code into the MdiParent, into my class from which every MdiChild inherits but the code within

if(m.Msg == WM_MDINEXT)

never gets executed, because the m.Msg is never 548.
Any ideas why this could happen?
private const int WM_MDINEXT = 0x0224;
 
protected override void WndProc(ref Message m)
        {
            System.Diagnostics.Debug.WriteLine(m.Msg);
            if(m.Msg == WM_MDINEXT)
            {
                base.WndProc(ref m);
 
                if(this.ActiveMdiChild is MetaDetail)
                {
                    Application.DoEvents();
                    PostMessage(this.Handle, WM_MDINEXT, 0, 0);
                }
            }
 
            base.WndProc(ref m);
        }
 
        [DllImport("user32.dll")]
        public static extern System.IntPtr PostMessage(System.IntPtr hwnd, int msg, int wParam, int lParam);

Open in new window

0
Turn Insights Into Action

You’ve already invested in ITSM tools, chat applications, automation utilities, and more. Fortify these solutions with intelligent communications so you can drive business processes forward.

With xMatters, you'll never miss a beat.

 
LVL 86

Accepted Solution

by:
Mike Tomlinson earned 500 total points
ID: 22776541
As I said before, the WM_MDINEXT message gets sent to the MdiClient window (the gray area in the MdiParent).  This is a child window of the main MdiParent.  If you look in the Load() event, I am iterating over the Controls() collection looking for this control:

    Private Sub Form4_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles MyBase.Load
        For Each ctl As Control In Me.Controls
            If TypeOf ctl Is MdiClient Then
                client = New MyMdiClient(ctl, Me)
                Exit For
            End If
        Next
    End Sub

I pass the MdiClient to an instance of the MyMdiClient() class...which Inherits from NativeWindow().  So in my example, WndProc() is actually for the MdiClient ONLY and not the entire MdiParent.

I'll code up a working example in C# 2005 Express today if you can't figure it out.
0
 
LVL 86

Assisted Solution

by:Mike Tomlinson
Mike Tomlinson earned 500 total points
ID: 22777990
Here is a working C# version:  (C# 2005 Express)
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Diagnostics;
using System.Runtime.InteropServices;
 
namespace WindowsApplication1
{
    public partial class frmMdiParent : Form
    {
        public frmMdiParent()
        {
            InitializeComponent();
        }
 
        private MyMdiClient client = null;
 
        private void frmMdiParent_Load(object sender, EventArgs e)
        {
            foreach (Control ctl in this.Controls)
            {
                if (ctl is MdiClient)
                {
                    this.client = new MyMdiClient((MdiClient)ctl, this);
                }
            }
 
            Form1 f1 = new Form1();
            f1.MdiParent = this;
            f1.Show();
 
            Form2 f2 = new Form2();
            f2.MdiParent = this;
            f2.Show();
 
            Form3 f3 = new Form3();
            f3.MdiParent = this;
            f3.Show();
        }
 
        private class MyMdiClient : NativeWindow
        {
 
            private Form MyMdiParent = null;
            private const int WM_MDINEXT = 0x224;
 
            [DllImport("user32.dll")]
            public static extern int PostMessage(IntPtr handle, int msg, int wParam, int lParam);
            
            public MyMdiClient(MdiClient mdiClient, Form mdiParent)
            {
                this.AssignHandle(mdiClient.Handle);
                this.MyMdiParent = mdiParent;
            }
 
            protected override void WndProc(ref Message m)
            {
                switch (m.Msg)
                {
                    case WM_MDINEXT:
                        Debug.Print("WM_MDINEXT");
                        Debug.Print("Before: " + this.MyMdiParent.ActiveMdiChild.Name);
 
                        base.WndProc(ref m);
                        Application.DoEvents();
 
                        Debug.Print("After: " + this.MyMdiParent.ActiveMdiChild.Name);
 
                        // Skip over Form2 when Ctrl-Tab is pressed
                        if (this.MyMdiParent.ActiveMdiChild is Form2)
                        {                            
                            PostMessage(this.Handle, WM_MDINEXT, 0, 0);
                        }
 
                        return;
 
                }
 
                base.WndProc(ref m);
            }
 
        }
 
    }
}

Open in new window

0
 
LVL 3

Author Comment

by:Arikael
ID: 22783564
ah, sorry, I didn't unterstand the term MdiClient at first.
Now everything works as expected

I changed the code in MdiParent to (see code below), client was not defined in the c# code.

P.S. I raised the points to 500, thank you very much for your help :-)
  MyMdiClient client;
 
            foreach(Control ctl in this.Controls)
            {
                if(ctl is MdiClient)
                {
                    client = new MyMdiClient((MdiClient)ctl, this);
                    break;
                }
            }

Open in new window

0

Featured Post

More Than Just A Video Library

Train for your certification. Learn the latest DevOps tools. Grow your skillset to do better work.

At Linux Academy, we release new training modules every week so you'll always be up to date on the latest tech.

Question has a verified solution.

If you are experiencing a similar issue, please ask a related question

Introduction Hi all and welcome to my first article on Experts Exchange. A while ago, someone asked me if i could do some tutorials on object oriented programming. I decided to do them on C#. Now you may ask me, why's that? Well, one of the re…
Calculating holidays and working days is a function that is often needed yet it is not one found within the Framework. This article presents one approach to building a working-day calculator for use in .NET.
There's a multitude of different network monitoring solutions out there, and you're probably wondering what makes NetCrunch so special. It's completely agentless, but does let you create an agent, if you desire. It offers powerful scalability …
In this video we outline the Physical Segments view of NetCrunch network monitor. By following this brief how-to video, you will be able to learn how NetCrunch visualizes your network, how granular is the information collected, as well as where to f…

691 members asked questions and received personalized solutions in the past 7 days.

Join the community of 500,000 technology professionals and ask your questions.

Join & Ask a Question