|
Option Explicit |
|
Public sKML As String |
|
Public GradientColorName() |
|
Public s1 As String, s2 As String, s3 As String, s4 As String, s5 As String, s6 As String |
|
|
|
Sub VCC2KML() |
|
Dim r As Range |
|
Dim d |
|
Dim rowDate As Long, colDate As Long |
|
|
|
colDate = Cells.Find(What:="/CapturedTrack/Trackpoints/Trackpoint/@dateTime", _ |
|
SearchDirection:=xlPrevious, _ |
|
SearchOrder:=xlByColumns).Column |
|
|
|
'assumes Velocitek has time, h, Lat Lon spd continuous. Could be buggy |
|
Set r = ActiveSheet.Cells(3, colDate).Resize(Application.CountA(Columns(colDate)) - 1, 5) |
|
|
|
d = r.Value |
|
|
|
Call GPSLLSpd2KML(d) |
|
|
|
Set r = Nothing |
|
|
|
End Sub |
|
|
|
Sub GPSLLSpd2KML(d) |
|
Dim dCull, dSpd |
|
Dim r As Range |
|
Dim colorIdx As Long |
|
Dim lon As Double, lat As Double |
|
Dim SOG As Double, time As String |
|
Dim minSpdPlot As Double, maxSpdPlot As Double |
|
Dim i As Long |
|
Dim OrigNPts As Long, CulledNPts As Long |
|
Dim maxDistBetwPts As Double |
|
Dim minSpd As Double, maxSpd As Double, AvSpd As Double, SDSPd As Double |
|
Dim title As String |
|
Dim path As String, fnm As String |
|
|
|
Const colTime As Long = 1 'unused |
|
Const colLat As Long = 3 'DD.DDD |
|
Const colLon As Long = 4 'DD.DDD |
|
Const colSOG As Long = 5'this could be Altitude for example |
|
|
|
Dim ccOffset As Long 'do not delete |
|
Dim ccIncrement As Long |
|
ccIncrement = 64 'fiddle with this to speed up or not. seems that between 2 and 1000 is OK |
|
ccOffset = 0 |
|
|
|
sKML = vbNullString |
|
|
|
s1 = "<Placemark>" & vbCr & " <styleUrl>#" 'to speed up output of placemarks |
|
s2 = "</styleUrl>" & vbCr & " <Point>" & vbCr & " <coordinates>" & vbCr & " " |
|
s3 = ",0" & vbCr & " </coordinates>" & vbCr & " </Point>" & vbCr & "</Placemark>" & vbCr |
|
s4 = "<Placemark>" & vbCr & "<description>" |
|
s5 = "</description>" & vbCr |
|
s6 = " <styleUrl>#" |
|
|
|
' |
|
' <Placemark> |
|
' <TimeStamp> |
|
' <when>2007-01-14T21:05:02Z</when> |
|
' </TimeStamp> |
|
' <styleUrl>#paddle-a</styleUrl> |
|
|
|
GradientColorName = Array("darkblue", "blue", "deepskyblue", "cyan", "MediumSpringGreen", "yellowgreen", "yellow", "orange", "OrangeRed", "red") |
|
|
|
'coarsen track points |
|
maxDistBetwPts = InputBox("Input min distance between pts, metres", "Cull redundant points", 3) |
|
Call CullData(d, dCull, maxDistBetwPts, OrigNPts, CulledNPts) |
|
ReDim dSpd(1 To UBound(dCull, 1), 1 To 1) |
|
For i = 1 To UBound(dCull, 1) |
|
dSpd(i, 1) = dCull(i, 5) |
|
Next |
|
Call SpdStats(dSpd, minSpd, maxSpd, AvSpd, SDSPd) |
|
|
|
title = "Spds for Colour: " & _ |
|
"Min= " & Format(minSpd, "0.0") & " Max= " & Format(maxSpd, "0.0") & _ |
|
" Av= " & Format(AvSpd, "0.0") & " SD= " & Format(SDSPd, "0.0") |
|
|
|
'default min Speed is Average-2*StDev |
|
minSpdPlot = InputBox("Enter low speed for Blue", title, Format(findMax(AvSpd - 2 * SDSPd, 0), "0")) |
|
maxSpdPlot = InputBox("Enter high speed for Red", title, Format(maxSpd, "0.0")) |
|
|
|
Call outputLine3(sKML, "<?xml version=""1.0"" encoding=""UTF-8""?>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, "<kml xmlns=""http://www.opengis.net/kml/2.2"">", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, "<Document>", ccOffset, ccIncrement) |
|
|
|
Call setupStyleIcons(ccOffset, ccIncrement) |
|
|
|
Dim m |
|
m = Application.StatusBar |
|
|
|
For i = 1 To UBound(dCull, 1) |
|
|
|
Application.StatusBar = "writing KML " & Format(i / UBound(dCull, 1), "0" & "%") 'ADDS ~5% to time |
|
|
|
SOG = CDbl(dCull(i, colSOG)) |
|
colorIdx = GradientColor(SOG, minSpdPlot, maxSpdPlot, 1, 10) |
|
lon = CDbl(dCull(i, colLon)) |
|
lat = CDbl(dCull(i, colLat)) |
|
time = CStr(dCull(i, colTime)) |
|
|
|
Call makePlacemarkWithTimeAndSpeed(CStr(GradientColorName(colorIdx - 1)), lon, lat, time, SOG, ccOffset, ccIncrement) |
|
|
|
Next |
|
|
|
Application.StatusBar = m |
|
|
|
Call outputLine3(sKML, "</Document>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, "</kml>", ccOffset, ccIncrement) |
|
|
|
Call writeKMLFile |
|
|
|
End Sub |
|
|
|
Sub CullData(arr1, cullArr, maxDist As Double, OrigNPts As Long, CulledNPts As Long) |
|
Dim arrD1Cull() |
|
Dim i As Long, j As Long, k As Long 'i-orig array, j - culled array, k - columns |
|
Dim cullDist As Boolean |
|
Dim cntDist As Long |
|
Dim nRowsData As Long, nColsData As Long |
|
Dim DistOrig2CullPt As Double |
|
Const colLat As Long = 3 |
|
Const colLon As Long = 4 |
|
|
|
nRowsData = UBound(arr1, 1) |
|
nColsData = UBound(arr1, 2) |
|
|
|
ReDim arrD1Cull(1 To nRowsData, 1 To nColsData) |
|
|
|
'initialise first row |
|
For k = 1 To nColsData |
|
arrD1Cull(1, k) = arr1(1, k) |
|
Next |
|
j = 1 'counter for culled array |
|
cullDist = True |
|
|
|
For i = 1 To nRowsData |
|
DistOrig2CullPt = 1852 * PsnToRange(CDbl(arr1(i, colLat)), CDbl(arr1(i, colLon)), CDbl(arrD1Cull(j, colLat)), CDbl(arrD1Cull(j, colLon))) |
|
If DistOrig2CullPt > maxDist Then |
|
cullDist = False |
|
cntDist = cntDist + 1 'culled points count |
|
j = j + 1 |
|
For k = 1 To nColsData |
|
arrD1Cull(j, k) = arr1(i, k) |
|
Next |
|
cullDist = True |
|
End If |
|
|
|
Next |
|
|
|
'resize culledArray |
|
ReDim cullArr(1 To cntDist, 1 To nColsData) |
|
For i = 1 To cntDist |
|
For j = 1 To nColsData |
|
cullArr(i, j) = arrD1Cull(i, j) |
|
Next |
|
Next |
|
OrigNPts = nRowsData |
|
CulledNPts = OrigNPts - cntDist + 1 |
|
|
|
End Sub |
|
|
|
Sub writeKMLFile() |
|
|
|
Call TxtFileDriver(sKML, "kml") |
|
|
|
End Sub |
|
|
|
Function PsnToRange(P1LatDeg As Double, P1LonDeg As Double, P2LatDeg As Double, P2LonDeg As Double) ', VMag As VectorType) |
|
'nm |
|
Dim y As Double |
|
Dim X As Double |
|
Const rad As Double = 57.2957795130823 |
|
|
|
X = 60 * ModOneEighty(P2LonDeg - P1LonDeg) * Cos(P1LatDeg / rad) |
|
y = 60 * ModOneEighty(P2LatDeg - P1LatDeg) |
|
|
|
PsnToRange = Sqr(X * X + y * y) |
|
|
|
End Function |
|
|
|
Sub SpdStats(d, minSpd As Double, maxSpd As Double, AvSpd As Double, SDSPd As Double) |
|
|
|
With Application.WorksheetFunction |
|
minSpd = .Min(d) |
|
maxSpd = .Max(d) |
|
AvSpd = .Average(d) |
|
SDSPd = .StDev(d) |
|
End With |
|
|
|
End Sub |
|
|
|
Sub outputLine3(ByRef Dest As String, ByRef Source As String, ByRef ccOffset As Long, ByRef ccIncrement As Long) |
|
Call Concat2(Dest, Source, ccOffset, ccIncrement) |
|
End Sub |
|
|
|
Sub Concat2(ByRef Dest As String, ByRef Source As String, ccOffset As Long, ccIncrement As Long) |
|
'dim coffset as long put at top of calling Sub. say 1000 ? It increases string by this length in chunks |
|
'Public ccOffset As Long 'do not delete |
|
'Public ccIncrement As Long 'Number of characters to allocate every timePublic GradientColorName() |
|
Dim L As Long |
|
|
|
L = Len(Source) |
|
If (ccOffset + L) >= Len(Dest) Then |
|
If L > ccIncrement Then |
|
Dest = Dest & Space$(L) |
|
Else |
|
Dest = Dest & Space$(ccIncrement) |
|
End If |
|
End If |
|
|
|
Mid$(Dest, ccOffset + 1, L) = Source |
|
ccOffset = ccOffset + L |
|
|
|
Dest = Left$(Dest, ccOffset) |
|
|
|
End Sub |
|
|
|
Sub setupStyleIcons(ccOffset As Long, ccIncrement As Long) |
|
Dim i As Long |
|
Dim strColor As String |
|
Dim strHexColor As String |
|
|
|
For i = 1 To 10 |
|
strColor = GradientColorName(i - 1) |
|
strHexColor = GradientColorHex(i - 1) |
|
Call outputLine3(sKML, "<Style id=""" & strColor & """>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " <IconStyle>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " <color>" & strHexColor & "</color>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " <scale>.3</scale>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " <Icon>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " <href>http://maps.google.com/mapfiles/kml/pal2/icon26.png</href>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " </Icon>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, " </IconStyle>", ccOffset, ccIncrement) |
|
Call outputLine3(sKML, "</Style>", ccOffset, ccIncrement) |
|
Next |
|
|
|
End Sub |
|
|
|
Sub makePlacemarkWithTimeAndSpeed(ByRef colorNm As String, lon As Double, lat As Double, _ |
|
time As String, SOG As Double, ccOffset As Long, ByRef ccIncrement As Long) |
|
|
|
Call outputLine3(sKML, s4 & time & vbCr & SOG & s5, ccOffset, ccIncrement) |
|
Call outputLine3(sKML, s6 & colorNm & s2 & lon & "," & lat & s3, ccOffset, ccIncrement) |
|
|
|
End Sub |
|
|
|
Sub TxtFileDriver(txt As String, Optional extension) |
|
Dim path As String, fnm As String |
|
|
|
Call Folder_Picker(path) |
|
|
|
If Not IsMissing(extension) Then |
|
fnm = InputBox("type filename") |
|
fnm = fnm & "." & CStr(extension) |
|
Else |
|
fnm = InputBox("type filename with extension") |
|
End If |
|
|
|
Call writeTxtFile(txt, path, fnm) |
|
|
|
End Sub |
|
|
|
Sub Folder_Picker(fldr As String) |
|
|
|
With Application.FileDialog(msoFileDialogFolderPicker) |
|
.InitialFileName = ActiveWorkbook.path & "\" |
|
.Show |
|
If .SelectedItems.Count = 0 Then GoTo 1 |
|
fldr = .SelectedItems(1) |
|
fldr = fldr & "\" |
|
1 End With |
|
End Sub |
|
|
|
Sub writeTxtFile(txt As String, path As String, fnm As String, Optional extension) 'path As String, fnm As String) |
|
Dim fs, A |
|
|
|
fnm = makePathFileExt(path, fnm) |
|
|
|
Set fs = CreateObject("Scripting.FileSystemObject") |
|
Set A = fs.CreateTextFile(fnm, True, True) |
|
|
|
A.WriteLine (txt) |
|
|
|
Close |
|
|
|
End Sub |
|
|
|
Function findMax(A As Double, b As Double) As Double |
|
|
|
findMax = -(A > b) * A + -(b >= A) * b |
|
|
|
End Function |
|
|
|
Function GradientColor(val As Double, lowVal As Double, highVal As Double, _ |
|
lowColorIndex As Long, NumbrColors As Long) As Long |
|
Dim ratio As Single |
|
|
|
ratio = (val - lowVal) / (highVal - lowVal) |
|
If ratio > 1 Then ratio = 1 |
|
|
|
GradientColor = Int(ratio * (NumbrColors - 1)) + lowColorIndex |
|
If GradientColor > NumbrColors + lowColorIndex Then GradientColor = NumbrColors + lowColorIndex 'saturated max |
|
|
|
If GradientColor < lowColorIndex Then GradientColor = lowColorIndex 'saturated min |
|
|
|
End Function |
|
|
|
Function ModThreeSixty(A As Double) As Double |
|
|
|
ModThreeSixty = A - 360 * Int(A / 360) |
|
|
|
End Function |
|
Function ModOneEighty(A As Double) As Double |
|
'final range value is between |
|
' 180 and -180 degrees |
|
|
|
ModOneEighty = ModThreeSixty(A + 180) - 180 |
|
|
|
End Function |
|
Function GradientColorHex(colorIdx As Long) As String 'changes colors 16 to 32 from blue to red |
|
' Colours based on Nile/Hairyears thermal gradient scale |
|
Dim i As Long |
|
Dim arr() As Variant |
|
|
|
i = colorIdx |
|
|
|
' arr = Array("darkblue", "blue", "deepskyblue", "cyan", "MediumSpringGreen", "yellowgreen", "yellow", "orange", "OrangeRed", "red") |
|
arr = Array("ff8b0000", "ffff0000", "ffffbf00", "ffffff00", "ff9afa00", "ff32cd9a", "ff00ffff", "ff00a5ff", "ff0045ff", "ff0000ff") |
|
|
|
GradientColorHex = arr(i) |
|
|
|
End Function |
|
Function makePathFileExt(path As String, fnm As String, Optional extension) As String |
|
If Not IsMissing(extension) Then |
|
fnm = fnm & "." & extension |
|
ElseIf Mid(fnm, Len(fnm) - 3, 1) = "." Then |
|
'fnm=fnm 'assumes .xxx format |
|
Else |
|
fnm = fnm & "." & "txt" |
|
End If |
|
|
|
makePathFileExt = path & fnm |
|
|
|
End Function |