|
|
|
@@ -20,7 +20,7 @@
|
|
|
|
-- "Where should I build a padel court?"
|
|
|
|
-- "Where should I build a padel court?"
|
|
|
|
-- Computed for ALL locations — zero-court locations score highest on supply gap.
|
|
|
|
-- Computed for ALL locations — zero-court locations score highest on supply gap.
|
|
|
|
-- H3 catchment methodology: addressable market and supply gap use a regional
|
|
|
|
-- H3 catchment methodology: addressable market and supply gap use a regional
|
|
|
|
-- H3 catchment (res-4 cell + 6 neighbours, ~462km², ~15-18km radius).
|
|
|
|
-- H3 catchment (res-5 cell + 6 neighbours, ~24km radius).
|
|
|
|
--
|
|
|
|
--
|
|
|
|
-- 25 pts addressable market — log-scaled catchment population, ceiling 500K
|
|
|
|
-- 25 pts addressable market — log-scaled catchment population, ceiling 500K
|
|
|
|
-- 20 pts economic power — income PPS, normalised to 35,000
|
|
|
|
-- 20 pts economic power — income PPS, normalised to 35,000
|
|
|
|
@@ -63,30 +63,30 @@ base AS (
|
|
|
|
l.padel_venues_per_100k,
|
|
|
|
l.padel_venues_per_100k,
|
|
|
|
l.nearest_padel_court_km,
|
|
|
|
l.nearest_padel_court_km,
|
|
|
|
l.tennis_courts_within_25km,
|
|
|
|
l.tennis_courts_within_25km,
|
|
|
|
l.h3_cell_res4
|
|
|
|
l.h3_cell_res5
|
|
|
|
FROM foundation.dim_locations l
|
|
|
|
FROM foundation.dim_locations l
|
|
|
|
),
|
|
|
|
),
|
|
|
|
-- Aggregate population and court counts per H3 cell (res 4, ~10km edge).
|
|
|
|
-- Aggregate population and court counts per H3 cell (res 5, ~8.5km edge).
|
|
|
|
-- Grouping by cell first (~30-50K distinct cells vs 140K locations) keeps the
|
|
|
|
-- Grouping by cell first (~50-80K distinct cells vs 140K locations) keeps the
|
|
|
|
-- subsequent lateral join small.
|
|
|
|
-- subsequent lateral join small.
|
|
|
|
hex_stats AS (
|
|
|
|
hex_stats AS (
|
|
|
|
SELECT
|
|
|
|
SELECT
|
|
|
|
h3_cell_res4,
|
|
|
|
h3_cell_res5,
|
|
|
|
SUM(population) AS hex_population,
|
|
|
|
SUM(population) AS hex_population,
|
|
|
|
SUM(padel_venue_count) AS hex_padel_courts
|
|
|
|
SUM(padel_venue_count) AS hex_padel_courts
|
|
|
|
FROM foundation.dim_locations
|
|
|
|
FROM foundation.dim_locations
|
|
|
|
GROUP BY h3_cell_res4
|
|
|
|
GROUP BY h3_cell_res5
|
|
|
|
),
|
|
|
|
),
|
|
|
|
-- For each location, sum hex_stats across the cell + 6 neighbours (k_ring=1).
|
|
|
|
-- For each location, sum hex_stats across the cell + 6 neighbours (k_ring=1).
|
|
|
|
-- Effective catchment: ~462km², ~15-18km radius — realistic driving distance.
|
|
|
|
-- Effective catchment: ~24km radius — realistic driving distance.
|
|
|
|
catchment AS (
|
|
|
|
catchment AS (
|
|
|
|
SELECT
|
|
|
|
SELECT
|
|
|
|
l.geoname_id,
|
|
|
|
l.geoname_id,
|
|
|
|
SUM(hs.hex_population) AS catchment_population,
|
|
|
|
SUM(hs.hex_population) AS catchment_population,
|
|
|
|
SUM(hs.hex_padel_courts) AS catchment_padel_courts
|
|
|
|
SUM(hs.hex_padel_courts) AS catchment_padel_courts
|
|
|
|
FROM base l,
|
|
|
|
FROM base l,
|
|
|
|
LATERAL (SELECT UNNEST(h3_grid_disk(l.h3_cell_res4, 1)) AS cell) ring
|
|
|
|
LATERAL (SELECT UNNEST(h3_grid_disk(l.h3_cell_res5, 1)) AS cell) ring
|
|
|
|
JOIN hex_stats hs ON hs.h3_cell_res4 = ring.cell
|
|
|
|
JOIN hex_stats hs ON hs.h3_cell_res5 = ring.cell
|
|
|
|
GROUP BY l.geoname_id
|
|
|
|
GROUP BY l.geoname_id
|
|
|
|
),
|
|
|
|
),
|
|
|
|
-- Match dim_cities via (country_code, geoname_id) to get city_slug + exact venue count.
|
|
|
|
-- Match dim_cities via (country_code, geoname_id) to get city_slug + exact venue count.
|
|
|
|
|