Change/Amend VBA Code To Run Fast

hrayani

Well-known Member
Joined
Jul 23, 2010
Messages
1,549
Office Version
  1. 2016
Platform
  1. Windows
Hello Friends,

I have a set of data which contains shipment records. I have to hide the rows where Column O
shows the status of the Shipment.

I am using the below mentioned code to hide rows based on Column O

PHP:
Private Sub CommandButton1_Click()

For a = 3 To 500

If Worksheets("RO").Cells(a, 15).Value = "Shipped" Then
Worksheets("RO").Rows(a).Hidden = False
End If
If Worksheets("RO").Cells(a, 15).Value = "In Progress" Then
Worksheets("RO").Rows(a).Hidden = True
End If
Next

End Sub

PHP:
Private Sub CommandButton2_Click()

For a = 3 To 500

If Worksheets("RO").Cells(a, 15).Value = "Shipped" Then
Worksheets("RO").Rows(a).Hidden = True
End If
If Worksheets("RO").Cells(a, 15).Value = "In Progress" Then
Worksheets("RO").Rows(a).Hidden = False
End If
Next

End Sub

PHP:
Private Sub CommandButton3_Click()

For a = 3 To 500

If Worksheets("RO").Cells(a, 15).Value = "Shipped" Then
Worksheets("RO").Rows(a).Hidden = False
End If
If Worksheets("RO").Cells(a, 15).Value = "In Progress" Then
Worksheets("RO").Rows(a).Hidden = False
End If
Next

End Sub


Is there any other way out (code) that can provide the desired result a bit fast or can we
change the code to make it work fast

Any help would be appreciated

Regards,

Humayun
 

Excel Facts

Excel Joke
Why can't spreadsheets drive cars? They crash too often!
Hi there.

Assuming not all 500 rows are in use, you can set the loop to just check active rows. Also, by turning off screen updating, it will run faster. Your CommandButton3_Click code would then look like this:
Code:
Private Sub CommandButton3_Click()

Application.ScreenUpdating = False
lastrow = Worksheets("RO").Cells(Rows.Count, 1).End(xlUp).Row

For a = 3 To lastrow

If Worksheets("RO").Cells(a, 15).Value = "Shipped" Then
Worksheets("RO").Rows(a).Hidden = False
End If
If Worksheets("RO").Cells(a, 15).Value = "In Progress" Then
Worksheets("RO").Rows(a).Hidden = False
End If
Next
Application.ScreenUpdating = True
End Sub
 
Last edited:
Upvote 0
I should point out that that code will only give the correct last row if all the rows are unhidden, but adding the application.screenupdating will make it much quicker anyway, so you could leave it as 3 to 500
 
Last edited:
Upvote 0
One of the main reasons that Vba is slow is the time taken to access the worksheet from VBa is a relatively long time.
To speed up vba the easiest way is to minimise the number of accesses to the worksheet. What is interesting is that the time taken to access a single cell on the worksheet in vba is almost identical as the time taken to access a large range if it is done in one action.
I have a simple rule for fast VBA: NEVER ACCESS THE WORKSHEET IN A LOOP.
Unfortunately in your case where you want to hide and unhide rows we do have access the worksheet in a loop, but we can eliminate testing the cell for shipped or in progress from the loop which should halve the time or so.
Thry this:
Code:
inarr = Range(Cells(1, 15), Cells(500, 15))

For a = 3 To 500


If inarr(a, 1) = "Shipped" Then
Worksheets("RO").Rows(a).Hidden = False
End If
If inarr(a, 1) = "In Progress" Then
Worksheets("RO").Rows(a).Hidden = True
End If
Next a
 
Upvote 0
Thanks jmacleary, for the code.

I tried the code you provided but its taking same time as before. Only difference is i cannot see the
screen updating - understandably that you have set it to false.
 
Upvote 0
Thanks offthelip for code,

Its working but taking same or more time i would say....

Mine one is the fastest :)

Lets see if any friend has a solution to it...
 
Upvote 0
Unfortunately in your case .. we do have access the worksheet in a loop ..
I disagree - see below


hrayani,
Give this a try in a copy of your workbook.
Timing will depend on hardware, what else you have in the sheet (eg formulas) etc., but for me this was more than 10 times faster.
Assumes column Z can be used as a helper. If not choose any free column.

Rich (BB code):
Private Sub CommandButton1_Click()
  Dim Data As Variant, b As Variant
  Dim i As Long
  
  Application.ScreenUpdating = False
  Application.Calculation = xlCalculationManual
  With Worksheets("RO")
    Data = .Range("O3:O500").Value
    ReDim b(1 To UBound(Data), 1 To 1)
    For i = 1 To UBound(Data)
      Select Case LCase(Data(i, 1))
        Case "shipped": b(i, 1) = 9
        Case "in progress": b(i, 1) = "x"
      End Select
    Next i
    With .Range("Z3").Resize(UBound(b))
      .Value = b
      On Error Resume Next
      .SpecialCells(xlConstants, xlNumbers).EntireRow.Hidden = False
      .SpecialCells(xlConstants, xlTextValues).EntireRow.Hidden = True
      On Error GoTo 0
      .ClearContents
    End With
  End With
  Application.Calculation = xlCalculationAutomatic
  Application.ScreenUpdating = True
End Sub
 
Last edited:
Upvote 0
Thanks Peter,

I tried the code but its debugging with this part
Value = b
of the code highlighting.


Assumes column Z can be used as a helper. If not choose any free column

sorry i did not get it ???? Do i have to do anything in this Column ?
My Data ends at Column O (In Progress/Shipped)
 
Last edited:
Upvote 0
I tried the code but its debugging with this part of the code highlighting.
Value = b
Have you accidentally deleted the "." from before the word Value?




sorry i did not get it ???? Do i have to do anything in this Column ?
My Data ends at Column O (In Progress/Shipped)
If your data ends at column O you don't have to do anything about that part of the code.
 
Last edited:
Upvote 0
as another attempt try following:


In Standard Module

Code:
Sub ShowRows(ByVal Shipped As Boolean, ByVal InProgress As Boolean)
    Dim LastRow As Long, a As Long
    On Error GoTo myerror
    With Application
        .ScreenUpdating = False: .EnableEvents = False: .Calculation = xlCalculationManual
    End With
    With ThisWorkbook.Worksheets("RO")
            .Cells.Rows.Hidden = False
        LastRow = .Cells(.Rows.Count, 15).End(xlUp).Row
        For a = 3 To LastRow
            .Cells(a, 1).EntireRow.Hidden = IIf(.Cells(a, 15).Value = "Shipped", Shipped, _
                                            IIf(.Cells(a, 15).Value = "In Progress", InProgress, False))
        Next
    End With
myerror:
    With Application
        .ScreenUpdating = True: .EnableEvents = True: .Calculation = xlCalculationAutomatic
    End With
End Sub


Your Sheet CommandButton Codes

Code:
Private Sub CommandButton1_Click()


    ShowRows Shipped:=False, InProgress:=True


End Sub


Private Sub CommandButton2_Click()


    ShowRows Shipped:=True, InProgress:=False


End Sub


Private Sub CommandButton3_Click()


    ShowRows Shipped:=False, InProgress:=False


End Sub


Code Not tested as on granddad duty today but hopefully, may help you

Dave
 
Upvote 0

Forum statistics

Threads
1,225,767
Messages
6,186,916
Members
453,386
Latest member
testmaster

We've detected that you are using an adblocker.

We have a great community of people providing Excel help here, but the hosting costs are enormous. You can help keep this site running by allowing ads on MrExcel.com.
Allow Ads at MrExcel

Which adblocker are you using?

Disable AdBlock

Follow these easy steps to disable AdBlock

1)Click on the icon in the browser’s toolbar.
2)Click on the icon in the browser’s toolbar.
2)Click on the "Pause on this site" option.
Go back

Disable AdBlock Plus

Follow these easy steps to disable AdBlock Plus

1)Click on the icon in the browser’s toolbar.
2)Click on the toggle to disable it for "mrexcel.com".
Go back

Disable uBlock Origin

Follow these easy steps to disable uBlock Origin

1)Click on the icon in the browser’s toolbar.
2)Click on the "Power" button.
3)Click on the "Refresh" button.
Go back

Disable uBlock

Follow these easy steps to disable uBlock

1)Click on the icon in the browser’s toolbar.
2)Click on the "Power" button.
3)Click on the "Refresh" button.
Go back
Back
Top