Excel Pivot Table or CrossTab to Flat List
Excel to CrossTabI often find myself struggling to deal with data that has been rendered almost unusable by the data provider who has converted it to a cross tab format. Usually they think they are helping and have probably started with a nice flat list then spent ages formatting it so the data spreads out across the columns. Sometimes they have achieved this by putting the data into a pivot table but then have pasted the table as values and removed all links to the underlying data. Very helpful.
Yesterday I had to deal with a data file that looked a little like this only it spread out across 455 columns and was frankly useless.
I wrote a post a while back that demonstrated how to convert a simple cross tab back to a manageable data list but I wanted to expand this with a slightly more complex example which had more field headings.
So starting with the table above which was probably a pasted copy from a pivot table I applied a little bit of formatting to achieve the following starting table.
What I have done is remove the grouping that the pivot table applies and ensured that I have the relevant week number, project and department details against each name
I often wish to convert pivot table outputs back into a data file and the way I fill in the blanks caused by grouping data is to prepare a sheet with formulas that copy values from above (or the side as appropriate):
I then copy the pivot table data and then use the paste special command selecting paste values and skip blanks as the options.
This results in the following output, which I then copy and paste over itself as values, to give me the desired structure.
Having spent a bit of time on the structure of my starting file I can run the macro that will work through each row and transpose the data from a columnar layout to one based on rows.
The advantage of this format is that I can now create my own pivot tables and cut the data as I see fit to produce multiple summary formats according to my audience.
Here’s the macro I use:
Sub CrossTabToList()
Dim wsCrossTab As Worksheet
Dim wsList As Worksheet
Dim iLastCol As Long
Dim iLastRow As Long
Dim iLastRowList As Long
Dim rngCTab As Range ‘Used for range in Sheet1 cross tab sheet
Dim rngList As Range ‘Destination range for the list
Dim ROW As LongSet wsCrossTab = Worksheets(“Sheet1″) ‘AMEND TO SHOW SHEET NUMBER WITH THE CROSS TAB
Set wsList = Worksheets.Add‘Find the last row in Sheet1 with the cross tab
iLastRow = wsCrossTab.Cells(Rows.Count, “A”).End(xlUp).ROW‘Set the initial value for the row in the destination worksheet
iLastRowList = 2‘Find the last column in Sheet1 with the cross tab
iLastCol = wsCrossTab.Range(“A2″).End(xlToRight).Column‘SET THE HEADING TITLES IN THE LIST SHEET
wsList.Range(“A1:F1″) = Array(“NAME”, “PROJECT”, “TYPE”, “PLAN/ACTUAL”, “WEEK”, “HOURS”)‘Start looping through the cross tab data
For ROW = 3 To iLastRow ‘START AT ROW 3 AS THIS IS WHERE DATA BEGINS
Set rngCTab = wsCrossTab.Range(“A” & ROW, “C” & ROW) ‘initial value A3 SETS THE RANGE TO‘INCLUDE ALL STATIC DATA – IN THIS CASE NAME, PROJECT, TYPE
Set rngList = wsList.Range(“A” & iLastRowList) ‘initial value A3‘Copy individual names in Col A (A3 initially) into as many rows as there are data columns
‘in the cross tab (less 3 for Col A-C).
rngCTab.Copy rngList.Resize(iLastCol – 3)‘SELECT THE HEADING ROW WITH FORECAST/ACTUAL
‘Move up ROW (INITIALLY 3) rows less TWO and across 3 columns (using offset function). Copy.
rngCTab.Offset(-(ROW – 2), 3).Resize(, iLastCol – 3).Copy‘Paste transpose to columns in the list sheet alongside the static data
rngList.Offset(0, 3).PasteSpecial Transpose:=True‘SELECT THE ROW WITH THE WEEK NUMBERS
‘Move up ROW (INITIALLY 3) rows less ONE and across 3 columns (using offset function). Copy.
rngCTab.Offset(-(ROW – 1), 3).Resize(, iLastCol – 3).Copy‘Paste transpose to columns in the list sheet alongside the static data
rngList.Offset(0, 4).PasteSpecial Transpose:=True‘Staying on same row (3 initially) copy the data from the cross tab
rngCTab.Offset(, 3).Resize(, iLastCol – 3).Copy‘Past transpose as column in list sheet
rngList.Offset(0, 5).PasteSpecial Transpose:=True‘Set the new last row in list sheet to be just below the last name copied
iLastRowList = iLastRowList + (iLastCol – 3)‘increment ROW by 1
Next ROWEnd Sub
I’m afraid the VBA code isn’t rendering very well – you’ll have to replace all the ” and ‘ marks before the macro editor will recognise the code.
Here is a copy of file – you could copy the macro code from here: Excel to CrossTab.xls





















Yay!
Thank you for posting this.
This is just Great! The only thing I can’t figure, how to do the same but without week sub rows – just four columns with names and multiple columns with data. If you can help it will just save me..
Fantastic script, I like the simplicity of some of the code, cutting out large swathes of unneccesary switching between sheets.
I’m using 2003 and had to change the formatting from your post, “–” to “-”, and ‘, “ and ″ to ‘, ” and ” respectively. Could youput a link on here like you did for your previous version
It could do with Application.ScreenUpdating = False putting at the start to speed it up.
It would be nice if it brought up boxes so that you could vary the data type, e.g. number of columns of static data, headings, number of columns of “forecast/actual” etc…. but I guess you can’t do everything!
Awesome & thanks
I’m sorry about the problem with the speech marks and apostrophes. I have the same problem when I copy the code – I think it must be related to the CSS code I use on my blog. I really should sort it out.
Do you mean a link to the actual spreadsheet?
I’ve added a link to the file in case that helps.