Removing the Title bar of external application using c#
Asked Answered
C

7

19

My application starts up another external application.

I want to remove the title bar of this external application once it has started.

Is this feasible, and if so how would it be done?

Based on the comments I am using the working code below

//Finds a window by class name
[DllImport("USER32.DLL")]
public static extern IntPtr FindWindow(string lpClassName, string lpWindowName);

//Sets a window to be a child window of another window
[DllImport("USER32.DLL")]
public static extern IntPtr SetParent(IntPtr hWndChild, IntPtr hWndNewParent);

//Sets window attributes
[DllImport("USER32.DLL")]
public static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);

//Gets window attributes
[DllImport("USER32.DLL")]
public static extern int GetWindowLong(IntPtr hWnd, int nIndex);

[DllImport("user32.dll", EntryPoint = "FindWindow", SetLastError = true)]
static extern IntPtr FindWindowByCaption(IntPtr ZeroOnly, string lpWindowName);


//assorted constants needed
public static int GWL_STYLE = -16;
public static int WS_CHILD = 0x40000000; //child window
public static int WS_BORDER = 0x00800000; //window with border
public static int WS_DLGFRAME = 0x00400000; //window with double border but no title
public static int WS_CAPTION = WS_BORDER | WS_DLGFRAME; //window with a title bar

public void WindowsReStyle()
{
    Process[] Procs = Process.GetProcesses();
    foreach (Process proc in Procs)
    {
        if (proc.ProcessName.StartsWith("notepad"))
        {
            IntPtr pFoundWindow = proc.MainWindowHandle;
            int style = GetWindowLong(pFoundWindow, GWL_STYLE);
            SetWindowLong(pFoundWindow, GWL_STYLE, (style & ~WS_CAPTION));
        }
    }
}
Centimeter answered 13/5, 2010 at 9:7 Comment(1)
public static int WS_CAPTION = WS_BORDER | WS_DLGFRAME; is incorrect since WS_DLGFRAME is not part of caption. Reference: msdn.microsoft.com/en-us/library/windows/desktop/…Burg
P
13

No need to inject anything, you can just modify the windows style bits as using the API, e.g. this works for Notepad, however YMMV depending on the app you're playing with.

alt text

//Get current style
lCurStyle = GetWindowLong(hwnd, GWL_STYLE)

//remove titlebar elements
lCurStyle = lCurStyle And Not WS_CAPTION
lCurStyle = lCurStyle And Not WS_SYSMENU
lCurStyle = lCurStyle And Not WS_THICKFRAME
lCurStyle = lCurStyle And Not WS_MINIMIZE
lCurStyle = lCurStyle And Not WS_MAXIMIZEBOX

//apply new style
SetWindowLong hwnd, GWL_STYLE, lCurStyle

//reapply a 3d border
lCurStyle = GetWindowLong(hwnd, GWL_EXSTYLE)

SetWindowLong hwnd, GWL_EXSTYLE, lCurStyle Or WS_EX_DLGMODALFRAME

//redraw
SetWindowPos hwnd, 0, 0, 0, 0, 0, SWP_NOMOVE Or SWP_NOSIZE Or SWP_FRAMECHANGED
Peracid answered 13/5, 2010 at 10:30 Comment(2)
Sure; #2833328 ;)Peracid
The imageshack URL seems to be "dead"?Former
S
9

Well, Alex never elaborated with the code, well at least it wasn't a plug-n-play solution, but still the main credit for this goes to him... It IS kind of buggy unless you use "SetParent" to put it in some kind of container(such as a form or panel) Just thought I would share the result.

Visual Basic:

Option Strict On
Public Class Form1
    Const WS_BORDER As Integer = 8388608
    Const WS_DLGFRAME As Integer = 4194304
    Const WS_CAPTION As Integer = WS_BORDER Or WS_DLGFRAME
    Const WS_SYSMENU As Integer = 524288
    Const WS_THICKFRAME As Integer = 262144
    Const WS_MINIMIZE As Integer = 536870912
    Const WS_MAXIMIZEBOX As Integer = 65536
    Const GWL_STYLE As Integer = -16&
    Const GWL_EXSTYLE As Integer = -20&
    Const WS_EX_DLGMODALFRAME As Integer = &H1L
    Const SWP_NOMOVE As Integer = &H2
    Const SWP_NOSIZE As Integer = &H1
    Const SWP_FRAMECHANGED As Integer = &H20
    Const MF_BYPOSITION As UInteger = &H400
    Const MF_REMOVE As UInteger = &H1000
    Declare Auto Function GetWindowLong Lib "user32.dll" (ByVal hWnd As IntPtr, ByVal nIndex As Integer) As Integer
    Declare Auto Function SetWindowLong Lib "user32.dll" (ByVal hWnd As IntPtr, ByVal nIndex As Integer, ByVal dwNewLong As Integer) As Integer
    Declare Auto Function SetWindowPos Lib "user32.dll" (ByVal hWnd As IntPtr, ByVal hWndInsertAfter As IntPtr, ByVal X As Integer, ByVal Y As Integer, ByVal cx As Integer, ByVal cy As Integer, ByVal uFlags As Integer) As Boolean
    Public Sub MakeExternalWindowBorderless(ByVal MainWindowHandle As IntPtr)
        Dim Style As Integer
        Style = GetWindowLong(MainWindowHandle, GWL_STYLE)
        Style = Style And Not WS_CAPTION
        Style = Style And Not WS_SYSMENU
        Style = Style And Not WS_THICKFRAME
        Style = Style And Not WS_MINIMIZE
        Style = Style And Not WS_MAXIMIZEBOX
        SetWindowLong(MainWindowHandle, GWL_STYLE, Style)
        Style = GetWindowLong(MainWindowHandle, GWL_EXSTYLE)
        SetWindowLong(MainWindowHandle, GWL_EXSTYLE, Style Or WS_EX_DLGMODALFRAME)
        SetWindowPos(MainWindowHandle, New IntPtr(0), 0, 0, 0, 0, SWP_NOMOVE Or SWP_NOSIZE Or SWP_FRAMECHANGED)
    End Sub
End Class

C Sharp(C#)

using System.Runtime.InteropServices;
public class Form1
{
    const int WS_BORDER = 8388608;
    const int WS_DLGFRAME = 4194304;
    const int WS_CAPTION = WS_BORDER | WS_DLGFRAME;
    const int WS_SYSMENU = 524288;
    const int WS_THICKFRAME = 262144;
    const int WS_MINIMIZE = 536870912;
    const int WS_MAXIMIZEBOX = 65536;
    const int GWL_STYLE = -16L;
    const int GWL_EXSTYLE = -20L;
    const int WS_EX_DLGMODALFRAME = 0x1L;
    const int SWP_NOMOVE = 0x2;
    const int SWP_NOSIZE = 0x1;
    const int SWP_FRAMECHANGED = 0x20;
    const uint MF_BYPOSITION = 0x400;
    const uint MF_REMOVE = 0x1000;
    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true)]
    public static extern int GetWindowLong(IntPtr hWnd, int nIndex);
    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true)]
    public static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);
    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true)]
    public static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, int uFlags);
    public void MakeExternalWindowBorderless(IntPtr MainWindowHandle)
    {
        int Style = 0;
        Style = GetWindowLong(MainWindowHandle, GWL_STYLE);
        Style = Style & ~WS_CAPTION;
        Style = Style & ~WS_SYSMENU;
        Style = Style & ~WS_THICKFRAME;
        Style = Style & ~WS_MINIMIZE;
        Style = Style & ~WS_MAXIMIZEBOX;
        SetWindowLong(MainWindowHandle, GWL_STYLE, Style);
        Style = GetWindowLong(MainWindowHandle, GWL_EXSTYLE);
        SetWindowLong(MainWindowHandle, GWL_EXSTYLE, Style | WS_EX_DLGMODALFRAME);
        SetWindowPos(MainWindowHandle, new IntPtr(0), 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_FRAMECHANGED);
    }
}

Again, Thank you Alex

Simeon answered 16/1, 2013 at 4:7 Comment(1)
Hi, thanks for your code, it works for me after correction some errors in the C# section. Style & !WS_CAPTION does not work. You need to write Style & ~WS_CAPTION.Savitt
T
9
Process[] processes = Process.GetProcessesByName("notepad");
IntPtr windowHandle = processes[0].MainWindowHandle;

const int GWL_STYLE = (-16); 
const UInt32 WS_VISIBLE = 0x10000000;
SetWindowLong(windowHandle, GWL_STYLE, (WS_VISIBLE));`
Theron answered 16/3, 2013 at 12:37 Comment(0)
S
1

This is very similar to a previously asked question, and I'm pretty sure that the answer is you can't do it. (or, if you can, you need to dig into the Windows API, which can be challenging, depending on your experience.)

How to add button to other apps window title bar (XP/Vista)

Swithin answered 13/5, 2010 at 9:15 Comment(1)
This is not correct. You can do it, and the solutions listed here do work.Brookes
G
0

If Syncfusion, no. If not, yes.

VB:

Const WS_BORDER As Integer = 8388608
Const WS_DLGFRAME As Integer = 4194304
Const WS_CAPTION As Integer = WS_BORDER Or WS_DLGFRAME
Const WS_SYSMENU As Integer = 524288
Const WS_THICKFRAME As Integer = 262144
Const WS_MINIMIZE As Integer = 536870912
Const WS_MAXIMIZEBOX As Integer = 65536
Const GWL_STYLE As Integer = -16&
Const GWL_EXSTYLE As Integer = -20&
Const WS_EX_DLGMODALFRAME As Integer = &H1L
Const SWP_NOMOVE As Integer = &H2
Const SWP_NOSIZE As Integer = &H1
Const SWP_FRAMECHANGED As Integer = &H20
Const MF_BYPOSITION As UInteger = &H400
Const MF_REMOVE As UInteger = &H1000
Declare Auto Function GetWindowLong Lib "user32.dll" (ByVal hWnd As IntPtr, ByVal nIndex As Integer) As Integer
Declare Auto Function SetWindowLong Lib "user32.dll" (ByVal hWnd As IntPtr, ByVal nIndex As Integer, ByVal dwNewLong As Integer) As Integer
Declare Auto Function SetWindowPos Lib "user32.dll" (ByVal hWnd As IntPtr, ByVal hWndInsertAfter As IntPtr, ByVal X As Integer, ByVal Y As Integer, ByVal cx As Integer, ByVal cy As Integer, ByVal uFlags As Integer) As Boolean
Public Sub MakeExternalWindowBorderless(ByVal MainWindowHandle As IntPtr)
    Dim Style As Integer
    Style = GetWindowLong(MainWindowHandle, GWL_STYLE)
    Style = Style And Not WS_CAPTION
    Style = Style And Not WS_SYSMENU
    Style = Style And Not WS_THICKFRAME
    Style = Style And Not WS_MINIMIZE
    Style = Style And Not WS_MAXIMIZEBOX
    SetWindowLong(MainWindowHandle, GWL_STYLE, Style)
    Style = GetWindowLong(MainWindowHandle, GWL_EXSTYLE)
    SetWindowLong(MainWindowHandle, GWL_EXSTYLE, Style Or WS_EX_DLGMODALFRAME)
    SetWindowPos(MainWindowHandle, New IntPtr(0), 0, 0, 0, 0, SWP_NOMOVE Or SWP_NOSIZE Or SWP_FRAMECHANGED)
End Sub

C#:

const int WS_BORDER = 8388608;
const int WS_DLGFRAME = 4194304;
const int WS_CAPTION = WS_BORDER | WS_DLGFRAME;
const int WS_SYSMENU = 524288;
const int WS_THICKFRAME = 262144;
const int WS_MINIMIZE = 536870912;
const int WS_MAXIMIZEBOX = 65536;
const int GWL_STYLE = -16L;
const int GWL_EXSTYLE = -20L;
const int WS_EX_DLGMODALFRAME = 0x1L;
const int SWP_NOMOVE = 0x2;
const int SWP_NOSIZE = 0x1;
const int SWP_FRAMECHANGED = 0x20;
const uint MF_BYPOSITION = 0x400;
const uint MF_REMOVE = 0x1000;
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true)]
public static extern int GetWindowLong(IntPtr hWnd, int nIndex);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true)]
public static extern int SetWindowLong(IntPtr hWnd, int nIndex, int dwNewLong);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true, ExactSpelling = true)]
public static extern bool SetWindowPos(IntPtr hWnd, IntPtr hWndInsertAfter, int X, int Y, int cx, int cy, int uFlags);
public void MakeExternalWindowBorderless(IntPtr MainWindowHandle)
{
    int Style = 0;
    Style = GetWindowLong(MainWindowHandle, GWL_STYLE);
    Style = Style & ~WS_CAPTION;
    Style = Style & ~WS_SYSMENU;
    Style = Style & ~WS_THICKFRAME;
    Style = Style & ~WS_MINIMIZE;
    Style = Style & ~WS_MAXIMIZEBOX;
    SetWindowLong(MainWindowHandle, GWL_STYLE, Style);
    Style = GetWindowLong(MainWindowHandle, GWL_EXSTYLE);
    SetWindowLong(MainWindowHandle, GWL_EXSTYLE, Style | WS_EX_DLGMODALFRAME);
    SetWindowPos(MainWindowHandle, new IntPtr(0), 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_FRAMECHANGED);
}
Grugru answered 1/4, 2020 at 15:45 Comment(0)
E
0

Python version of Alex's answer if someone else needs:

import win32gui

#  get a handle to the window
windowHandle = win32gui.FindWindowEx(None, None, None, "Untitled - Notepad")

GWL_STYLE = -16  #  see https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-setwindowlonga

#  get current window style
currentStyle = win32gui.GetWindowLong(windowHandle, GWL_STYLE)

#  remove titlebar elements
currentStyle = currentStyle & ~(0x00C00000)  #  WS_CAPTION
currentStyle = currentStyle & ~(0x00080000)  #  WS_SYSMENU
currentStyle = currentStyle & ~(0x00040000)  #  WS_THICKFRAME
currentStyle = currentStyle & ~(0x20000000)  #  WS_MINIMIZE
currentStyle = currentStyle & ~(0x00010000)  #  WS_MAXIMIZEBOX

#  apply new style
win32gui.SetWindowLong(windowHandle, GWL_STYLE, currentStyle)
Eloyelreath answered 23/11, 2020 at 8:59 Comment(0)
P
-1

In general, you can't do that unless there is direct support for in by the application you are starting (for example, if it takes a command line switch for removal of the title bar).

You can only control the things that are already present on the ProcessStartInfo class (i.e. open a new window, start minimized/maximized etc...).

Potvaliant answered 13/5, 2010 at 9:14 Comment(3)
Is there any other way to hide the title bar of this third-party application ? It has only one form for which i need to remove it.Centimeter
@karthik - Only if it already supports this functionality. This is not something you can control directly.Potvaliant
Hey, you DO have control over that. The Win32 User API (user32) is open and you can control any window with it.Grugru

© 2022 - 2024 — McMap. All rights reserved.