diff --git a/private/blog/Articles.js b/private/blog/Articles.js index 9fad0f8..e216b9a 100644 --- a/private/blog/Articles.js +++ b/private/blog/Articles.js @@ -36,13 +36,13 @@ module.exports = class Articles extends DatabaseInterface { async getArticleById(id) { return await this.store.getFromDatabase( - `getArticleById_${id}`, `getArticleById`, {id}, 'one' + `getArticleById_${id}`, `getArticleById`, {id}, 'oneOrNone' ); } async getArticleByHandle(handle) { return await this.store.getFromDatabase( - `getArticleByHandle_${handle}`, `getArticleByHandle`, {handle}, 'one' + `getArticleByHandle_${handle}`, `getArticleByHandle`, {handle}, 'oneOrNone' ); } diff --git a/private/server/api/methods/blog/GetBlogArticle.js b/private/server/api/methods/blog/GetBlogArticle.js new file mode 100644 index 0000000..10fbebb --- /dev/null +++ b/private/server/api/methods/blog/GetBlogArticle.js @@ -0,0 +1,52 @@ +// Copyright (c) 2018 Dominic Masters +// +// MIT License +// +// Permission is hereby granted, free of charge, to any person obtaining +// a copy of this software and associated documentation files (the +// "Software"), to deal in the Software without restriction, including +// without limitation the rights to use, copy, modify, merge, publish, +// distribute, sublicense, and/or sell copies of the Software, and to +// permit persons to whom the Software is furnished to do so, subject to +// the following conditions: +// +// The above copyright notice and this permission notice shall be +// included in all copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF +// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND +// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE +// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION +// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION +// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + +const + APIHandler = require('./../../APIHandler'), + sanitizeHtml = require('sanitize-html') +; + +const ERRORS = { + missingHandle: "Missing article handle.", + notFound: "Cannot find that article." +}; + +module.exports = class GetBlogArticle extends APIHandler { + constructor(api) { + super(api, ['GET'], '/blog/article'); + } + + async handle(request) { + if(!request.hasString('article', 128)) return { ok: false, data: ERRORS.missingHandle }; + + let handle = request.getApp().createHandle(request.getString('article', 128)); + let article = await request.getApp().getArticles().getArticleByHandle(handle); + + if(!article) return { ok: false, data: ERRORS.notFound }; + + return { + ok: true, + data: article + }; + } +} diff --git a/public/api/api.js b/public/api/api.js index 42e2f70..fded761 100644 --- a/public/api/api.js +++ b/public/api/api.js @@ -42,7 +42,7 @@ export const get = async (url, params, test) => { if(process.env.NODE_ENV === 'development') { console.log('testing mode'); return await new Promise((resolve,reject) => { - setTimeout(e => resolve(test), 1000); + //setTimeout(e => resolve(test), 1000); }); } diff --git a/public/assets/images/photo.jpg b/public/assets/images/photo.jpg deleted file mode 100644 index 6b05361..0000000 Binary files a/public/assets/images/photo.jpg and /dev/null differ diff --git a/public/blog/Blog.jsx b/public/blog/Blog.jsx index 6167cbf..40e3d6b 100644 --- a/public/blog/Blog.jsx +++ b/public/blog/Blog.jsx @@ -38,6 +38,14 @@ const TestBlogs = { articles: [...Array(20).keys()].map(TestBlogArticle) } +//Functions for normalization +const NormalizeArticle = article => { + article.url = `/blog/article/${article.handle}`; + article.image = require(`@assets/images/${article.image}`); + return article; +}; + +//Template Wrappers export const withBlogTemplate = WrappedComponent => { return class extends React.Component { constructor(props) { @@ -59,12 +67,7 @@ export const withBlogTemplate = WrappedComponent => { this.setState({ pending: true, page, perPage }); get('blog', { page, perPage }, TestBlogs).then(blog => { let { articles, pages } = blog; - - articles.forEach(article => { - article.url = `/blogs/articles/${article.handle}` - article.image = require(`@assets/images/${article.image}`); - }); - + articles.forEach(NormalizeArticle); this.setState({ pending: undefined, error: undefined, articles, pages }); }).catch(e => { console.error(e); @@ -77,3 +80,34 @@ export const withBlogTemplate = WrappedComponent => { } } }; + + +export const withArticleTemplate = WrappedComponent => { + return class extends React.Component { + constructor(props) { + super(props); + + this.state = { + pending: true, + error: undefined, + article: undefined + }; + } + + componentDidMount() { + let { article } = this.props.match.params; + this.setState({ pending: true }); + get('blog/article', { article }, TestBlogArticle(1)).then(article => { + NormalizeArticle(article); + this.setState({ pending: undefined, error: undefined, article }); + }).catch(error => { + console.error(error); + this.setState({ pending: undefined, error }); + }); + } + + render() { + return ; + } + } +} diff --git a/public/components/App.jsx b/public/components/App.jsx index a0a9eb2..cf203b8 100644 --- a/public/components/App.jsx +++ b/public/components/App.jsx @@ -40,7 +40,8 @@ const AppRoutes = (props) => { import ('@pages/contact/ContactPage') } /> import('@pages/legal/privacy/PrivacyPolicyPage') } /> - import('@pages/blog/BlogPage') } /> + import('@pages/blog/BlogPage') } /> + import('@pages/blog/article/ArticlePage') } /> ); }; diff --git a/public/components/page/Page.scss b/public/components/page/Page.scss index a634edc..4dbdef8 100644 --- a/public/components/page/Page.scss +++ b/public/components/page/Page.scss @@ -11,7 +11,7 @@ @import '~@styles/global'; .c-page { - flex-grow: 1; + min-height: 80vh; &.has-background { position: relative; diff --git a/public/components/section/blog/article/FeaturedArticleSection.jsx b/public/components/section/blog/article/FeaturedArticleSection.jsx index f89c87d..1fb3414 100644 --- a/public/components/section/blog/article/FeaturedArticleSection.jsx +++ b/public/components/section/blog/article/FeaturedArticleSection.jsx @@ -44,7 +44,10 @@ export default withLanguage(props => {
- +
diff --git a/public/language/en-AU.jsx b/public/language/en-AU.jsx index 697a8f5..d7cc5d9 100644 --- a/public/language/en-AU.jsx +++ b/public/language/en-AU.jsx @@ -25,10 +25,6 @@ export default { }, "blog": { - "error": { - "title": "Failed to get the blog", - "body": "Failed to get the blogs and articles from the server, please try again later or refresh your browser." - }, "article": { "readMore": "Read More" } @@ -231,7 +227,19 @@ export default { }, "blog": { - "title": "Blog" + "title": "Blog", + "error": { + "title": "Failed to get the blog", + "body": "Failed to get the blogs and articles from the server, please try again later or refresh your browser." + } + }, + + "article": { + "title": "Article", + "error": { + "title": "Failed to get article", + "body": "Failed to get the article from the server, please try again later or refresh your browser." + } } }, diff --git a/public/pages/blog/BlogPage.jsx b/public/pages/blog/BlogPage.jsx index 3f158de..f548928 100644 --- a/public/pages/blog/BlogPage.jsx +++ b/public/pages/blog/BlogPage.jsx @@ -39,9 +39,10 @@ import Styles from './BlogPage.scss'; export default withBlogTemplate(withLanguage(props => { let { lang, articles, page, pages, pending, error } = props; + let l = lang.pages.blog; let children; - if(error) error = ; + if(error) error = ; if(pending) pending = ; if(articles && articles.length) { @@ -49,21 +50,23 @@ export default withBlogTemplate(withLanguage(props => { - + ); } return ( { error } { pending } { children } - ); })); diff --git a/public/pages/blog/BlogPage.scss b/public/pages/blog/BlogPage.scss index 4b01acc..bfaf635 100644 --- a/public/pages/blog/BlogPage.scss +++ b/public/pages/blog/BlogPage.scss @@ -6,4 +6,7 @@ * 1.0.0 - 2018/10/30 */ .p-blog-page { + &__pagination { + padding: 6em 0; + } } diff --git a/public/pages/blog/article/ArticlePage.jsx b/public/pages/blog/article/ArticlePage.jsx index d93ee36..80f6a92 100644 --- a/public/pages/blog/article/ArticlePage.jsx +++ b/public/pages/blog/article/ArticlePage.jsx @@ -22,3 +22,75 @@ // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. import React from 'react'; + +import { withLanguage } from '@public/language/Language'; +import { withArticleTemplate } from '@public/blog/Blog'; + +import Page, { PageBoundary } from '@components/page/Page'; + +import ErrorSection from '@sections/error/ErrorSection'; +import ClearSection from '@sections/layout/ClearSection'; + +import Loader from '@objects/loading/Loader'; +import ContentBox from '@objects/content/box/ContentBox'; +import { Title, Paragraph } from '@objects/typography/typography'; +import Image from '@objects/image/Image'; + +import Styles from './ArticlePage.scss'; + +export default withArticleTemplate(withLanguage(props => { + let { error, pending, article, lang } = props; + let l = lang.pages.article; + + + let children; + if(error) error = ; + if(pending) pending = ; + + if(article) { + children = ( + +
+ {/* Title */} + + + </ContentBox> + + {/* Image */} + <div className="p-article-page__picture"> + <ContentBox box> + <Image + src={ article.image } maxWidth="800" loadable + className="p-article-page__picture-image" + /> + </ContentBox> + </div> + + {/* Description */} + <ContentBox box itemProp="description" className="p-article-page__description"> + <Paragraph> + { article.description || article.shortDescription } + </Paragraph> + </ContentBox> + </article> + </PageBoundary> + ); + } + + return ( + <Page + style="article-page" className="p-article-page" + title={error ? l.error.title : l.title} + background={require('@assets/images/banners/sunset.svg')} + > + <ClearSection /> + { error } + { pending } + { children } + <ClearSection /> + </Page> + ); +})); diff --git a/public/pages/blog/article/ArticlePage.scss b/public/pages/blog/article/ArticlePage.scss index e69de29..b655ba5 100644 --- a/public/pages/blog/article/ArticlePage.scss +++ b/public/pages/blog/article/ArticlePage.scss @@ -0,0 +1,43 @@ +@import '~@styles/global'; + +$p-article-page--spacing: 1.5em; + +.p-article-page { + &__article { + display: flex; + flex-wrap: wrap; + align-content: stretch; + } + + &__header, + &__picture, + &__description + { + width: 100%; + } + + &__picture { + position: relative; + + &-image {width: 100%;} + &-box { + position: sticky; + top:0; + } + } + + @include t-media-query($s-small-up) { + position: relative; + &__header { + margin-bottom: $p-article-page--spacing; + } + &__picture { + width: 40%; + padding-right: $p-article-page--spacing; + } + &__description { + width: 60%; + padding-left: $p-article-page--spacing; + } + } +} diff --git a/public/pages/contact/ContactPage.jsx b/public/pages/contact/ContactPage.jsx index 6fae912..5e35de6 100644 --- a/public/pages/contact/ContactPage.jsx +++ b/public/pages/contact/ContactPage.jsx @@ -111,7 +111,7 @@ class ContactPage extends React.Component { style="contact-page" className="p-contact-page" title={ lang.pages.contact.title } - background={ require('@assets/images/banners/sunset.svg') } + background={ require('@assets/images/banners/hills-night.svg') } > <ClearSection /> <PageBoundary small children={inners} />