2019-01-19 13:47:08 -05:00
|
|
|
import urlapi from 'url';
|
2018-10-01 07:45:33 -04:00
|
|
|
import React, { Fragment } from 'react';
|
2018-09-30 19:13:01 -04:00
|
|
|
import { Link, NavLink } from 'react-router-dom';
|
2018-09-09 17:22:44 -04:00
|
|
|
|
2018-09-11 15:59:59 -04:00
|
|
|
import Sidebar from './sidebar';
|
|
|
|
import Tooltip from './tooltip';
|
2018-09-13 15:41:42 -04:00
|
|
|
import InfoBox from './info-box';
|
2018-11-13 05:45:35 -05:00
|
|
|
import { EditIcon } from './icons';
|
2018-11-30 04:26:28 -05:00
|
|
|
import { parseCategoryURL } from '../parse';
|
2018-09-11 15:59:59 -04:00
|
|
|
|
2018-10-03 16:47:49 -04:00
|
|
|
import CONFIG from './fields-config.json';
|
2018-09-30 19:32:24 -04:00
|
|
|
|
2018-10-03 16:47:49 -04:00
|
|
|
|
|
|
|
const BuildingView = (props) => {
|
|
|
|
if (!props.building_id){
|
|
|
|
return (
|
|
|
|
<Sidebar title="Building Not Found">
|
|
|
|
<InfoBox msg="We can't find that one anywhere - try the map again?" />
|
|
|
|
<div className="buttons-container with-space">
|
2018-11-29 17:00:53 -05:00
|
|
|
<Link to="/view/age.html" className="btn btn-secondary">Back to maps</Link>
|
2018-10-03 16:47:49 -04:00
|
|
|
</div>
|
|
|
|
</Sidebar>
|
|
|
|
);
|
|
|
|
}
|
2018-11-30 04:26:28 -05:00
|
|
|
const cat = parseCategoryURL(props.match.url);
|
2018-10-03 16:47:49 -04:00
|
|
|
return (
|
2018-11-29 17:00:53 -05:00
|
|
|
<Sidebar title={`Data available for this building`} back={`/view/${cat}.html`}>
|
2018-10-03 16:47:49 -04:00
|
|
|
{
|
|
|
|
CONFIG.map(section_props => (
|
|
|
|
<DataSection
|
2018-11-29 17:00:53 -05:00
|
|
|
key={section_props.slug} cat={cat}
|
2018-10-03 16:47:49 -04:00
|
|
|
building_id={props.building_id}
|
2018-10-04 17:50:33 -04:00
|
|
|
{...section_props}>
|
2018-10-03 16:47:49 -04:00
|
|
|
{
|
2018-10-25 09:36:52 -04:00
|
|
|
section_props.fields.map(field_props => {
|
2019-01-19 13:47:08 -05:00
|
|
|
|
|
|
|
switch (field_props.type) {
|
|
|
|
case "uprn_list":
|
|
|
|
return <UPRNsDataEntry
|
|
|
|
key={field_props.slug}
|
|
|
|
title={field_props.title}
|
|
|
|
value={props.uprns}
|
|
|
|
tooltip={field_props.tooltip} />
|
|
|
|
case "text_multi":
|
|
|
|
return <MultiDataEntry
|
|
|
|
key={field_props.slug}
|
|
|
|
title={field_props.title}
|
|
|
|
value={props[field_props.slug]}
|
|
|
|
tooltip={field_props.tooltip} />
|
|
|
|
default:
|
|
|
|
return <DataEntry
|
|
|
|
key={field_props.slug}
|
|
|
|
title={field_props.title}
|
|
|
|
value={props[field_props.slug]}
|
|
|
|
tooltip={field_props.tooltip} />
|
|
|
|
}
|
2018-10-25 09:36:52 -04:00
|
|
|
})
|
2018-10-03 16:47:49 -04:00
|
|
|
}
|
|
|
|
</DataSection>
|
|
|
|
))
|
|
|
|
}
|
|
|
|
</Sidebar>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const DataSection = (props) => {
|
2018-11-29 17:00:53 -05:00
|
|
|
const match = props.cat === props.slug;
|
2018-09-30 19:32:24 -04:00
|
|
|
return (
|
2018-10-03 16:47:49 -04:00
|
|
|
<section id={props.slug} className={(props.inactive)? "data-section inactive": "data-section"}>
|
2018-10-25 07:15:19 -04:00
|
|
|
<header className={(match? "active " : "") + " section-header view"}>
|
2018-10-04 17:50:33 -04:00
|
|
|
<NavLink
|
2018-11-29 17:00:53 -05:00
|
|
|
to={`/view/${props.slug}/building/${props.building_id}.html`}
|
2018-10-25 05:47:19 -04:00
|
|
|
title={(props.inactive)? 'Coming soon… Click the ? for more info.' :
|
|
|
|
(match)? 'Hide details' : 'Show details'}
|
2018-10-04 17:50:33 -04:00
|
|
|
isActive={() => match}>
|
2018-10-01 11:59:39 -04:00
|
|
|
<h3 className="h3">{props.title}</h3>
|
|
|
|
</NavLink>
|
2018-10-05 04:10:20 -04:00
|
|
|
<nav className="icon-buttons">
|
2018-10-01 11:59:39 -04:00
|
|
|
{
|
2018-10-04 17:50:33 -04:00
|
|
|
props.help?
|
2018-10-25 06:48:25 -04:00
|
|
|
<a className="icon-button help" title="Find out more" href={props.help}>
|
2018-11-13 05:44:12 -05:00
|
|
|
More info
|
2018-10-01 11:59:39 -04:00
|
|
|
</a>
|
|
|
|
: null
|
|
|
|
}
|
|
|
|
{
|
|
|
|
!props.inactive?
|
2018-10-04 17:50:33 -04:00
|
|
|
<NavLink className="icon-button edit" title="Edit data"
|
2018-11-29 17:00:53 -05:00
|
|
|
to={`/edit/${props.slug}/building/${props.building_id}.html`}>
|
2018-10-25 06:48:25 -04:00
|
|
|
Edit
|
2018-10-01 11:59:39 -04:00
|
|
|
<EditIcon />
|
|
|
|
</NavLink>
|
|
|
|
: null
|
|
|
|
}
|
2018-10-05 04:10:20 -04:00
|
|
|
</nav>
|
2018-10-01 11:59:39 -04:00
|
|
|
</header>
|
2018-10-03 16:47:49 -04:00
|
|
|
{ match? <dl className="data-list">{props.children}</dl> : null }
|
2018-09-30 19:32:24 -04:00
|
|
|
</section>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-10-03 16:47:49 -04:00
|
|
|
const DataEntry = (props) => (
|
|
|
|
<Fragment>
|
|
|
|
<dt>
|
|
|
|
{ props.title }
|
|
|
|
{ props.tooltip? <Tooltip text={ props.tooltip } /> : null }
|
|
|
|
</dt>
|
2018-10-25 08:49:24 -04:00
|
|
|
<dd>{(props.value != null)? props.value : '\u00A0'}</dd>
|
2018-10-03 16:47:49 -04:00
|
|
|
</Fragment>
|
|
|
|
);
|
2018-09-09 17:22:44 -04:00
|
|
|
|
2019-01-19 13:47:08 -05:00
|
|
|
const MultiDataEntry = (props) => {
|
|
|
|
let content;
|
|
|
|
|
|
|
|
if (props.value && props.value.length) {
|
|
|
|
content = <ul>{
|
|
|
|
props.value.map((item, index) => {
|
|
|
|
return <li key={index}><a href={sanitise_url(item)}>{item}</a></li>
|
|
|
|
})
|
|
|
|
}</ul>
|
|
|
|
} else {
|
|
|
|
content = '\u00A0'
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Fragment>
|
|
|
|
<dt>
|
|
|
|
{ props.title }
|
|
|
|
{ props.tooltip? <Tooltip text={ props.tooltip } /> : null }
|
|
|
|
</dt>
|
|
|
|
<dd>{ content }</dd>
|
|
|
|
</Fragment>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function sanitise_url(string){
|
|
|
|
let url_
|
|
|
|
|
|
|
|
// http or https
|
|
|
|
if (!(string.substring(0, 7) === 'http://' || string.substring(0, 8) === 'https://')){
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
url_ = document.createElement('a')
|
|
|
|
url_.href = string
|
|
|
|
} catch (error) {
|
|
|
|
try {
|
|
|
|
url_ = urlapi.parse(string)
|
|
|
|
} catch (error) {
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// required (www.example.com)
|
|
|
|
if (!url_.hostname || url_.hostname === '' || url_.hostname === 'localhost'){
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
|
|
|
|
// optional (/some/path)
|
|
|
|
// url_.pathname;
|
|
|
|
|
|
|
|
// optional (?name=value)
|
|
|
|
// url_.search;
|
|
|
|
|
|
|
|
// optional (#anchor)
|
|
|
|
// url_.hash;
|
|
|
|
|
|
|
|
return `${url_.protocol}//${url_.hostname}${url_.pathname || ''}${url_.search || ''}${url_.hash || ''}`
|
|
|
|
}
|
|
|
|
|
2018-10-25 09:36:52 -04:00
|
|
|
const UPRNsDataEntry = (props) => {
|
|
|
|
const uprns = props.value || [];
|
|
|
|
const no_parent = uprns.filter(uprn => uprn.parent_uprn == null);
|
|
|
|
const with_parent = uprns.filter(uprn => uprn.parent_uprn != null);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Fragment>
|
|
|
|
<dt>
|
|
|
|
{ props.title }
|
|
|
|
{ props.tooltip? <Tooltip text={ props.tooltip } /> : null }
|
|
|
|
</dt>
|
|
|
|
<dd><ul className="uprn-list">
|
|
|
|
<Fragment>{
|
|
|
|
no_parent.length?
|
|
|
|
no_parent.map(uprn => (
|
2018-10-25 09:46:07 -04:00
|
|
|
<li key={uprn.uprn}>{uprn.uprn}</li>
|
2018-10-25 09:36:52 -04:00
|
|
|
))
|
|
|
|
: '\u00A0'
|
|
|
|
}</Fragment>
|
|
|
|
{
|
|
|
|
with_parent.length?
|
|
|
|
<details>
|
|
|
|
<summary>Children</summary>
|
|
|
|
{
|
|
|
|
with_parent.map(uprn => (
|
2018-10-25 09:46:07 -04:00
|
|
|
<li key={uprn.uprn}>{uprn.uprn} (child of {uprn.parent_uprn})</li>
|
2018-10-25 09:36:52 -04:00
|
|
|
))
|
|
|
|
}
|
|
|
|
</details>
|
|
|
|
: null
|
|
|
|
}
|
|
|
|
</ul></dd>
|
|
|
|
</Fragment>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2018-09-09 17:22:44 -04:00
|
|
|
export default BuildingView;
|