I’ve uploaded the latest house price data from the Land Registry to my website. Not much excitement in the overall figures but there’s more interesting data at the regional level. A year after being the City of Culture, Hull is suffering the hangover and the wild swings continue in central London (although this is mostly down to the small number of sales that occur in the WC postcode area).
Thursday, August 30, 2018
Thursday, August 23, 2018
Decoding paths in here maps
In my panic to move my website off Google Maps, I took some shortcuts. One of those was to continue to use the helper function google.maps.geometry.encoding.decodePath. I primarily use this for decoding the paths of Strava segments, since these are encoded using the same algorithm as Google. But loading up the Google Maps API script for a single function is rather ridiculous, so I’ve converted the path decoder from the rather marvellous (and seemingly abandoned) Strava.NET library.
For your pleasure, here is some TypeScript that will create a here maps LineString from an encoded path. Other map providers are available and adjusting it for them should be fairly straightforward.
function decodePath(polylinechars: string): H.geo.LineString { var poly = new H.geo.LineString(); if (polylinechars == null || polylinechars === "") { return poly; } var index = 0; var currentLat = 0; var currentLng = 0; while (index < polylinechars.length) { // calculate next latitude var sum = 0; var shifter = 0; var next5bits; do { next5bits = polylinechars.charCodeAt(index++) - 63; sum |= (next5bits & 31) << shifter; shifter += 5; } while (next5bits >= 32 && index < polylinechars.length); if (index >= polylinechars.length) break; currentLat += (sum & 1) == 1 ? ~(sum >> 1) : (sum >> 1); // calculate next longitude sum = 0; shifter = 0; do { next5bits = polylinechars.charCodeAt(index++) - 63; sum |= (next5bits & 31) << shifter; shifter += 5; } while (next5bits >= 32 && index < polylinechars.length); if (index >= polylinechars.length && next5bits >= 32) break; currentLng += (sum & 1) == 1 ? ~(sum >> 1) : (sum >> 1); poly.pushPoint(new H.geo.Point(currentLat / 100000.0, currentLng / 100000.0)); } return poly; }
Tuesday, August 21, 2018
UK postcode date August 2018
I’ve uploaded the latest postcode data from the ONS to my website. I’ve checked the data with my usual sanity check but let me know if you spot anything amiss
Thursday, August 02, 2018
Here Maps full screen control
One thing missing from Here Maps is a full screen control, but fortunately it’s possible to add your own custom controls to the map. This isn’t brilliantly documented but a support person from Here Maps was able to help me out with an example. That combined with the fact I’d built something similar in Google Maps before it had its own full screen control. So here’s a TypeScript function that will add a full screen button to a Here Map.
function hereMapsFullScreenControl(ui: H.ui.UI) { // add custom UI control const myCustomControl = new H.ui.Control(); myCustomControl.addClass("customControl"); ui.addControl("myCustomControl", myCustomControl); // Set the position of the control in the UI's layout myCustomControl.setAlignment(H.ui.LayoutAlignment.TOP_RIGHT); const myCustomPanel = new H.ui.base.OverlayPanel(); myCustomPanel.addClass("customPanel"); myCustomControl.addChild(myCustomPanel); // store original styles const mapDiv = ui.getMap().getElement() as HTMLElement; let divStyle: CSSStyleDeclaration = mapDiv.style; if (mapDiv.runtimeStyle) { divStyle = mapDiv.runtimeStyle; } const originalPos: string = divStyle.position; let originalWidth: string = divStyle.width; let originalHeight: string = divStyle.height; // ie8 hack if (originalWidth === "") { originalWidth = mapDiv.style.width; } if (originalHeight === "") { originalHeight = mapDiv.style.height; } const originalTop: string = divStyle.top; const originalLeft: string = divStyle.left; const originalZIndex: string = divStyle.zIndex; let bodyStyle: CSSStyleDeclaration = document.body.style; if (document.body.runtimeStyle) { bodyStyle = document.body.runtimeStyle; } const originalOverflow: string = bodyStyle.overflow; const myCustomButton = new H.ui.base.PushButton({ label: "Full screen", onStateChange: (evt) => { // OK, button state changed... if it's currently down if (myCustomButton.getState() === H.ui.base.Button.State.DOWN) { // go full screen mapDiv.style.position = "fixed"; mapDiv.style.width = "100%"; mapDiv.style.height = "100%"; mapDiv.style.top = "0"; mapDiv.style.left = "0"; mapDiv.style.zIndex = "100"; document.body.style.overflow = "hidden"; } else { // exit full screen if (originalPos === "") { mapDiv.style.position = "relative"; } else { mapDiv.style.position = originalPos; } mapDiv.style.width = originalWidth; mapDiv.style.height = originalHeight; mapDiv.style.top = originalTop; mapDiv.style.left = originalLeft; mapDiv.style.zIndex = originalZIndex; document.body.style.overflow = originalOverflow; } ui.getMap().getViewPort().resize(); }, }); myCustomPanel.addChild(myCustomButton as any); myCustomPanel.setState(H.ui.base.OverlayPanel.State.OPEN); }
Wednesday, August 01, 2018
Land Registry data for June 2018
The server is still churning through the data for the England and Wales house price data for June 2018 but the top-level data is there. Annual inflation has dropped a little, but not enough to warrant any excitement.