Setup server to handle article page (untested)
This commit is contained in:
@ -36,13 +36,13 @@ module.exports = class Articles extends DatabaseInterface {
|
|||||||
|
|
||||||
async getArticleById(id) {
|
async getArticleById(id) {
|
||||||
return await this.store.getFromDatabase(
|
return await this.store.getFromDatabase(
|
||||||
`getArticleById_${id}`, `getArticleById`, {id}, 'one'
|
`getArticleById_${id}`, `getArticleById`, {id}, 'oneOrNone'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getArticleByHandle(handle) {
|
async getArticleByHandle(handle) {
|
||||||
return await this.store.getFromDatabase(
|
return await this.store.getFromDatabase(
|
||||||
`getArticleByHandle_${handle}`, `getArticleByHandle`, {handle}, 'one'
|
`getArticleByHandle_${handle}`, `getArticleByHandle`, {handle}, 'oneOrNone'
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
52
private/server/api/methods/blog/GetBlogArticle.js
Normal file
52
private/server/api/methods/blog/GetBlogArticle.js
Normal file
@ -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
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -42,7 +42,7 @@ export const get = async (url, params, test) => {
|
|||||||
if(process.env.NODE_ENV === 'development') {
|
if(process.env.NODE_ENV === 'development') {
|
||||||
console.log('testing mode');
|
console.log('testing mode');
|
||||||
return await new Promise((resolve,reject) => {
|
return await new Promise((resolve,reject) => {
|
||||||
setTimeout(e => resolve(test), 1000);
|
//setTimeout(e => resolve(test), 1000);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 73 KiB |
@ -38,6 +38,14 @@ const TestBlogs = {
|
|||||||
articles: [...Array(20).keys()].map(TestBlogArticle)
|
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 => {
|
export const withBlogTemplate = WrappedComponent => {
|
||||||
return class extends React.Component {
|
return class extends React.Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
@ -59,12 +67,7 @@ export const withBlogTemplate = WrappedComponent => {
|
|||||||
this.setState({ pending: true, page, perPage });
|
this.setState({ pending: true, page, perPage });
|
||||||
get('blog', { page, perPage }, TestBlogs).then(blog => {
|
get('blog', { page, perPage }, TestBlogs).then(blog => {
|
||||||
let { articles, pages } = blog;
|
let { articles, pages } = blog;
|
||||||
|
articles.forEach(NormalizeArticle);
|
||||||
articles.forEach(article => {
|
|
||||||
article.url = `/blogs/articles/${article.handle}`
|
|
||||||
article.image = require(`@assets/images/${article.image}`);
|
|
||||||
});
|
|
||||||
|
|
||||||
this.setState({ pending: undefined, error: undefined, articles, pages });
|
this.setState({ pending: undefined, error: undefined, articles, pages });
|
||||||
}).catch(e => {
|
}).catch(e => {
|
||||||
console.error(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 <WrappedComponent {...this.props} {...this.state} />;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -40,7 +40,8 @@ const AppRoutes = (props) => {
|
|||||||
<RouteWrapper exact path="/contact" page={ () => import ('@pages/contact/ContactPage') } />
|
<RouteWrapper exact path="/contact" page={ () => import ('@pages/contact/ContactPage') } />
|
||||||
<RouteWrapper exact path="/legal/privacy" page={ () => import('@pages/legal/privacy/PrivacyPolicyPage') } />
|
<RouteWrapper exact path="/legal/privacy" page={ () => import('@pages/legal/privacy/PrivacyPolicyPage') } />
|
||||||
|
|
||||||
<RouteWrapper exact path="/blog/:page?" page={ () => import('@pages/blog/BlogPage') } />
|
<RouteWrapper exact path="/blog/:page?" page={ ()=>import('@pages/blog/BlogPage') } />
|
||||||
|
<RouteWrapper exact path="/blog/article/:article?" page={ ()=>import('@pages/blog/article/ArticlePage') } />
|
||||||
</Routes>
|
</Routes>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
@import '~@styles/global';
|
@import '~@styles/global';
|
||||||
|
|
||||||
.c-page {
|
.c-page {
|
||||||
flex-grow: 1;
|
min-height: 80vh;
|
||||||
|
|
||||||
&.has-background {
|
&.has-background {
|
||||||
position: relative;
|
position: relative;
|
||||||
|
@ -44,7 +44,10 @@ export default withLanguage(props => {
|
|||||||
<article role="article" itemScope itemType="http://schema.org/Article" className="c-featured-article">
|
<article role="article" itemScope itemType="http://schema.org/Article" className="c-featured-article">
|
||||||
<ContentBox box className="c-featured-article__content">
|
<ContentBox box className="c-featured-article__content">
|
||||||
<NavLink to={ article.url } className="c-featured-article__box is-image">
|
<NavLink to={ article.url } className="c-featured-article__box is-image">
|
||||||
<Image src={ article.image } className="c-featured-article__image" maxWidth="800" />
|
<Image
|
||||||
|
src={ article.image } className="c-featured-article__image"
|
||||||
|
maxWidth="800" loadable
|
||||||
|
/>
|
||||||
</NavLink>
|
</NavLink>
|
||||||
|
|
||||||
<div className="c-featured-article__box is-content">
|
<div className="c-featured-article__box is-content">
|
||||||
|
@ -25,10 +25,6 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
"blog": {
|
"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": {
|
"article": {
|
||||||
"readMore": "Read More"
|
"readMore": "Read More"
|
||||||
}
|
}
|
||||||
@ -231,7 +227,19 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
"blog": {
|
"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."
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -39,9 +39,10 @@ import Styles from './BlogPage.scss';
|
|||||||
|
|
||||||
export default withBlogTemplate(withLanguage(props => {
|
export default withBlogTemplate(withLanguage(props => {
|
||||||
let { lang, articles, page, pages, pending, error } = props;
|
let { lang, articles, page, pages, pending, error } = props;
|
||||||
|
let l = lang.pages.blog;
|
||||||
|
|
||||||
let children;
|
let children;
|
||||||
if(error) error = <ErrorSection title={lang.blog.error.title} body={lang.blog.error.body} error={error} />;
|
if(error) error = <ErrorSection title={l.error.title} body={l.error.body} error={error} />;
|
||||||
if(pending) pending = <Loader />;
|
if(pending) pending = <Loader />;
|
||||||
|
|
||||||
if(articles && articles.length) {
|
if(articles && articles.length) {
|
||||||
@ -49,21 +50,23 @@ export default withBlogTemplate(withLanguage(props => {
|
|||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<FeaturedArticleSection article={ articles.shift() } />
|
<FeaturedArticleSection article={ articles.shift() } />
|
||||||
<ArticleGridSection articles={ articles } />
|
<ArticleGridSection articles={ articles } />
|
||||||
<Pagination page={ page } pages={ pages } to="/blog/$page" />
|
<Pagination
|
||||||
|
className="p-blog-page__pagination" page={ page } pages={ pages }
|
||||||
|
to="/blog/$page"
|
||||||
|
/>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Page
|
<Page
|
||||||
style="blog-page" className="p-blog-page" title={ lang.pages.blog.title }
|
style="blog-page" className="p-blog-page" title={ error ? l.error.title : l.title }
|
||||||
background={require('@assets/images/banners/sunset.svg')}
|
background={require('@assets/images/banners/sunset.svg')}
|
||||||
>
|
>
|
||||||
<ClearSection />
|
<ClearSection />
|
||||||
{ error }
|
{ error }
|
||||||
{ pending }
|
{ pending }
|
||||||
{ children }
|
{ children }
|
||||||
<ClearSection />
|
|
||||||
</Page>
|
</Page>
|
||||||
);
|
);
|
||||||
}));
|
}));
|
||||||
|
@ -6,4 +6,7 @@
|
|||||||
* 1.0.0 - 2018/10/30
|
* 1.0.0 - 2018/10/30
|
||||||
*/
|
*/
|
||||||
.p-blog-page {
|
.p-blog-page {
|
||||||
|
&__pagination {
|
||||||
|
padding: 6em 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,3 +22,75 @@
|
|||||||
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||||
|
|
||||||
import React from 'react';
|
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 = <ErrorSection title={l.error.title} body={l.error.body} error={error} />;
|
||||||
|
if(pending) pending = <Loader />;
|
||||||
|
|
||||||
|
if(article) {
|
||||||
|
children = (
|
||||||
|
<PageBoundary>
|
||||||
|
<article
|
||||||
|
role="article" itemScope itemType="http://schema.org/Article"
|
||||||
|
className="p-article-page__article"
|
||||||
|
>
|
||||||
|
{/* Title */}
|
||||||
|
<ContentBox box itemProp="name" className="p-article-page__header">
|
||||||
|
<Title children={ article.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>
|
||||||
|
);
|
||||||
|
}));
|
||||||
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -111,7 +111,7 @@ class ContactPage extends React.Component {
|
|||||||
style="contact-page"
|
style="contact-page"
|
||||||
className="p-contact-page"
|
className="p-contact-page"
|
||||||
title={ lang.pages.contact.title }
|
title={ lang.pages.contact.title }
|
||||||
background={ require('@assets/images/banners/sunset.svg') }
|
background={ require('@assets/images/banners/hills-night.svg') }
|
||||||
>
|
>
|
||||||
<ClearSection />
|
<ClearSection />
|
||||||
<PageBoundary small children={inners} />
|
<PageBoundary small children={inners} />
|
||||||
|
Reference in New Issue
Block a user