Analyze a Trip by Image Metadata
Using pictures and their metadata, you can perform geo computations to visualize and analyze a walking tour of Lisbon, Portugal.
Import and analyze a collection of photos taken during a visit.
In[1]:=
data:image/s3,"s3://crabby-images/22cb2/22cb262af3581cb8c1ca8d1ff4234eaecade7a62" alt="Click for copyable input"
files = Map[
img \[Function] ExampleData[img, "FilePath"],
ExampleData[{"TestImageSet", "Lisbon2016"}]
];
Gather all the information required in an association.
In[2]:=
data:image/s3,"s3://crabby-images/03fa5/03fa5dd02df3cada50b5632b03f4ef5f347bb3f9" alt="Click for copyable input"
labels = {"Thumbnail", "DateTime", "GeoPosition"};
In[3]:=
data:image/s3,"s3://crabby-images/84407/844077cb6851c979a78840dab6dde58e5d3f6705" alt="Click for copyable input"
dataSet = Dataset@Map[
file \[Function]
AssociationThread[
labels ->
Import[file, {labels}, IncludeMetaInformation -> False]],
files
]
Out[3]=
data:image/s3,"s3://crabby-images/22d1a/22d1a6bf62b82c5a9de693804a54facf8ddc4e27" alt=""
Visualize locations where images where taken.
In[4]:=
data:image/s3,"s3://crabby-images/0c346/0c3463806b9dc168831e217d10d9c7b5d01792b3" alt="Click for copyable input"
gps = dataSet[[All, "GeoPosition"]];
GeoGraphics[GeoMarker@gps, GeoRangePadding -> Quantity[100, "Meters"]]
Out[4]=
data:image/s3,"s3://crabby-images/e3312/e3312245e27c5ecfb6993f64025d37d549e4c562" alt=""
Compute the walking distance to take all pictures and return.
In[5]:=
data:image/s3,"s3://crabby-images/1f8f5/1f8f58ecc5e955dcb5a30c211d18e3581bf66f20" alt="Click for copyable input"
closeLoop = path \[Function] Append[path, First[path]];
In[6]:=
data:image/s3,"s3://crabby-images/5b4d2/5b4d2132518316bd7893521b175459ae05056bbb" alt="Click for copyable input"
sortedGPS =
closeLoop@
Normal@dataSet[SortBy["DateTime"], Take[#GeoPosition, All, 2] &];
UnitConvert[
TravelDistance[sortedGPS, TravelMethod -> "Walking"], "Kilometers"]
Out[6]=
data:image/s3,"s3://crabby-images/64a58/64a58a040b2c917935b1d6c31658f0a77aac4bf3" alt=""
Visualize the path to take all photos in temporal order.
In[7]:=
data:image/s3,"s3://crabby-images/567c5/567c5d6f01eb12ffa223d23981b524ed8e90ab93" alt="Click for copyable input"
travel = TravelDirections[sortedGPS, TravelMethod -> "Walking"]
Out[8]=
data:image/s3,"s3://crabby-images/d1b9d/d1b9def71f38d14753ec0251bb21b276c99ff6d1" alt=""
In[9]:=
data:image/s3,"s3://crabby-images/9cd6b/9cd6b3fba50388d80d4cc25e04418473ce46d432" alt="Click for copyable input"
Animate[
GeoGraphics[{
Style[Normal@travel["Dataset"][1 ;; n, "Path"], Thick, Red],
dataSet[All,
GeoMarker[Take[#GeoPosition, All, 2], #Thumbnail] &]},
GeoRangePadding -> Quantity[200, "Meters"]
],
{n, 1, Length[travel["Dataset"]] + 1, 1}
]
data:image/s3,"s3://crabby-images/01152/0115264133f75da8e4180aad4e4ae92f32968147" alt=""
Compute and visualize the shortest path that could have been used instead.
In[10]:=
data:image/s3,"s3://crabby-images/9a174/9a1745c220028904643f09a044421cf36164d809" alt="Click for copyable input"
optimalPath =
FindShortestTour[Normal@sortedGPS,
DistanceFunction -> (QuantityMagnitude[
TravelDistance[{#1, #2}, TravelMethod -> "Walking"],
"Kilometer"] &)]
Out[10]=
data:image/s3,"s3://crabby-images/5b64b/5b64b08ab63fc20d8ee30c36230f77fc7d1de416" alt=""
In[11]:=
data:image/s3,"s3://crabby-images/29c3f/29c3f95a5c2b388099636b37d735e444cdef55a0" alt="Click for copyable input"
Quantity[First@optimalPath, "Kilometers"]
Out[11]=
data:image/s3,"s3://crabby-images/dc388/dc388bf21836f3a06bf0416c24fa0912180d122a" alt=""
In[12]:=
data:image/s3,"s3://crabby-images/abb77/abb77ce054633b5ad174e9ae72cfa48cf653bcc0" alt="Click for copyable input"
shortestGPS = sortedGPS[[Last@optimalPath]];
In[13]:=
data:image/s3,"s3://crabby-images/3291f/3291f70d8bd15686a18db6f68e3b5689f3279663" alt="Click for copyable input"
shortestTravel =
TravelDirections[shortestGPS, TravelMethod -> "Walking"]
Out[14]=
data:image/s3,"s3://crabby-images/59437/594377127259259a186f8aa147338828473a5bee" alt=""
In[15]:=
data:image/s3,"s3://crabby-images/606e8/606e823e9cf791043ddc2247d81546701f850591" alt="Click for copyable input"
Animate[
GeoGraphics[{
Style[Normal@shortestTravel["Dataset"][1 ;; n, "Path"], Thick,
Blue],
dataSet[All,
GeoMarker[Take[#GeoPosition, All, 2], #Thumbnail] &]},
GeoRangePadding -> Quantity[200, "Meters"]
],
{n, 1, Length[shortestTravel["Dataset"]] + 1, 1}
]
data:image/s3,"s3://crabby-images/4fe2c/4fe2cd5c65e086ab83cd302f132f97c609ae8810" alt=""