Posted 23 September 2010 - 09:02 PM
You get this error if you call ResizeControls directly.
You need to call ResizeForms. That's where iheight is defined.
In other news:
It seems that GetLocation should be called by ResizeForms, is this an omission? I stuck it in there and it seems to work...
...however, resizing the form won't let me drag edges too close together.
Posted 11 October 2010 - 08:53 AM
I have a set of procedures, much like what you describe, that do a good job of resizing just about everything in a VB6 dialog. There's just a couple of small glitches.
1. In a dialog having a picturebox into which a picture is stretchblt-ed and some pixels are drawn. During a resize, the picture disappears. I can get it back by reselecting and loading the picture. During form load, the picture fails to display. Again, reselecting and loading the picture gets it back. It's as if picture loading is not allowed during a Load or Resize event. Redisplay is always a part of the resize event.
2. the other issue is with a dialog that has runtime created controls (scrollbars). If I don't use my resizing, all is well. If I use resizing, as soon as I change the size by the tiniest amount, the display goes haywire. The dynamically created scrollbars disappear, much of the gray background goes white, and several controls no longer display. Again, redisplay is a part of the resize event.
A couple of comments on your tutorial:
1. The dialog needs to save its size in the Resize event so that the same size can be used on form load in future executions.
2. It's a good thing you used .Left, .Top, .Width, and .Height in your resize as .Move does not work for ALL controls. Specifically, Comboboxes are not supported.
3. Your code does not support all control types. I found that lines don't have the same size variables, but use X1,Y1 and X2,Y2.
Posted 14 October 2010 - 09:04 AM
Problem #2 solved. My saving of the dialog specifics was taking place before the dynamically created scrollbars were set up. An On Error Resume Next was hiding the problem.
Problem #1 evaded, though not elegantly. I created a short timer to allow auto-clicking the background select combobox after the Resize event. The background now survives resize and shows up on the initial form load.
Posted 09 November 2010 - 01:54 AM
by d way im new here. novice in programming.
im just making my own system for my practise. cause im not that good enough.
thanks for the source. it will really help me a lot. but a have 1 more problem. i don't know how I'm gonna put it in my system that I'm creating. can somebody please tell me how.
by d way I'm new here. novice in programming.
I'm just making my own system for my practice. cause I'm not that good enough.
Posted 08 December 2010 - 10:03 PM
I'm trying to get the screen size using
xx.height = screen.height
xx.width = screen.width
When I try and run the code I get the following error when it reaches this line
Runtime error '424': Object required
What am I missing here, why doesn't it recognise 'screen' as a valid object?
Posted 18 February 2011 - 10:59 AM
On Error Resume Next
That will bypass the TabIndex error, if you leave the code as is. Nonetheless, none of that tabindex code is required at all. There is a much simpler solution by just referencing each control by its name -- but the original contributor has a bug which must be fixed first as follows.
Update the GetLocation procedure from this:
With List(i) .Name = curr_obj ...
With List(i) .Name = curr_obj.Name ...
With that corrected, you can completely erase and replace ResizeControls with this much simpler code that references each control's properties by the name of the control (stored in the List() array). That will remove all TabIndex errors as we don't need that property for anything functionality of these routines:
Public Sub ResizeControls(frm As Form) On Error Resume Next Dim i As Integer ' Get ratio of initial form size to current form size x_size = frm.height / iHeight y_size = frm.width / iWidth 'Loop though all the control objects in the array 'based on the upper bound of the # of controls For i = 0 To UBound(List) frm(List(i).Name).Left = List(i).Left * y_size frm(List(i).Name).width = List(i).width * y_size frm(List(i).Name).height = List(i).height * x_size frm(List(i).Name).Top = List(i).Top * x_size Next Err.Clear: On Error GoTo 0 End Sub
Finally, clean the rest of the code by removing the Index member from the Control type as follows:
Private Type Control Index As Integer Name As String ...
Private Type Control Name As String ...
And in GetLocation, changing this:
With List(i) .Name = curr_obj.Name .Index = curr_obj.TabIndex .Left = curr_obj.Left ...
With List(i) .Name = curr_obj.Name .Left = curr_obj.Left ...
There's also math issues with this code that create bigger margins around things as the form size grows, but I haven't debugged them yet, and the issue is negligible anyway.
Posted 18 February 2011 - 02:27 PM
Posted 13 July 2011 - 12:00 AM
- Resize a form based on screen size
- Resize the controls on the form based on the form size
- Resize the font size of all controls based on the above 2 items
The code I'm about to show you I have in a Code Module, names FormControl that I include in all VB6 projects I create. This comes in real handy because as developers we don't know what screen resolution a client, or anyone else using our software, will be using. The first thing I have in my module is some Global variables that will be used through out the module, so I, of course, make them Global and accessible to the entire module. Here are the Globals you need to add to your Code Module:
Private List() As Control Private curr_obj As Object Private iHeight As Integer Private iWidth As Integer Private x_size As Double Private y_size As Double
As you will see, the Globals are Private, I don't want them to be accessed from outside this Code Module. The first Global is an Array or type Control. This is a private Type I have created to hold all the properties of the controls on the form. Creating your own user defined type saves a lot of headaches down the road, and allows for resizing all controls in a single loop, referencing your type:
Private Type Control Index As Integer Name As String Left As Integer Top As Integer width As Integer height As Integer End Type
As you can see, this Type holds all the information needed about a control: Index, Name, Left, Top, Width and Height, these items will come in use later, when we write the procedure to resize all the controls at once. Now lets talk about resizing a form based on the current screen resolution we will be referencing the Screen Object available to us in Visual Basic 6.
The 2 properties we're concerned with are Width and Height. These properties of the Screen Object give us access to the screen size available to us. So, to set the Forms size in relationship to the Screen size, we will be accessing the properties of the Form Object, mainly the width and height properties. In my Module I set the forms size to the screen's size divided by 2, you may want to test and find your own resolution. Here's the simple procedure for resizing your form in relationship to the available screen resolution:
Public Sub ResizeForm(frm As Form) 'Set the forms height frm.height = Screen.height / 2 'Set the forms width frm.width = Screen.width / 2 'Resize all of the controls 'based on the forms new size ResizeControls frm End Sub
Simple isn't it, we change the forms size based on the screen size, then we reference a procedure called ResizeControls, we do this because resizing the form alone isn't what we're after. Well doing only this will cause some pretty ugly user interfaces, simply because you may be resizing your form, but the controls are staying the same size, which isn't a good thing.
So, you can either write a really long and ugly procedure to resize each control individually,not very maintainable, especially if you rename or add controls, or you can write a nice neat little procedure, based on a user defined type, which we have in our Code Module, and loop through them like this:
Public Sub ResizeControls(frm As Form) Dim i As Integer ' Get ratio of initial form size to current form size x_size = frm.height / iHeight y_size = frm.width / iWidth 'Loop though all the objects on the form 'Based on the upper bound of the # of controls For i = 0 To UBound(List) 'Grad each control individually For Each curr_obj In frm 'Check to make sure its the right control If curr_obj.TabIndex = List(i).Index Then 'Then resize the control With curr_obj .Left = List(i).Left * y_size .width = List(i).width * y_size .height = List(i).height * x_size .Top = List(i).Top * x_size End With End If 'Get the next control Next curr_obj Next i End Sub
Here we use the UBound Function To get the highest index on the form, which gives us the last control's index. We then loop through all the controls, stopping at the highest index, and change the size of the control. Before this procedure can actually work, we need to know the current location of each control, well I have a solution for that as well.
In this Module is a method called GetLocation. What this method does is it logs the current position of each control, looping through all the controls located in our List() Array, which is populated with the controls on the form. On each iteration of the loop, we use the ReDim Statement and the Preserve keyword to increment the size of the array by 1 and preserve the objects already in the array. The code for that method is as follows:
Public Sub GetLocation(frm As Form) Dim i As Integer ' Load the current positions of each object into a user defined type array. ' This information will be used to rescale them in the Resize function. 'Loop through each control For Each curr_obj In frm 'Resize the Array by 1, and preserve 'the original objects in the array ReDim Preserve List(i) With List(i) .Name = curr_obj .Index = curr_obj.TabIndex .Left = curr_obj.Left .Top = curr_obj.Top .width = curr_obj.width .height = curr_obj.height End With i = i + 1 Next curr_obj ' This is what the object sizes will be compared to on rescaling. iHeight = frm.height iWidth = frm.width End Sub
Our Code Module is almost complete, there is but one thing we need to think about resizing, and that is the font size used in the application. Resizing, say a Label, and not resizing the font being used on the label will make for an ugly User Interface as well. So what we do is, we get the value of x_size, which is one of our Globals, and set in the ResizeControls method, we multiply that by 8 and that gives the font size I'm looking for, you needs may vary. Here is the code for resizing the font size:
Public Function SetFontSize() As Integer 'Make sure x_size is greater than 0 If Int(x_size) > 0 Then 'Set the font size SetFontSize = Int(x_size * 8) End If End Function
That's all the code you need for your form and control resizing based on the users screen resolution. It's all a matter of getting the mathematics right, then resetting the size of everything on the form to fit nicely in the current screen.
Now for how to use it, in the Form_Load Event you add a call to:
Private Sub Form_Load() GetLocation Me CenterForm Me ResizeForm Me lblInstructions.Font = SetFontSize() End Sub
This Code Module also offers another feature. If someone were the resize your form while havin the application running, if you put a call to ResizeControls in the Form_Resize Event, this will take care of resizing everthing:
Private Sub Form_Resize() ResizeControls Me lblInstructions.FontSize = SetFontSize() End Sub
I am providing the Code Module this code is in, this Module and code is under the GNU General Public License, so you can use, modify or distribute as you see fit, but the license header must stay intact. I hope you have found this tutorial informative and useful. For all your Visual Basic 6 Reference Needs go to the Visual Basic 6.0 Resource Center. Thank you for reading.
This code will not work if the form contains an image or timer control.The reason is that controls like this do not have tabIndex like properties.Can you correct it
Posted 13 July 2011 - 07:11 AM
So I signed up in hope of help.
I got vb6 and loaded this module and added code to form_load()
I get a err.num 438
"Run-time error '438' Object doesn't support this property or method"
It don't like tab.index
I tried a few changes but nothing helped.
Any clues to why I am getting this error?
The problem is that not all controls support all the size properties
Posted 28 February 2012 - 09:09 AM
well i attach my code and try it if you want. it worked with almost everything except tabs.
a normal call to this would be:
and if you want to setup manually some controls just do this before the resize controls
SetResize Me, RESIZEUP, TypeName(Command1) ' this will resize just up the screen all the command buttons
ComboBox1.Tag = RESIZEUP ' this will resize this specific combo box up the screen
Text1.Tag = RESIZEUPWIDTH ' this will resize up the screen and only the with of this specific text box
and you can explore more alternatives.
Posted 08 August 2012 - 11:12 AM
I applied the code to resize a DataGrid Form two and a SSTab and everything perfect.
But in another Form with a DataGrid and Tab SSTab but with two, I have problems.
The problem is that in the Tab (0) which is the first thing to appear, overlapping controls Tab (1) and the controls that appear in the Tab (0) and Tab (1) do not appear in the Tab (1) .
How I can fix this problem?.
Posted 21 March 2013 - 02:21 AM
The code in Form Load is just:
The code in GetLocation with my modification to handel the error:
Public Sub GetLocation(frm As Form)
Dim i As Integer
On Error Resume Next
' Load the current positions of each object into a user defined type array.
' This information will be used to rescale them in the Resize function.
'Loop through each control
For Each curr_obj In frm
'Resize the Array by 1, and preserve
'the original objects in the array
ReDim Preserve List(i)
.Name = curr_obj
.Index = curr_obj.TabIndex
.Left = curr_obj.Left
.Top = curr_obj.Top
.width = curr_obj.width
.height = curr_obj.height
If Err > 0 Then
i = i - 1
Err = 0
i = i + 1
On Error GoTo ErrorGetLocation
' This is what the object sizes will be compared to on rescaling.
iHeight = frm.height
iWidth = frm.width
MsgBox "Error " & Err.Number & " (" & Err.Description & ") in procedure GetLocation of Formular Form1"
in Form_Resize I use only:
Private Sub Form_Resize()
In Resize Control I adjust directly the font size:
Public Sub ResizeControls(frm As Form)
Dim i As Integer
' Get ratio of initial form size to current form size
If iHeight = 0 Or iWidth = 0 Then Exit Sub
x_size = frm.height / iHeight
y_size = frm.width / iWidth
'Loop though all the objects on the form
'Based on the upper bound of the # of controls
For i = 0 To UBound(List)
'Grad each control individually
For Each curr_obj In frm
'Check to make sure its the right control
On Error GoTo GetNextControl
If curr_obj.TabIndex = List(i).Index Then
'Then resize the control
.Left = List(i).Left * y_size
.width = List(i).width * y_size
.height = List(i).height * x_size
.Top = List(i).Top * x_size
'If curr_obj.Name = "txtFieldContent" Then Stop
.FontSize = SetFontSize()
'Get the next control
Err = 0