Tab month tracker












1














The code works; the problem is in the processing. I feel like the code can further improved and I believe the answer is Arrays however, my knowledge is very limited. Here, I loop through ~ 1000 columns. Each column has a start and end date; which will span from 1 day to 20 days, averaging ~ 3-5 days for each column. Upwards of 5000 lines are moved through and it shows in the speed of return. I will be adding If statements and I feel that if I add too much more where I am at that the program will crash.



I am hoping to speed it up. I believe arrays will do this however, the only array I use in here is borrowed from SO.



Tab Month Tracker



Raw Data Columns



Tabs Example



Download:Mock Data.xlsx



'Function to return array for dates between Start Date and End Date
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd

dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function



'Sub to move raw data into predictable format
Sub Program()

Application.ScreenUpdating = False

Dim dateStartCell As Range, dateEndCell As Range, StartDate As Range, Cell As Range
Dim allDates As Collection
Dim currentDateSter As Variant
Dim currentDate As Date
Dim TestDate As Integer

Dim NextRow As Long
Dim AdvRow As Long

Dim Facility As String
Dim Unit As String
Dim TheDay As String
Dim TheUnit As String
Dim Pax As String

Dim Test1 As Boolean
Dim Test2 As Boolean

Set StartDate = Range("E2:E1000")

NextRow = 2

Sheets("Raw").Activate

'Evaluating Each Date in Range
For Each Cell In StartDate

Set dateStartCell = Range("E" & NextRow)
Set dateEndCell = Range("G" & NextRow)
Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)

Facility = Cells(NextRow, 3)
Unit = Cells(NextRow, 2)
Pax = Cells(NextRow, 12)
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
currentDate = CDate(currentDateSter)
Sheets(MonthName(Month(currentDate), True) & Year(currentDate)).Activate


AdvRow = 3
PropRow = Empty
Test1 = False
Test2 = False
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
AdvRow = AdvRow + 1
PropRow = AdvRow

TheDay = Cells(AdvRow, 1)
TheUnit = Cells(AdvRow, 2)

If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False
End If

If TheDay = TheUnit Then
Test2 = True
Else: Test2 = False
End If

Loop Until Test1 = True Or Test2 = True



Cells(PropRow, 2).Value = Unit
Cells(PropRow, 1).Value = Day(currentDate)
Cells(PropRow, 3).Value = Pax




Sheets("Raw").Activate

Next currentDateSter

NextRow = NextRow + 1
Next Cell
Application.ScreenUpdating = True
End Sub









share|improve this question









New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.




















  • The first thing that you need to do is stop Activating the Worksheets. Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset). This will speed up your code immensely.
    – TinMan
    Dec 22 at 15:53










  • Can you provide a mock workbook? It will need to have a couple of rows of mock Raw Data and a couple of matching entries on one of the monthly tabs.
    – TinMan
    Dec 22 at 15:55










  • Yeah, I have to scrub some data, Someone changed the name of the post, and it is sort of correct, but that is only part of what is happening. Essentially column b hold a organization, column E holds a start date and column G holds an end date. for each organization in b I am breaking it up from the start date to the end date 1 instance of organization for each day. The if statements I will add in, will put facilities arrayed across the rest of the month tab trackers.
    – Jon Dee
    Dec 22 at 18:11












  • [link to scrubbed file]:ufile.io/che18
    – Jon Dee
    Dec 22 at 18:39
















1














The code works; the problem is in the processing. I feel like the code can further improved and I believe the answer is Arrays however, my knowledge is very limited. Here, I loop through ~ 1000 columns. Each column has a start and end date; which will span from 1 day to 20 days, averaging ~ 3-5 days for each column. Upwards of 5000 lines are moved through and it shows in the speed of return. I will be adding If statements and I feel that if I add too much more where I am at that the program will crash.



I am hoping to speed it up. I believe arrays will do this however, the only array I use in here is borrowed from SO.



Tab Month Tracker



Raw Data Columns



Tabs Example



Download:Mock Data.xlsx



'Function to return array for dates between Start Date and End Date
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd

dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function



'Sub to move raw data into predictable format
Sub Program()

Application.ScreenUpdating = False

Dim dateStartCell As Range, dateEndCell As Range, StartDate As Range, Cell As Range
Dim allDates As Collection
Dim currentDateSter As Variant
Dim currentDate As Date
Dim TestDate As Integer

Dim NextRow As Long
Dim AdvRow As Long

Dim Facility As String
Dim Unit As String
Dim TheDay As String
Dim TheUnit As String
Dim Pax As String

Dim Test1 As Boolean
Dim Test2 As Boolean

Set StartDate = Range("E2:E1000")

NextRow = 2

Sheets("Raw").Activate

'Evaluating Each Date in Range
For Each Cell In StartDate

Set dateStartCell = Range("E" & NextRow)
Set dateEndCell = Range("G" & NextRow)
Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)

Facility = Cells(NextRow, 3)
Unit = Cells(NextRow, 2)
Pax = Cells(NextRow, 12)
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
currentDate = CDate(currentDateSter)
Sheets(MonthName(Month(currentDate), True) & Year(currentDate)).Activate


AdvRow = 3
PropRow = Empty
Test1 = False
Test2 = False
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
AdvRow = AdvRow + 1
PropRow = AdvRow

TheDay = Cells(AdvRow, 1)
TheUnit = Cells(AdvRow, 2)

If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False
End If

If TheDay = TheUnit Then
Test2 = True
Else: Test2 = False
End If

Loop Until Test1 = True Or Test2 = True



Cells(PropRow, 2).Value = Unit
Cells(PropRow, 1).Value = Day(currentDate)
Cells(PropRow, 3).Value = Pax




Sheets("Raw").Activate

Next currentDateSter

NextRow = NextRow + 1
Next Cell
Application.ScreenUpdating = True
End Sub









share|improve this question









New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.




















  • The first thing that you need to do is stop Activating the Worksheets. Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset). This will speed up your code immensely.
    – TinMan
    Dec 22 at 15:53










  • Can you provide a mock workbook? It will need to have a couple of rows of mock Raw Data and a couple of matching entries on one of the monthly tabs.
    – TinMan
    Dec 22 at 15:55










  • Yeah, I have to scrub some data, Someone changed the name of the post, and it is sort of correct, but that is only part of what is happening. Essentially column b hold a organization, column E holds a start date and column G holds an end date. for each organization in b I am breaking it up from the start date to the end date 1 instance of organization for each day. The if statements I will add in, will put facilities arrayed across the rest of the month tab trackers.
    – Jon Dee
    Dec 22 at 18:11












  • [link to scrubbed file]:ufile.io/che18
    – Jon Dee
    Dec 22 at 18:39














1












1








1







The code works; the problem is in the processing. I feel like the code can further improved and I believe the answer is Arrays however, my knowledge is very limited. Here, I loop through ~ 1000 columns. Each column has a start and end date; which will span from 1 day to 20 days, averaging ~ 3-5 days for each column. Upwards of 5000 lines are moved through and it shows in the speed of return. I will be adding If statements and I feel that if I add too much more where I am at that the program will crash.



I am hoping to speed it up. I believe arrays will do this however, the only array I use in here is borrowed from SO.



Tab Month Tracker



Raw Data Columns



Tabs Example



Download:Mock Data.xlsx



'Function to return array for dates between Start Date and End Date
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd

dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function



'Sub to move raw data into predictable format
Sub Program()

Application.ScreenUpdating = False

Dim dateStartCell As Range, dateEndCell As Range, StartDate As Range, Cell As Range
Dim allDates As Collection
Dim currentDateSter As Variant
Dim currentDate As Date
Dim TestDate As Integer

Dim NextRow As Long
Dim AdvRow As Long

Dim Facility As String
Dim Unit As String
Dim TheDay As String
Dim TheUnit As String
Dim Pax As String

Dim Test1 As Boolean
Dim Test2 As Boolean

Set StartDate = Range("E2:E1000")

NextRow = 2

Sheets("Raw").Activate

'Evaluating Each Date in Range
For Each Cell In StartDate

Set dateStartCell = Range("E" & NextRow)
Set dateEndCell = Range("G" & NextRow)
Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)

Facility = Cells(NextRow, 3)
Unit = Cells(NextRow, 2)
Pax = Cells(NextRow, 12)
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
currentDate = CDate(currentDateSter)
Sheets(MonthName(Month(currentDate), True) & Year(currentDate)).Activate


AdvRow = 3
PropRow = Empty
Test1 = False
Test2 = False
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
AdvRow = AdvRow + 1
PropRow = AdvRow

TheDay = Cells(AdvRow, 1)
TheUnit = Cells(AdvRow, 2)

If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False
End If

If TheDay = TheUnit Then
Test2 = True
Else: Test2 = False
End If

Loop Until Test1 = True Or Test2 = True



Cells(PropRow, 2).Value = Unit
Cells(PropRow, 1).Value = Day(currentDate)
Cells(PropRow, 3).Value = Pax




Sheets("Raw").Activate

Next currentDateSter

NextRow = NextRow + 1
Next Cell
Application.ScreenUpdating = True
End Sub









share|improve this question









New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.











The code works; the problem is in the processing. I feel like the code can further improved and I believe the answer is Arrays however, my knowledge is very limited. Here, I loop through ~ 1000 columns. Each column has a start and end date; which will span from 1 day to 20 days, averaging ~ 3-5 days for each column. Upwards of 5000 lines are moved through and it shows in the speed of return. I will be adding If statements and I feel that if I add too much more where I am at that the program will crash.



I am hoping to speed it up. I believe arrays will do this however, the only array I use in here is borrowed from SO.



Tab Month Tracker



Raw Data Columns



Tabs Example



Download:Mock Data.xlsx



'Function to return array for dates between Start Date and End Date
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd

dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function



'Sub to move raw data into predictable format
Sub Program()

Application.ScreenUpdating = False

Dim dateStartCell As Range, dateEndCell As Range, StartDate As Range, Cell As Range
Dim allDates As Collection
Dim currentDateSter As Variant
Dim currentDate As Date
Dim TestDate As Integer

Dim NextRow As Long
Dim AdvRow As Long

Dim Facility As String
Dim Unit As String
Dim TheDay As String
Dim TheUnit As String
Dim Pax As String

Dim Test1 As Boolean
Dim Test2 As Boolean

Set StartDate = Range("E2:E1000")

NextRow = 2

Sheets("Raw").Activate

'Evaluating Each Date in Range
For Each Cell In StartDate

Set dateStartCell = Range("E" & NextRow)
Set dateEndCell = Range("G" & NextRow)
Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)

Facility = Cells(NextRow, 3)
Unit = Cells(NextRow, 2)
Pax = Cells(NextRow, 12)
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
currentDate = CDate(currentDateSter)
Sheets(MonthName(Month(currentDate), True) & Year(currentDate)).Activate


AdvRow = 3
PropRow = Empty
Test1 = False
Test2 = False
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
AdvRow = AdvRow + 1
PropRow = AdvRow

TheDay = Cells(AdvRow, 1)
TheUnit = Cells(AdvRow, 2)

If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False
End If

If TheDay = TheUnit Then
Test2 = True
Else: Test2 = False
End If

Loop Until Test1 = True Or Test2 = True



Cells(PropRow, 2).Value = Unit
Cells(PropRow, 1).Value = Day(currentDate)
Cells(PropRow, 3).Value = Pax




Sheets("Raw").Activate

Next currentDateSter

NextRow = NextRow + 1
Next Cell
Application.ScreenUpdating = True
End Sub






performance vba excel






share|improve this question









New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.











share|improve this question









New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.









share|improve this question




share|improve this question








edited Dec 22 at 19:24









TinMan

1,054110




1,054110






New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.









asked Dec 21 at 14:05









Jon Dee

83




83




New contributor




Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.





New contributor





Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.






Jon Dee is a new contributor to this site. Take care in asking for clarification, commenting, and answering.
Check out our Code of Conduct.












  • The first thing that you need to do is stop Activating the Worksheets. Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset). This will speed up your code immensely.
    – TinMan
    Dec 22 at 15:53










  • Can you provide a mock workbook? It will need to have a couple of rows of mock Raw Data and a couple of matching entries on one of the monthly tabs.
    – TinMan
    Dec 22 at 15:55










  • Yeah, I have to scrub some data, Someone changed the name of the post, and it is sort of correct, but that is only part of what is happening. Essentially column b hold a organization, column E holds a start date and column G holds an end date. for each organization in b I am breaking it up from the start date to the end date 1 instance of organization for each day. The if statements I will add in, will put facilities arrayed across the rest of the month tab trackers.
    – Jon Dee
    Dec 22 at 18:11












  • [link to scrubbed file]:ufile.io/che18
    – Jon Dee
    Dec 22 at 18:39


















  • The first thing that you need to do is stop Activating the Worksheets. Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset). This will speed up your code immensely.
    – TinMan
    Dec 22 at 15:53










  • Can you provide a mock workbook? It will need to have a couple of rows of mock Raw Data and a couple of matching entries on one of the monthly tabs.
    – TinMan
    Dec 22 at 15:55










  • Yeah, I have to scrub some data, Someone changed the name of the post, and it is sort of correct, but that is only part of what is happening. Essentially column b hold a organization, column E holds a start date and column G holds an end date. for each organization in b I am breaking it up from the start date to the end date 1 instance of organization for each day. The if statements I will add in, will put facilities arrayed across the rest of the month tab trackers.
    – Jon Dee
    Dec 22 at 18:11












  • [link to scrubbed file]:ufile.io/che18
    – Jon Dee
    Dec 22 at 18:39
















The first thing that you need to do is stop Activating the Worksheets. Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset). This will speed up your code immensely.
– TinMan
Dec 22 at 15:53




The first thing that you need to do is stop Activating the Worksheets. Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset). This will speed up your code immensely.
– TinMan
Dec 22 at 15:53












Can you provide a mock workbook? It will need to have a couple of rows of mock Raw Data and a couple of matching entries on one of the monthly tabs.
– TinMan
Dec 22 at 15:55




Can you provide a mock workbook? It will need to have a couple of rows of mock Raw Data and a couple of matching entries on one of the monthly tabs.
– TinMan
Dec 22 at 15:55












Yeah, I have to scrub some data, Someone changed the name of the post, and it is sort of correct, but that is only part of what is happening. Essentially column b hold a organization, column E holds a start date and column G holds an end date. for each organization in b I am breaking it up from the start date to the end date 1 instance of organization for each day. The if statements I will add in, will put facilities arrayed across the rest of the month tab trackers.
– Jon Dee
Dec 22 at 18:11






Yeah, I have to scrub some data, Someone changed the name of the post, and it is sort of correct, but that is only part of what is happening. Essentially column b hold a organization, column E holds a start date and column G holds an end date. for each organization in b I am breaking it up from the start date to the end date 1 instance of organization for each day. The if statements I will add in, will put facilities arrayed across the rest of the month tab trackers.
– Jon Dee
Dec 22 at 18:11














[link to scrubbed file]:ufile.io/che18
– Jon Dee
Dec 22 at 18:39




[link to scrubbed file]:ufile.io/che18
– Jon Dee
Dec 22 at 18:39










2 Answers
2






active

oldest

votes


















0














Editor Options



The first thing that I would recommend is adjusting your VBEditor options.



Checking Require Variable Declaration will automatically put Option Explicit at the top of newly created code modules. This makes it easier to clean up code as you modify it and catch undeclared variables, such as, PropRow.



Unchecking Auto Syntax Check will prevent the Syntax Error MsgBox from appearing will you are writing your code. You will still know that there is a syntax error because the text is red but you will not have to stop to click the message.



VBE Options



Download Rubberduck VBA: UserForm1.Show and use it's code formatting tool. This tool will not only save a ton of time in formatting but will help catch unclosed blocks of code.



Data Typing



Using the correct data type is crucial to writing solid code. It will prevent unintended bugs from creeping in and improve the overall performance of the code. TheDay should be typed as Long because it will always be an Integer. Note: There is no advantage to using a smaller data type, such as: Byte or Integer. It looks like TheUnit should probably be long also bit that might because of the dummy data.



Dynamic Ranges



Using Dynamic Ranges Range("E2", Range("E" & Rows.Count).End(xlUp)) over staatic ranges Set StartDate = Range("E2:E1000") will prevent you from having to update the code as rows are added and optimize the code as the rows are deleted.



Loops



If you are going to iterate over each cells in the range then you should use the Cell object. Resolving the Cell is not free. It is causing the CPU to do extra work.




For Each Cell In startDate



Here is how you should use this loop:




    Set dateStartCell = Cell.Offset(0, 4).Value
Set dateEndCell = Cell.Offset(0, 6).Value



Otherwise just use a standard For Loop.




For r  = 2 to Range("E" & Rows.Count).End(xlUp).Row



In many cases it makes sense to have another function return a collection and iterate over it. After all, the fewer tasks that a subroutine performs the easier it is to test. This is not one of those cases.




For Each currentDateSter In allDates



Basically, all the collection is used for is to start an iteration at the start date and add 1 to until you reach the end date. Not only can this be accomplished a lot cheaper by using a standard For Loop but it makes the more condense and easier to read.




For dateOf = dateStartCell.Value to dateEndCell.Value



Selecting and Activating



It is rarely necessary to Select or Activate an Object. It is much better to fully qualify your Objects and refer to them directly. This is the biggest slow down in your code.



Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset)



If Statements



I prefer to make direct boolean assignments over the bulkier If blocks.




Test1 = TheDay = Day(currentDate) And TheUnit = Unit
Test2 = TheDay = TheUnit



Test2 is misleading. Its true function is to test whether or not Cells(AdvRow, 2) is empty.



Test1 and Test2 are not very descriptive names. I would prefer dataMatched and emtpyRow but would have eliminated both variables by using the code below.




Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""



Raw Data: Deleted Rows



Deleted rows in the Raw Data will not reflect in the monthly reports. This could lead to big problems and should be addressed.



Refactored Code



This code ran 95% faster the the original. The code could further be improved by using arrays for each month's data but that is way outside the scope of this website.



Sub Program2()
Dim t As Double: t = Timer
Application.ScreenUpdating = False
Application.Calculation = xlCalculationManual

Dim data As Variant
With Worksheets("Raw") 'Load the data into an Array
data = .Range("A2:N2", .Cells(.Rows.Count, "E").End(xlUp)).Value
End With

Dim dateOf As Date
Dim r1 As Long

For r1 = 1 To UBound(data)
For dateOf = data(r1, 5) To data(r1, 7)
Dim wsMonth As Worksheet, wsName As String

If wsName <> Format(dateOf, "mmmyyyy") Then
wsName = Format(dateOf, "mmmyyyy")
Set wsMonth = Worksheets(wsName)
End If

With wsMonth
Dim r2 As Long
For r2 = 4 To .Cells(.Rows.Count, "A").End(xlUp).Row + 1
Dim TheDay As Long
Dim TheUnit As Long
Dim Pax As String
TheDay = Day(dateOf)
TheUnit = data(r1, 2)
Pax = data(r1, 12)
If (.Cells(r2, 1).Value = TheDay And .Cells(r2, 2).Value = TheUnit) Then
.Cells(r2, 3).Value = Pax
Exit For
ElseIf .Cells(r2, "A").Value = "" Then
.Cells(r2, 1).Value = TheDay
.Cells(r2, 2).Value = TheUnit
.Cells(r2, 3).Value = Pax
Exit For
End If
Next
End With
Next
Next
Application.Calculation = xlCalculationAutomatic
Application.ScreenUpdating = True
Debug.Print Round(Timer - t, 2)
End Sub


Addendum



In order to speed up the code I would use arrays to write the data to each month in one operation and dictionaries because of their lightning fast look-up speed. These references will help:




  • Excel VBA Introduction Part 39 - Dictionaries

  • Excel VBA Introduction Part 25 - Arrays






share|improve this answer























  • Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
    – Jon Dee
    Dec 23 at 14:31










  • @JonDee Thanks. I'll edit it later.
    – TinMan
    Dec 23 at 14:35










  • LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
    – Jon Dee
    Dec 23 at 14:41










  • I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
    – Jon Dee
    Dec 23 at 14:49










  • I cannot up vote your post as I am a NEWB, not enough points.
    – Jon Dee
    Dec 23 at 14:50



















0














There are many areas of improvement in this coding.



A most important part of code hygiene is proper indenting, and always use Option Explicit.



Also, name your variables to something meaningful. For example, you use StartDate, but it is not a date (which the name implies), but a range.



You comment that your first function returns an Array, but it actually returns a Collection. Arrays are ordered, Collections are not, particularly in VBA.



You have some Boolean anti-patterns happening:



If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False ' No need to run things onto a single line, especially if this is inconsistent with the other code.
End If


Can simply be:



Test1 = (TheDay = Day(currentDate)) And (TheUnit = Unit ) ' perhaps "FindMatch" is better descriptive name.


You set AdvRow and PropRow (what are these anyway - proper naming?) relative to each other within the loop, but you don't change either in that loop - so a single variable (AdvRow) will suffice.



You don't error check to ensure that the data you are reading is the right form - what happens if the data sheet does not exist, or that cell that is read is not a date?



You use NextRow while in a loop - but you already access a cell in the loop that tells you what the row is. This is one variable that can be dropped. And you are using NextRow as the CurrentRow - this is another example of a confusing variable name.



A big performance hit will come from having three nested loops, but also accessing each cell individually within those loops. Each time you make the program switch from looking at the VBA to looking at the Excel ranges is a cost in performance - this is why taking a range and putting it into an array improves efficiency.



'Function to return Collection of dates between Start Date and End Date
'**** You don't check to see if Start comes before End - what does it mean if they are the wrong way round?
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd
dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function

'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With

'will be a more efficient way of setting the array, but this will do for now
' 0..998, 0..4 array - datestart, dateend, facility, unit, pax

'Removes the following code:
'Set StartDate = Range("E2:E1000")
'NextRow = 2
'Sheets("Raw").Activate
'For Each Cell In StartDate
'Set dateStartCell = Range("E" & NextRow)
'Set dateEndCell = Range("G" & NextRow)

For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
'Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))

'Following code is no longer necessary
' Facility = Cells(NextRow, 3)
' Unit = Cells(NextRow, 2)
' Pax = Cells(NextRow, 12)

'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
'PropRow = AdvRow

theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Removing all my additional comments in Program gives you:



'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With
For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Of course, there may be some other logic paths, or even, perhaps, using Excel native functions that could help refine the problem.



I haven't been able to test the code (naturally), but it does compile in VBA.






share|improve this answer























  • rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
    – Jon Dee
    Dec 23 at 14:57










  • startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
    – Jon Dee
    Dec 23 at 14:59












  • I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
    – AJD
    Dec 23 at 21:02













Your Answer





StackExchange.ifUsing("editor", function () {
return StackExchange.using("mathjaxEditing", function () {
StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix) {
StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["\$", "\$"]]);
});
});
}, "mathjax-editing");

StackExchange.ifUsing("editor", function () {
StackExchange.using("externalEditor", function () {
StackExchange.using("snippets", function () {
StackExchange.snippets.init();
});
});
}, "code-snippets");

StackExchange.ready(function() {
var channelOptions = {
tags: "".split(" "),
id: "196"
};
initTagRenderer("".split(" "), "".split(" "), channelOptions);

StackExchange.using("externalEditor", function() {
// Have to fire editor after snippets, if snippets enabled
if (StackExchange.settings.snippets.snippetsEnabled) {
StackExchange.using("snippets", function() {
createEditor();
});
}
else {
createEditor();
}
});

function createEditor() {
StackExchange.prepareEditor({
heartbeatType: 'answer',
autoActivateHeartbeat: false,
convertImagesToLinks: false,
noModals: true,
showLowRepImageUploadWarning: true,
reputationToPostImages: null,
bindNavPrevention: true,
postfix: "",
imageUploader: {
brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
allowUrls: true
},
onDemand: true,
discardSelector: ".discard-answer"
,immediatelyShowMarkdownHelp:true
});


}
});






Jon Dee is a new contributor. Be nice, and check out our Code of Conduct.










draft saved

draft discarded


















StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f210120%2ftab-month-tracker%23new-answer', 'question_page');
}
);

Post as a guest















Required, but never shown

























2 Answers
2






active

oldest

votes








2 Answers
2






active

oldest

votes









active

oldest

votes






active

oldest

votes









0














Editor Options



The first thing that I would recommend is adjusting your VBEditor options.



Checking Require Variable Declaration will automatically put Option Explicit at the top of newly created code modules. This makes it easier to clean up code as you modify it and catch undeclared variables, such as, PropRow.



Unchecking Auto Syntax Check will prevent the Syntax Error MsgBox from appearing will you are writing your code. You will still know that there is a syntax error because the text is red but you will not have to stop to click the message.



VBE Options



Download Rubberduck VBA: UserForm1.Show and use it's code formatting tool. This tool will not only save a ton of time in formatting but will help catch unclosed blocks of code.



Data Typing



Using the correct data type is crucial to writing solid code. It will prevent unintended bugs from creeping in and improve the overall performance of the code. TheDay should be typed as Long because it will always be an Integer. Note: There is no advantage to using a smaller data type, such as: Byte or Integer. It looks like TheUnit should probably be long also bit that might because of the dummy data.



Dynamic Ranges



Using Dynamic Ranges Range("E2", Range("E" & Rows.Count).End(xlUp)) over staatic ranges Set StartDate = Range("E2:E1000") will prevent you from having to update the code as rows are added and optimize the code as the rows are deleted.



Loops



If you are going to iterate over each cells in the range then you should use the Cell object. Resolving the Cell is not free. It is causing the CPU to do extra work.




For Each Cell In startDate



Here is how you should use this loop:




    Set dateStartCell = Cell.Offset(0, 4).Value
Set dateEndCell = Cell.Offset(0, 6).Value



Otherwise just use a standard For Loop.




For r  = 2 to Range("E" & Rows.Count).End(xlUp).Row



In many cases it makes sense to have another function return a collection and iterate over it. After all, the fewer tasks that a subroutine performs the easier it is to test. This is not one of those cases.




For Each currentDateSter In allDates



Basically, all the collection is used for is to start an iteration at the start date and add 1 to until you reach the end date. Not only can this be accomplished a lot cheaper by using a standard For Loop but it makes the more condense and easier to read.




For dateOf = dateStartCell.Value to dateEndCell.Value



Selecting and Activating



It is rarely necessary to Select or Activate an Object. It is much better to fully qualify your Objects and refer to them directly. This is the biggest slow down in your code.



Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset)



If Statements



I prefer to make direct boolean assignments over the bulkier If blocks.




Test1 = TheDay = Day(currentDate) And TheUnit = Unit
Test2 = TheDay = TheUnit



Test2 is misleading. Its true function is to test whether or not Cells(AdvRow, 2) is empty.



Test1 and Test2 are not very descriptive names. I would prefer dataMatched and emtpyRow but would have eliminated both variables by using the code below.




Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""



Raw Data: Deleted Rows



Deleted rows in the Raw Data will not reflect in the monthly reports. This could lead to big problems and should be addressed.



Refactored Code



This code ran 95% faster the the original. The code could further be improved by using arrays for each month's data but that is way outside the scope of this website.



Sub Program2()
Dim t As Double: t = Timer
Application.ScreenUpdating = False
Application.Calculation = xlCalculationManual

Dim data As Variant
With Worksheets("Raw") 'Load the data into an Array
data = .Range("A2:N2", .Cells(.Rows.Count, "E").End(xlUp)).Value
End With

Dim dateOf As Date
Dim r1 As Long

For r1 = 1 To UBound(data)
For dateOf = data(r1, 5) To data(r1, 7)
Dim wsMonth As Worksheet, wsName As String

If wsName <> Format(dateOf, "mmmyyyy") Then
wsName = Format(dateOf, "mmmyyyy")
Set wsMonth = Worksheets(wsName)
End If

With wsMonth
Dim r2 As Long
For r2 = 4 To .Cells(.Rows.Count, "A").End(xlUp).Row + 1
Dim TheDay As Long
Dim TheUnit As Long
Dim Pax As String
TheDay = Day(dateOf)
TheUnit = data(r1, 2)
Pax = data(r1, 12)
If (.Cells(r2, 1).Value = TheDay And .Cells(r2, 2).Value = TheUnit) Then
.Cells(r2, 3).Value = Pax
Exit For
ElseIf .Cells(r2, "A").Value = "" Then
.Cells(r2, 1).Value = TheDay
.Cells(r2, 2).Value = TheUnit
.Cells(r2, 3).Value = Pax
Exit For
End If
Next
End With
Next
Next
Application.Calculation = xlCalculationAutomatic
Application.ScreenUpdating = True
Debug.Print Round(Timer - t, 2)
End Sub


Addendum



In order to speed up the code I would use arrays to write the data to each month in one operation and dictionaries because of their lightning fast look-up speed. These references will help:




  • Excel VBA Introduction Part 39 - Dictionaries

  • Excel VBA Introduction Part 25 - Arrays






share|improve this answer























  • Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
    – Jon Dee
    Dec 23 at 14:31










  • @JonDee Thanks. I'll edit it later.
    – TinMan
    Dec 23 at 14:35










  • LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
    – Jon Dee
    Dec 23 at 14:41










  • I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
    – Jon Dee
    Dec 23 at 14:49










  • I cannot up vote your post as I am a NEWB, not enough points.
    – Jon Dee
    Dec 23 at 14:50
















0














Editor Options



The first thing that I would recommend is adjusting your VBEditor options.



Checking Require Variable Declaration will automatically put Option Explicit at the top of newly created code modules. This makes it easier to clean up code as you modify it and catch undeclared variables, such as, PropRow.



Unchecking Auto Syntax Check will prevent the Syntax Error MsgBox from appearing will you are writing your code. You will still know that there is a syntax error because the text is red but you will not have to stop to click the message.



VBE Options



Download Rubberduck VBA: UserForm1.Show and use it's code formatting tool. This tool will not only save a ton of time in formatting but will help catch unclosed blocks of code.



Data Typing



Using the correct data type is crucial to writing solid code. It will prevent unintended bugs from creeping in and improve the overall performance of the code. TheDay should be typed as Long because it will always be an Integer. Note: There is no advantage to using a smaller data type, such as: Byte or Integer. It looks like TheUnit should probably be long also bit that might because of the dummy data.



Dynamic Ranges



Using Dynamic Ranges Range("E2", Range("E" & Rows.Count).End(xlUp)) over staatic ranges Set StartDate = Range("E2:E1000") will prevent you from having to update the code as rows are added and optimize the code as the rows are deleted.



Loops



If you are going to iterate over each cells in the range then you should use the Cell object. Resolving the Cell is not free. It is causing the CPU to do extra work.




For Each Cell In startDate



Here is how you should use this loop:




    Set dateStartCell = Cell.Offset(0, 4).Value
Set dateEndCell = Cell.Offset(0, 6).Value



Otherwise just use a standard For Loop.




For r  = 2 to Range("E" & Rows.Count).End(xlUp).Row



In many cases it makes sense to have another function return a collection and iterate over it. After all, the fewer tasks that a subroutine performs the easier it is to test. This is not one of those cases.




For Each currentDateSter In allDates



Basically, all the collection is used for is to start an iteration at the start date and add 1 to until you reach the end date. Not only can this be accomplished a lot cheaper by using a standard For Loop but it makes the more condense and easier to read.




For dateOf = dateStartCell.Value to dateEndCell.Value



Selecting and Activating



It is rarely necessary to Select or Activate an Object. It is much better to fully qualify your Objects and refer to them directly. This is the biggest slow down in your code.



Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset)



If Statements



I prefer to make direct boolean assignments over the bulkier If blocks.




Test1 = TheDay = Day(currentDate) And TheUnit = Unit
Test2 = TheDay = TheUnit



Test2 is misleading. Its true function is to test whether or not Cells(AdvRow, 2) is empty.



Test1 and Test2 are not very descriptive names. I would prefer dataMatched and emtpyRow but would have eliminated both variables by using the code below.




Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""



Raw Data: Deleted Rows



Deleted rows in the Raw Data will not reflect in the monthly reports. This could lead to big problems and should be addressed.



Refactored Code



This code ran 95% faster the the original. The code could further be improved by using arrays for each month's data but that is way outside the scope of this website.



Sub Program2()
Dim t As Double: t = Timer
Application.ScreenUpdating = False
Application.Calculation = xlCalculationManual

Dim data As Variant
With Worksheets("Raw") 'Load the data into an Array
data = .Range("A2:N2", .Cells(.Rows.Count, "E").End(xlUp)).Value
End With

Dim dateOf As Date
Dim r1 As Long

For r1 = 1 To UBound(data)
For dateOf = data(r1, 5) To data(r1, 7)
Dim wsMonth As Worksheet, wsName As String

If wsName <> Format(dateOf, "mmmyyyy") Then
wsName = Format(dateOf, "mmmyyyy")
Set wsMonth = Worksheets(wsName)
End If

With wsMonth
Dim r2 As Long
For r2 = 4 To .Cells(.Rows.Count, "A").End(xlUp).Row + 1
Dim TheDay As Long
Dim TheUnit As Long
Dim Pax As String
TheDay = Day(dateOf)
TheUnit = data(r1, 2)
Pax = data(r1, 12)
If (.Cells(r2, 1).Value = TheDay And .Cells(r2, 2).Value = TheUnit) Then
.Cells(r2, 3).Value = Pax
Exit For
ElseIf .Cells(r2, "A").Value = "" Then
.Cells(r2, 1).Value = TheDay
.Cells(r2, 2).Value = TheUnit
.Cells(r2, 3).Value = Pax
Exit For
End If
Next
End With
Next
Next
Application.Calculation = xlCalculationAutomatic
Application.ScreenUpdating = True
Debug.Print Round(Timer - t, 2)
End Sub


Addendum



In order to speed up the code I would use arrays to write the data to each month in one operation and dictionaries because of their lightning fast look-up speed. These references will help:




  • Excel VBA Introduction Part 39 - Dictionaries

  • Excel VBA Introduction Part 25 - Arrays






share|improve this answer























  • Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
    – Jon Dee
    Dec 23 at 14:31










  • @JonDee Thanks. I'll edit it later.
    – TinMan
    Dec 23 at 14:35










  • LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
    – Jon Dee
    Dec 23 at 14:41










  • I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
    – Jon Dee
    Dec 23 at 14:49










  • I cannot up vote your post as I am a NEWB, not enough points.
    – Jon Dee
    Dec 23 at 14:50














0












0








0






Editor Options



The first thing that I would recommend is adjusting your VBEditor options.



Checking Require Variable Declaration will automatically put Option Explicit at the top of newly created code modules. This makes it easier to clean up code as you modify it and catch undeclared variables, such as, PropRow.



Unchecking Auto Syntax Check will prevent the Syntax Error MsgBox from appearing will you are writing your code. You will still know that there is a syntax error because the text is red but you will not have to stop to click the message.



VBE Options



Download Rubberduck VBA: UserForm1.Show and use it's code formatting tool. This tool will not only save a ton of time in formatting but will help catch unclosed blocks of code.



Data Typing



Using the correct data type is crucial to writing solid code. It will prevent unintended bugs from creeping in and improve the overall performance of the code. TheDay should be typed as Long because it will always be an Integer. Note: There is no advantage to using a smaller data type, such as: Byte or Integer. It looks like TheUnit should probably be long also bit that might because of the dummy data.



Dynamic Ranges



Using Dynamic Ranges Range("E2", Range("E" & Rows.Count).End(xlUp)) over staatic ranges Set StartDate = Range("E2:E1000") will prevent you from having to update the code as rows are added and optimize the code as the rows are deleted.



Loops



If you are going to iterate over each cells in the range then you should use the Cell object. Resolving the Cell is not free. It is causing the CPU to do extra work.




For Each Cell In startDate



Here is how you should use this loop:




    Set dateStartCell = Cell.Offset(0, 4).Value
Set dateEndCell = Cell.Offset(0, 6).Value



Otherwise just use a standard For Loop.




For r  = 2 to Range("E" & Rows.Count).End(xlUp).Row



In many cases it makes sense to have another function return a collection and iterate over it. After all, the fewer tasks that a subroutine performs the easier it is to test. This is not one of those cases.




For Each currentDateSter In allDates



Basically, all the collection is used for is to start an iteration at the start date and add 1 to until you reach the end date. Not only can this be accomplished a lot cheaper by using a standard For Loop but it makes the more condense and easier to read.




For dateOf = dateStartCell.Value to dateEndCell.Value



Selecting and Activating



It is rarely necessary to Select or Activate an Object. It is much better to fully qualify your Objects and refer to them directly. This is the biggest slow down in your code.



Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset)



If Statements



I prefer to make direct boolean assignments over the bulkier If blocks.




Test1 = TheDay = Day(currentDate) And TheUnit = Unit
Test2 = TheDay = TheUnit



Test2 is misleading. Its true function is to test whether or not Cells(AdvRow, 2) is empty.



Test1 and Test2 are not very descriptive names. I would prefer dataMatched and emtpyRow but would have eliminated both variables by using the code below.




Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""



Raw Data: Deleted Rows



Deleted rows in the Raw Data will not reflect in the monthly reports. This could lead to big problems and should be addressed.



Refactored Code



This code ran 95% faster the the original. The code could further be improved by using arrays for each month's data but that is way outside the scope of this website.



Sub Program2()
Dim t As Double: t = Timer
Application.ScreenUpdating = False
Application.Calculation = xlCalculationManual

Dim data As Variant
With Worksheets("Raw") 'Load the data into an Array
data = .Range("A2:N2", .Cells(.Rows.Count, "E").End(xlUp)).Value
End With

Dim dateOf As Date
Dim r1 As Long

For r1 = 1 To UBound(data)
For dateOf = data(r1, 5) To data(r1, 7)
Dim wsMonth As Worksheet, wsName As String

If wsName <> Format(dateOf, "mmmyyyy") Then
wsName = Format(dateOf, "mmmyyyy")
Set wsMonth = Worksheets(wsName)
End If

With wsMonth
Dim r2 As Long
For r2 = 4 To .Cells(.Rows.Count, "A").End(xlUp).Row + 1
Dim TheDay As Long
Dim TheUnit As Long
Dim Pax As String
TheDay = Day(dateOf)
TheUnit = data(r1, 2)
Pax = data(r1, 12)
If (.Cells(r2, 1).Value = TheDay And .Cells(r2, 2).Value = TheUnit) Then
.Cells(r2, 3).Value = Pax
Exit For
ElseIf .Cells(r2, "A").Value = "" Then
.Cells(r2, 1).Value = TheDay
.Cells(r2, 2).Value = TheUnit
.Cells(r2, 3).Value = Pax
Exit For
End If
Next
End With
Next
Next
Application.Calculation = xlCalculationAutomatic
Application.ScreenUpdating = True
Debug.Print Round(Timer - t, 2)
End Sub


Addendum



In order to speed up the code I would use arrays to write the data to each month in one operation and dictionaries because of their lightning fast look-up speed. These references will help:




  • Excel VBA Introduction Part 39 - Dictionaries

  • Excel VBA Introduction Part 25 - Arrays






share|improve this answer














Editor Options



The first thing that I would recommend is adjusting your VBEditor options.



Checking Require Variable Declaration will automatically put Option Explicit at the top of newly created code modules. This makes it easier to clean up code as you modify it and catch undeclared variables, such as, PropRow.



Unchecking Auto Syntax Check will prevent the Syntax Error MsgBox from appearing will you are writing your code. You will still know that there is a syntax error because the text is red but you will not have to stop to click the message.



VBE Options



Download Rubberduck VBA: UserForm1.Show and use it's code formatting tool. This tool will not only save a ton of time in formatting but will help catch unclosed blocks of code.



Data Typing



Using the correct data type is crucial to writing solid code. It will prevent unintended bugs from creeping in and improve the overall performance of the code. TheDay should be typed as Long because it will always be an Integer. Note: There is no advantage to using a smaller data type, such as: Byte or Integer. It looks like TheUnit should probably be long also bit that might because of the dummy data.



Dynamic Ranges



Using Dynamic Ranges Range("E2", Range("E" & Rows.Count).End(xlUp)) over staatic ranges Set StartDate = Range("E2:E1000") will prevent you from having to update the code as rows are added and optimize the code as the rows are deleted.



Loops



If you are going to iterate over each cells in the range then you should use the Cell object. Resolving the Cell is not free. It is causing the CPU to do extra work.




For Each Cell In startDate



Here is how you should use this loop:




    Set dateStartCell = Cell.Offset(0, 4).Value
Set dateEndCell = Cell.Offset(0, 6).Value



Otherwise just use a standard For Loop.




For r  = 2 to Range("E" & Rows.Count).End(xlUp).Row



In many cases it makes sense to have another function return a collection and iterate over it. After all, the fewer tasks that a subroutine performs the easier it is to test. This is not one of those cases.




For Each currentDateSter In allDates



Basically, all the collection is used for is to start an iteration at the start date and add 1 to until you reach the end date. Not only can this be accomplished a lot cheaper by using a standard For Loop but it makes the more condense and easier to read.




For dateOf = dateStartCell.Value to dateEndCell.Value



Selecting and Activating



It is rarely necessary to Select or Activate an Object. It is much better to fully qualify your Objects and refer to them directly. This is the biggest slow down in your code.



Watch: Excel VBA Introduction Part 5 - Selecting Cells (Range, Cells, Activecell, End, Offset)



If Statements



I prefer to make direct boolean assignments over the bulkier If blocks.




Test1 = TheDay = Day(currentDate) And TheUnit = Unit
Test2 = TheDay = TheUnit



Test2 is misleading. Its true function is to test whether or not Cells(AdvRow, 2) is empty.



Test1 and Test2 are not very descriptive names. I would prefer dataMatched and emtpyRow but would have eliminated both variables by using the code below.




Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""



Raw Data: Deleted Rows



Deleted rows in the Raw Data will not reflect in the monthly reports. This could lead to big problems and should be addressed.



Refactored Code



This code ran 95% faster the the original. The code could further be improved by using arrays for each month's data but that is way outside the scope of this website.



Sub Program2()
Dim t As Double: t = Timer
Application.ScreenUpdating = False
Application.Calculation = xlCalculationManual

Dim data As Variant
With Worksheets("Raw") 'Load the data into an Array
data = .Range("A2:N2", .Cells(.Rows.Count, "E").End(xlUp)).Value
End With

Dim dateOf As Date
Dim r1 As Long

For r1 = 1 To UBound(data)
For dateOf = data(r1, 5) To data(r1, 7)
Dim wsMonth As Worksheet, wsName As String

If wsName <> Format(dateOf, "mmmyyyy") Then
wsName = Format(dateOf, "mmmyyyy")
Set wsMonth = Worksheets(wsName)
End If

With wsMonth
Dim r2 As Long
For r2 = 4 To .Cells(.Rows.Count, "A").End(xlUp).Row + 1
Dim TheDay As Long
Dim TheUnit As Long
Dim Pax As String
TheDay = Day(dateOf)
TheUnit = data(r1, 2)
Pax = data(r1, 12)
If (.Cells(r2, 1).Value = TheDay And .Cells(r2, 2).Value = TheUnit) Then
.Cells(r2, 3).Value = Pax
Exit For
ElseIf .Cells(r2, "A").Value = "" Then
.Cells(r2, 1).Value = TheDay
.Cells(r2, 2).Value = TheUnit
.Cells(r2, 3).Value = Pax
Exit For
End If
Next
End With
Next
Next
Application.Calculation = xlCalculationAutomatic
Application.ScreenUpdating = True
Debug.Print Round(Timer - t, 2)
End Sub


Addendum



In order to speed up the code I would use arrays to write the data to each month in one operation and dictionaries because of their lightning fast look-up speed. These references will help:




  • Excel VBA Introduction Part 39 - Dictionaries

  • Excel VBA Introduction Part 25 - Arrays







share|improve this answer














share|improve this answer



share|improve this answer








edited Dec 23 at 15:49

























answered Dec 23 at 12:26









TinMan

1,054110




1,054110












  • Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
    – Jon Dee
    Dec 23 at 14:31










  • @JonDee Thanks. I'll edit it later.
    – TinMan
    Dec 23 at 14:35










  • LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
    – Jon Dee
    Dec 23 at 14:41










  • I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
    – Jon Dee
    Dec 23 at 14:49










  • I cannot up vote your post as I am a NEWB, not enough points.
    – Jon Dee
    Dec 23 at 14:50


















  • Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
    – Jon Dee
    Dec 23 at 14:31










  • @JonDee Thanks. I'll edit it later.
    – TinMan
    Dec 23 at 14:35










  • LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
    – Jon Dee
    Dec 23 at 14:41










  • I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
    – Jon Dee
    Dec 23 at 14:49










  • I cannot up vote your post as I am a NEWB, not enough points.
    – Jon Dee
    Dec 23 at 14:50
















Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
– Jon Dee
Dec 23 at 14:31




Loop Until (TheDay = Day(currentDate) And TheUnit = Unit) Or Cells(AdvRow, 2) = ""
– Jon Dee
Dec 23 at 14:31












@JonDee Thanks. I'll edit it later.
– TinMan
Dec 23 at 14:35




@JonDee Thanks. I'll edit it later.
– TinMan
Dec 23 at 14:35












LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
– Jon Dee
Dec 23 at 14:41




LOL, sorry, Newb, hit enter and was going to go into edit. I was meaning to say thank you for you time, but most importantly, I especially appreciate the explanation of the thought process and the references to assist. I only started playing with VBA a couple of weeks ago, so the language and syntax rules I am still trying to learn as I put together this project that will save me tons of time!
– Jon Dee
Dec 23 at 14:41












I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
– Jon Dee
Dec 23 at 14:49




I did have to change theunit to a string due to error 13 (which I am becoming very familiar with!), When I data cleansed, I went for simple versus helpful, my apologies. On arrays would you have a suggestion of any good references? Also, this code ran in 1/4 of the time I had. Next is to move it to the computer it will run on, that is where the biggest issues is.
– Jon Dee
Dec 23 at 14:49












I cannot up vote your post as I am a NEWB, not enough points.
– Jon Dee
Dec 23 at 14:50




I cannot up vote your post as I am a NEWB, not enough points.
– Jon Dee
Dec 23 at 14:50













0














There are many areas of improvement in this coding.



A most important part of code hygiene is proper indenting, and always use Option Explicit.



Also, name your variables to something meaningful. For example, you use StartDate, but it is not a date (which the name implies), but a range.



You comment that your first function returns an Array, but it actually returns a Collection. Arrays are ordered, Collections are not, particularly in VBA.



You have some Boolean anti-patterns happening:



If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False ' No need to run things onto a single line, especially if this is inconsistent with the other code.
End If


Can simply be:



Test1 = (TheDay = Day(currentDate)) And (TheUnit = Unit ) ' perhaps "FindMatch" is better descriptive name.


You set AdvRow and PropRow (what are these anyway - proper naming?) relative to each other within the loop, but you don't change either in that loop - so a single variable (AdvRow) will suffice.



You don't error check to ensure that the data you are reading is the right form - what happens if the data sheet does not exist, or that cell that is read is not a date?



You use NextRow while in a loop - but you already access a cell in the loop that tells you what the row is. This is one variable that can be dropped. And you are using NextRow as the CurrentRow - this is another example of a confusing variable name.



A big performance hit will come from having three nested loops, but also accessing each cell individually within those loops. Each time you make the program switch from looking at the VBA to looking at the Excel ranges is a cost in performance - this is why taking a range and putting it into an array improves efficiency.



'Function to return Collection of dates between Start Date and End Date
'**** You don't check to see if Start comes before End - what does it mean if they are the wrong way round?
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd
dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function

'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With

'will be a more efficient way of setting the array, but this will do for now
' 0..998, 0..4 array - datestart, dateend, facility, unit, pax

'Removes the following code:
'Set StartDate = Range("E2:E1000")
'NextRow = 2
'Sheets("Raw").Activate
'For Each Cell In StartDate
'Set dateStartCell = Range("E" & NextRow)
'Set dateEndCell = Range("G" & NextRow)

For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
'Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))

'Following code is no longer necessary
' Facility = Cells(NextRow, 3)
' Unit = Cells(NextRow, 2)
' Pax = Cells(NextRow, 12)

'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
'PropRow = AdvRow

theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Removing all my additional comments in Program gives you:



'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With
For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Of course, there may be some other logic paths, or even, perhaps, using Excel native functions that could help refine the problem.



I haven't been able to test the code (naturally), but it does compile in VBA.






share|improve this answer























  • rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
    – Jon Dee
    Dec 23 at 14:57










  • startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
    – Jon Dee
    Dec 23 at 14:59












  • I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
    – AJD
    Dec 23 at 21:02


















0














There are many areas of improvement in this coding.



A most important part of code hygiene is proper indenting, and always use Option Explicit.



Also, name your variables to something meaningful. For example, you use StartDate, but it is not a date (which the name implies), but a range.



You comment that your first function returns an Array, but it actually returns a Collection. Arrays are ordered, Collections are not, particularly in VBA.



You have some Boolean anti-patterns happening:



If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False ' No need to run things onto a single line, especially if this is inconsistent with the other code.
End If


Can simply be:



Test1 = (TheDay = Day(currentDate)) And (TheUnit = Unit ) ' perhaps "FindMatch" is better descriptive name.


You set AdvRow and PropRow (what are these anyway - proper naming?) relative to each other within the loop, but you don't change either in that loop - so a single variable (AdvRow) will suffice.



You don't error check to ensure that the data you are reading is the right form - what happens if the data sheet does not exist, or that cell that is read is not a date?



You use NextRow while in a loop - but you already access a cell in the loop that tells you what the row is. This is one variable that can be dropped. And you are using NextRow as the CurrentRow - this is another example of a confusing variable name.



A big performance hit will come from having three nested loops, but also accessing each cell individually within those loops. Each time you make the program switch from looking at the VBA to looking at the Excel ranges is a cost in performance - this is why taking a range and putting it into an array improves efficiency.



'Function to return Collection of dates between Start Date and End Date
'**** You don't check to see if Start comes before End - what does it mean if they are the wrong way round?
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd
dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function

'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With

'will be a more efficient way of setting the array, but this will do for now
' 0..998, 0..4 array - datestart, dateend, facility, unit, pax

'Removes the following code:
'Set StartDate = Range("E2:E1000")
'NextRow = 2
'Sheets("Raw").Activate
'For Each Cell In StartDate
'Set dateStartCell = Range("E" & NextRow)
'Set dateEndCell = Range("G" & NextRow)

For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
'Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))

'Following code is no longer necessary
' Facility = Cells(NextRow, 3)
' Unit = Cells(NextRow, 2)
' Pax = Cells(NextRow, 12)

'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
'PropRow = AdvRow

theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Removing all my additional comments in Program gives you:



'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With
For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Of course, there may be some other logic paths, or even, perhaps, using Excel native functions that could help refine the problem.



I haven't been able to test the code (naturally), but it does compile in VBA.






share|improve this answer























  • rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
    – Jon Dee
    Dec 23 at 14:57










  • startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
    – Jon Dee
    Dec 23 at 14:59












  • I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
    – AJD
    Dec 23 at 21:02
















0












0








0






There are many areas of improvement in this coding.



A most important part of code hygiene is proper indenting, and always use Option Explicit.



Also, name your variables to something meaningful. For example, you use StartDate, but it is not a date (which the name implies), but a range.



You comment that your first function returns an Array, but it actually returns a Collection. Arrays are ordered, Collections are not, particularly in VBA.



You have some Boolean anti-patterns happening:



If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False ' No need to run things onto a single line, especially if this is inconsistent with the other code.
End If


Can simply be:



Test1 = (TheDay = Day(currentDate)) And (TheUnit = Unit ) ' perhaps "FindMatch" is better descriptive name.


You set AdvRow and PropRow (what are these anyway - proper naming?) relative to each other within the loop, but you don't change either in that loop - so a single variable (AdvRow) will suffice.



You don't error check to ensure that the data you are reading is the right form - what happens if the data sheet does not exist, or that cell that is read is not a date?



You use NextRow while in a loop - but you already access a cell in the loop that tells you what the row is. This is one variable that can be dropped. And you are using NextRow as the CurrentRow - this is another example of a confusing variable name.



A big performance hit will come from having three nested loops, but also accessing each cell individually within those loops. Each time you make the program switch from looking at the VBA to looking at the Excel ranges is a cost in performance - this is why taking a range and putting it into an array improves efficiency.



'Function to return Collection of dates between Start Date and End Date
'**** You don't check to see if Start comes before End - what does it mean if they are the wrong way round?
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd
dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function

'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With

'will be a more efficient way of setting the array, but this will do for now
' 0..998, 0..4 array - datestart, dateend, facility, unit, pax

'Removes the following code:
'Set StartDate = Range("E2:E1000")
'NextRow = 2
'Sheets("Raw").Activate
'For Each Cell In StartDate
'Set dateStartCell = Range("E" & NextRow)
'Set dateEndCell = Range("G" & NextRow)

For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
'Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))

'Following code is no longer necessary
' Facility = Cells(NextRow, 3)
' Unit = Cells(NextRow, 2)
' Pax = Cells(NextRow, 12)

'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
'PropRow = AdvRow

theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Removing all my additional comments in Program gives you:



'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With
For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Of course, there may be some other logic paths, or even, perhaps, using Excel native functions that could help refine the problem.



I haven't been able to test the code (naturally), but it does compile in VBA.






share|improve this answer














There are many areas of improvement in this coding.



A most important part of code hygiene is proper indenting, and always use Option Explicit.



Also, name your variables to something meaningful. For example, you use StartDate, but it is not a date (which the name implies), but a range.



You comment that your first function returns an Array, but it actually returns a Collection. Arrays are ordered, Collections are not, particularly in VBA.



You have some Boolean anti-patterns happening:



If TheDay = Day(currentDate) And TheUnit = Unit Then
Test1 = True
Else: Test1 = False ' No need to run things onto a single line, especially if this is inconsistent with the other code.
End If


Can simply be:



Test1 = (TheDay = Day(currentDate)) And (TheUnit = Unit ) ' perhaps "FindMatch" is better descriptive name.


You set AdvRow and PropRow (what are these anyway - proper naming?) relative to each other within the loop, but you don't change either in that loop - so a single variable (AdvRow) will suffice.



You don't error check to ensure that the data you are reading is the right form - what happens if the data sheet does not exist, or that cell that is read is not a date?



You use NextRow while in a loop - but you already access a cell in the loop that tells you what the row is. This is one variable that can be dropped. And you are using NextRow as the CurrentRow - this is another example of a confusing variable name.



A big performance hit will come from having three nested loops, but also accessing each cell individually within those loops. Each time you make the program switch from looking at the VBA to looking at the Excel ranges is a cost in performance - this is why taking a range and putting it into an array improves efficiency.



'Function to return Collection of dates between Start Date and End Date
'**** You don't check to see if Start comes before End - what does it mean if they are the wrong way round?
Function GetDatesRange(dateStart As Date, dateEnd As Date) As Collection
Dim dates As New Collection
Dim currentDate As Date
currentDate = dateStart
Do While currentDate <= dateEnd
dates.Add currentDate
currentDate = DateAdd("d", 1, currentDate)
Loop
Set GetDatesRange = dates
End Function

'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With

'will be a more efficient way of setting the array, but this will do for now
' 0..998, 0..4 array - datestart, dateend, facility, unit, pax

'Removes the following code:
'Set StartDate = Range("E2:E1000")
'NextRow = 2
'Sheets("Raw").Activate
'For Each Cell In StartDate
'Set dateStartCell = Range("E" & NextRow)
'Set dateEndCell = Range("G" & NextRow)

For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
'Set allDates = GetDatesRange(dateStartCell.Value, dateEndCell.Value)
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))

'Following code is no longer necessary
' Facility = Cells(NextRow, 3)
' Unit = Cells(NextRow, 2)
' Pax = Cells(NextRow, 12)

'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
'PropRow = AdvRow

theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Removing all my additional comments in Program gives you:



'Sub to move raw data into predictable format
Sub Program()
Dim rawData As Variant
Dim currentRow As Long
Application.ScreenUpdating = False
With Sheets("Raw")
rawData = .Range(Union(.Range("E2:E1000"), .Range("G2:G1000"), .Range("C2:C1000"), .Range("B2:B1000"), .Range("L2:L1000"))).Value
End With
For currentRow = LBound(rawData, 1) To UBound(rawData, 1)
Dim allDates As Collection
Dim currentDateSter As Variant
Set allDates = GetDatesRange(CDate(rawData(currentRow, 0)), CDate(rawData(currentRow, 0)))
'Evaluating if the date and name already exist
For Each currentDateSter In allDates
Dim checkSheet As Worksheet ' not sure what to call this
Dim currentDate As Date
Dim advRow As Long
currentDate = CDate(currentDateSter) ' what if this is not a date?
Set checkSheet = Sheets(MonthName(Month(currentDate), True) & Year(currentDate))

advRow = 3
'evaluating if the date and name already exists if it does, and determines row for data entry
'eventually end up writing over data if it already exists however, column C has 125 unique possibilities
'that will fill another column in the month tabs
Do
Dim isMatch As Boolean ' Test1
Dim isOffsetMatch As Boolean ' Test2
Dim theDay As String
Dim theUnit As String
advRow = advRow + 1
theDay = checkSheet.Cells(advRow, 1) ' fully qualified access to cells - no ambiguity
theUnit = checkSheet.Cells(advRow, 2)
isMatch = (theDay = Day(currentDate)) And (theUnit = rawData(currentRow, 3))
isOffsetMatch = (theDay = theUnit)
Loop Until isMatch Or isOffsetMatch
checkSheet.Cells(advRow, 2).Value = rawData(currentRow, 3)
checkSheet.Cells(advRow, 1).Value = Day(currentDate)
checkSheet.Cells(advRow, 3).Value = rawData(currentRow, 4)
Next currentDateSter
Next currentRow
Application.ScreenUpdating = True
End Sub


Of course, there may be some other logic paths, or even, perhaps, using Excel native functions that could help refine the problem.



I haven't been able to test the code (naturally), but it does compile in VBA.







share|improve this answer














share|improve this answer



share|improve this answer








edited Dec 23 at 21:09

























answered Dec 22 at 21:26









AJD

1,2211213




1,2211213












  • rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
    – Jon Dee
    Dec 23 at 14:57










  • startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
    – Jon Dee
    Dec 23 at 14:59












  • I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
    – AJD
    Dec 23 at 21:02




















  • rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
    – Jon Dee
    Dec 23 at 14:57










  • startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
    – Jon Dee
    Dec 23 at 14:59












  • I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
    – AJD
    Dec 23 at 21:02


















rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
– Jon Dee
Dec 23 at 14:57




rawData = Sheets("Raw").Range("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000") gives me a run-time error "450" wrong number of arguments or invalid property assignment
– Jon Dee
Dec 23 at 14:57












startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
– Jon Dee
Dec 23 at 14:59






startdate will not be after enddate, the data source the info is being pulled from will not allow submission if it is incorrect. The data as is, is very predictable. I should have qualified that.
– Jon Dee
Dec 23 at 14:59














I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
– AJD
Dec 23 at 21:02






I had slipped into some Excel shorthand about evalutating this union. Try: rawData = Sheets("Raw").Range(Union("E2:E1000", "G2:G1000", "C2:C1000", "B2:B1000", "L2:L1000")).Value
– AJD
Dec 23 at 21:02












Jon Dee is a new contributor. Be nice, and check out our Code of Conduct.










draft saved

draft discarded


















Jon Dee is a new contributor. Be nice, and check out our Code of Conduct.













Jon Dee is a new contributor. Be nice, and check out our Code of Conduct.












Jon Dee is a new contributor. Be nice, and check out our Code of Conduct.
















Thanks for contributing an answer to Code Review Stack Exchange!


  • Please be sure to answer the question. Provide details and share your research!

But avoid



  • Asking for help, clarification, or responding to other answers.

  • Making statements based on opinion; back them up with references or personal experience.


Use MathJax to format equations. MathJax reference.


To learn more, see our tips on writing great answers.





Some of your past answers have not been well-received, and you're in danger of being blocked from answering.


Please pay close attention to the following guidance:


  • Please be sure to answer the question. Provide details and share your research!

But avoid



  • Asking for help, clarification, or responding to other answers.

  • Making statements based on opinion; back them up with references or personal experience.


To learn more, see our tips on writing great answers.




draft saved


draft discarded














StackExchange.ready(
function () {
StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f210120%2ftab-month-tracker%23new-answer', 'question_page');
}
);

Post as a guest















Required, but never shown





















































Required, but never shown














Required, but never shown












Required, but never shown







Required, but never shown

































Required, but never shown














Required, but never shown












Required, but never shown







Required, but never shown







Popular posts from this blog

Сан-Квентин

Алькесар

Josef Freinademetz