Making desktopwindow at runtime with controls inside?

in the attached project, I want to make a window the width of a button, and display a listbox inside to edit.
1/ how to display the window just below the button ?
2/ why can’t I resize the window vertically ?
3/ why does the close button of the window be not visible ?
4/ will it work under windows and linux ?
5/ why does the listbox display nothing instead of the two lines ?

thanks.

testdesktopwindow.xojo_binary_project.zip (6.1 KB)

If mainWindow<>Nil Then
  mainWindow.Show
Else
  mainWindow = New DesktopWindow
  If mainWindow=Nil Then Return
  mainWindow.Left = Me.Left+Self.Left
  mainWindow.Top = Me.Top+Self.Top
  mainWindow.Width = Me.Width
  mainWindow.MaximumWidth = Me.Width
  mainWindow.Height = 300
  mainWindow.MaximumHeight = 32768
  Var myBounds As New Rect
  myBounds.Left = Me.Left+Self.Left
  myBounds.Top = Me.Top+Self.Top
  myBounds.Height = 300
  myBounds.Width = Me.Width
  mainWindow.Bounds = myBounds
  mainWindow.Resizeable = True
  mainWindow.HasCloseButton = True
  mainWindow.Title = "Edit List"
  mainWindow.Type = DesktopWindow.Types.Document
  
  mainListBox = New DesktopListBox
  If mainListBox=Nil Then Return
  mainListBox.Left = 30
  mainListBox.Width = mainWindow.Width-30
  mainListBox.Height = mainWindow.Height
  mainListBox.LockBottom = True
  mainListBox.LockLeft = True
  mainListBox.LockRight = True
  mainListBox.LockTop = True
  mainListBox.DefaultRowHeight = 12
  mainListBox.AddRow "Line 1"
  mainListBox.AddRow "Line 2"
  
  mainPlusButton = New DesktopButton
  If mainPlusButton=Nil Then Return
  mainPlusButton.Left=1
  mainPlusButton.top=1
  mainPlusButton.Width=28
  mainPlusButton.Height=28
  mainPlusButton.Caption="➕"
  
  mainWindow.AddControl mainListBox
  mainWindow.AddControl mainPlusButton
  mainWindow.Show
  mainWindow.Refresh
End If

1 add the button height
2 & 3 bug, try using a window template and use = New MyDesktopWindow
4 it should
5 seems the list must be first part of the window then the AddRow get visible

1/ as you can see in the screen capture, the window is wayyy out of where it should be, the offset is not only the button height… sorry I was not precise. I want the window just below the button, and the width of the button, something line a combobox.
5/ I added the button before the listbox and it is the same, nothing displayed inside the listbox.

will try the other tricks.
thanks Markus.

1 at windows 11 the position was correct at my pc, i thought you had move it because it was complete wrong in screenshot.
today i tested with Xojo 2024r4.2

5/ I added the button before the listbox and it is the same, nothing displayed inside the listbox.

i meant this make them visible for me

mainWindow.AddControl mainListBox
 mainListBox.AddRow "Line 1"
  mainListBox.AddRow "Line 2"

if still not maybe set the column properties.

1 Like

I have to say, this is an interesting approach, creating a window in code rather than designing it in the IDE and opening an instance. I would expect it in other languages but not Xojo. :laughing:

The window is not opening below the button because when you are defining the myBounds rect (lines 12-17 in VNSButtonList.Pressed), in this case Me and Self both are the same thing: the instance of VNSButtonList.

Instead try:

var myParent as DesktopWindow = DesktopWindow(Self.Parent)
myBounds.Left = Self.Left + myParent.Left
myBounds.Top = Self.Top + myParent.Top

As for why there’s no close button, I think that’s a bug in the framework. Try this method (warning I have not tested it on Linux):

Public Sub WindowControls(mainWindow as DesktopWindow)
  #If TargetMacOS Then
    // macOS-specific window style control
    // Window Style Mask Constants
    Const NSTitledWindowMask = 1          // Window title bar
    Const NSClosableWindowMask = 2        // Close button
    Const NSResizableWindowMask = 8       // Resize controls
    
    // Note: We're explicitly excluding NSMiniaturizableWindowMask (4) - minimize button
    
    Declare Sub setStyleMask Lib "Cocoa.framework" selector "setStyleMask:" (obj_id As Integer, mask As Integer)
    
    Var WinHandle As OSHandle = mainWindow.Handle
    
    // Combine only the masks we want:
    // - Title bar
    // - Close button
    // - Resize controls
    // Explicitly NOT including minimize or maximize/zoom
    setStyleMask(WinHandle.Value, NSTitledWindowMask Or NSClosableWindowMask Or NSResizableWindowMask)
    
  #ElseIf TargetWindows Then
    // Windows-specific style control
    Const GWL_STYLE = -16
    Const WS_MAXIMIZEBOX = &H00010000
    Const WS_SYSMENU = &H00080000      // System menu (includes close button)
    Const WS_CAPTION = &H00C00000      // Title bar
    Const WS_THICKFRAME = &H00040000   // Sizing border (for resizable windows)
    Const WS_MINIMIZEBOX = &H00020000  // Minimize button
    
    Declare Function GetWindowLong Lib "User32" Alias "GetWindowLongW" (hwnd As Integer, nIndex As Integer) As Integer
    Declare Function SetWindowLong Lib "User32" Alias "SetWindowLongW" (hwnd As Integer, nIndex As Integer, dwNewLong As Integer) As Integer
    
    Var WinHandle As OSHandle = mainWindow.Handle
    Var Style As Integer = GetWindowLong(WinHandle.Value, GWL_STYLE)
    
    // First, make sure we DO have the styles needed for a close button
    Style = Style Or WS_CAPTION Or WS_SYSMENU
    
    // Then remove only the maximize box style while keeping other styles
    Style = Style And (Not WS_MAXIMIZEBOX)
    
    // Ensure resize border if window should be resizable
    If mainWindow.Resizeable Then
      Style = Style Or WS_THICKFRAME
    End If
    
    // IMPORTANT: Apply the style change to the window
    Var Result As Integer = SetWindowLong(WinHandle.Value, GWL_STYLE, Style)
    
    // Check if the operation succeeded
    If Result = 0 Then
      // GetLastError() can provide more info about what went wrong
      System.DebugLog("SetWindowLong failed to apply window style")
    End If
    
    // Force the window to redraw with the new styles
    Declare Sub SetWindowPos Lib "User32" (hwnd As Integer, hWndInsertAfter As Integer, x As Integer, y As Integer, cx As Integer, cy As Integer, uFlags As UInteger)
    Const SWP_FRAMECHANGED = &H0020
    Const SWP_NOMOVE = &H0002
    Const SWP_NOSIZE = &H0001
    Const SWP_NOZORDER = &H0004
    
    SetWindowPos(WinHandle.Value, 0, 0, 0, 0, 0, SWP_FRAMECHANGED Or SWP_NOMOVE Or SWP_NOSIZE Or SWP_NOZORDER)
    
  #ElseIf TargetLinux Then
    // Linux (GTK) specific code
    Const GDK_FUNC_MAXIMIZE = 4
    
    Declare Sub gtk_window_set_type_hint Lib "libgtk-3.so.0" (window As Integer, hint As Integer)
    Declare Sub gtk_window_set_functions Lib "libgtk-3.so.0" (window As Integer, functions As Integer)
    
    Var WinHandle As OSHandle = mainWindow.Handle
    
    // Get all window functions except maximize
    Var AllFunctions As Integer = -1  // All functions enabled
    Var NoMaximize As Integer = AllFunctions And (Not GDK_FUNC_MAXIMIZE)
    
    // Apply the window functions
    gtk_window_set_functions(WinHandle.Value, NoMaximize)
  #EndIf
End Sub

Call WindowControls (MainWindow) just before mainWindow.Show. Warning that I have not tested the Linux code, but MacOS and Windows seem to work.

1 Like

thanks for such extended code ! I wouldn’t expect it to be so complex…

the idea behind that is to create a window with code, so that it’s easier to deal with external modules
as you can’t embed a class (so a window) in an external module, it would be easier to handle multiple window creation with only one external module (and not also each external class/window)

as it seems still too much effort, although xojo add the ability to create controls at runtime, I will stop the experiment and go a more classic way.

thanks to all.