├── README.md
├── .firebaserc
├── public
├── 192x192.png
├── 384x384.png
├── 512x512.png
├── favicon.ico
├── index.html
└── manifest.json
├── src
├── App.test.js
├── reducers
│ ├── index.js
│ ├── story.js
│ └── comment.js
├── const
│ └── index.js
├── Default.js
├── App.css
├── actions
│ └── comment.js
├── components
│ ├── CommentItem.js
│ ├── Item.js
│ └── Comment.js
├── Jobs.js
├── Ask.js
├── Show.js
├── Newest.js
├── App.js
├── utils
│ └── scroll.js
├── Story.js
├── Container.js
├── registerServiceWorker.js
├── logo.svg
├── index.css
├── index.js
└── public
│ └── loading.js
├── firebase.json
├── .gitignore
└── package.json
/README.md:
--------------------------------------------------------------------------------
1 | # hackerNews-pwa
2 | hacker news pwa
3 |
--------------------------------------------------------------------------------
/.firebaserc:
--------------------------------------------------------------------------------
1 | {
2 | "projects": {
3 | "default": "hn-pwa-d8b2e"
4 | }
5 | }
6 |
--------------------------------------------------------------------------------
/public/192x192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stephenkingsley/hackerNews-pwa/HEAD/public/192x192.png
--------------------------------------------------------------------------------
/public/384x384.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stephenkingsley/hackerNews-pwa/HEAD/public/384x384.png
--------------------------------------------------------------------------------
/public/512x512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stephenkingsley/hackerNews-pwa/HEAD/public/512x512.png
--------------------------------------------------------------------------------
/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stephenkingsley/hackerNews-pwa/HEAD/public/favicon.ico
--------------------------------------------------------------------------------
/src/App.test.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import App from './App';
4 |
5 | it('renders without crashing', () => {
6 | const div = document.createElement('div');
7 | ReactDOM.render(, div);
8 | });
9 |
--------------------------------------------------------------------------------
/src/reducers/index.js:
--------------------------------------------------------------------------------
1 | import { combineReducers } from 'redux';
2 | import comment from './comment';
3 | import story from './story';
4 |
5 | const rootReducer = combineReducers({
6 | comment,
7 | story,
8 | });
9 |
10 | export default rootReducer;
11 |
--------------------------------------------------------------------------------
/src/const/index.js:
--------------------------------------------------------------------------------
1 | export const NEWS_LIST = 'NEWS_LIST';
2 | export const SHOW_LIST = 'SHOW_LIST';
3 | export const ASK_LIST = 'ASK_LIST';
4 | export const NEWEST_LIST = 'NEWEST_LIST';
5 | export const JOBS_LIST = 'JOBS_LIST';
6 |
7 | export const STORY = 'STORY';
8 |
--------------------------------------------------------------------------------
/src/Default.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { Redirect } from 'react-router-dom';
3 |
4 | class Default extends Component {
5 | render() {
6 | return
7 | }
8 | }
9 |
10 | export default Default;
11 |
--------------------------------------------------------------------------------
/firebase.json:
--------------------------------------------------------------------------------
1 | {
2 | "hosting": {
3 | "public": "build",
4 | "rewrites": [{
5 | "source": "**",
6 | "destination": "/index.html"
7 | }],
8 | "ignore": [
9 | "firebase.json",
10 | "**/.*",
11 | "**/node_modules/**"
12 | ]
13 | }
14 | }
15 |
--------------------------------------------------------------------------------
/src/reducers/story.js:
--------------------------------------------------------------------------------
1 | import { STORY } from '../const';
2 |
3 | const initState = {};
4 |
5 | export default (state = initState, action) => {
6 | switch (action.type) {
7 | case STORY:
8 | return {
9 | state: action.data,
10 | };
11 | default:
12 | return state;
13 | }
14 | };
15 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/ignore-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 |
6 | # testing
7 | /coverage
8 |
9 | # production
10 | /build
11 |
12 | # misc
13 | .DS_Store
14 | .env.local
15 | .env.development.local
16 | .env.test.local
17 | .env.production.local
18 |
19 | npm-debug.log*
20 | yarn-debug.log*
21 | yarn-error.log*
22 |
--------------------------------------------------------------------------------
/src/App.css:
--------------------------------------------------------------------------------
1 | .App {
2 | text-align: center;
3 | }
4 |
5 | .App-logo {
6 | animation: App-logo-spin infinite 20s linear;
7 | height: 80px;
8 | }
9 |
10 | .App-header {
11 | background-color: #222;
12 | height: 150px;
13 | padding: 20px;
14 | color: white;
15 | }
16 |
17 | .App-intro {
18 | font-size: large;
19 | }
20 |
21 | @keyframes App-logo-spin {
22 | from { transform: rotate(0deg); }
23 | to { transform: rotate(360deg); }
24 | }
25 |
--------------------------------------------------------------------------------
/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 | HNPWA
10 |
11 |
12 |
15 |
16 |
17 |
18 |
--------------------------------------------------------------------------------
/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "short_name": "HN PWA",
3 | "name": "Hacker News PWA",
4 | "icons": [
5 | {
6 | "src": "favicon.ico",
7 | "sizes": "192x192",
8 | "type": "image/png"
9 | },
10 | {
11 | "src": "384x384.png",
12 | "sizes": "384x384",
13 | "type": "image/png"
14 | },
15 | {
16 | "src": "512x512.png",
17 | "sizes": "512x512",
18 | "type": "image/png"
19 | }
20 | ],
21 | "start_url": "./index.html",
22 | "display": "standalone",
23 | "theme_color": "#000000",
24 | "background_color": "#ffffff"
25 | }
26 |
--------------------------------------------------------------------------------
/src/actions/comment.js:
--------------------------------------------------------------------------------
1 | export const getData = (type, page) =>
2 | dispatch => {
3 | fetch(`https://node-hnapi.herokuapp.com/${type}?page=${page}`)
4 | .then(res => res.json())
5 | .then(data => {
6 | dispatch(receiveData(`${type.toUpperCase()}_LIST`, data));
7 | })
8 | };
9 |
10 | export const getItem = (id) =>
11 | dispatch => {
12 | fetch(`https://node-hnapi.herokuapp.com/item/${id}`)
13 | .then(res => res.json())
14 | .then(data => {
15 | dispatch(receiveData('STORY', data));
16 | })
17 | };
18 |
19 | const receiveData = (type, data) => {
20 | return {
21 | type, data,
22 | }
23 | };
24 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "hacker-news-pwa",
3 | "version": "0.1.3",
4 | "private": true,
5 | "dependencies": {
6 | "react": "^16.5.2",
7 | "react-dom": "^16.5.2",
8 | "react-redux": "^5.0.7",
9 | "react-router": "^4.3.1",
10 | "react-router-dom": "^4.3.1",
11 | "redux": "^3.6.0",
12 | "redux-thunk": "^2.3.0"
13 | },
14 | "devDependencies": {
15 | "react-scripts": "2.0.5"
16 | },
17 | "scripts": {
18 | "start": "react-scripts start",
19 | "build": "react-scripts build",
20 | "deploy": "firebase deploy --only hosting",
21 | "push": "npm run build && npm run deploy",
22 | "test": "react-scripts test --env=jsdom",
23 | "eject": "react-scripts eject"
24 | },
25 | "browserslist": [
26 | ">0.2%",
27 | "not dead",
28 | "not ie <= 11",
29 | "not op_mini all"
30 | ]
31 | }
32 |
--------------------------------------------------------------------------------
/src/components/CommentItem.js:
--------------------------------------------------------------------------------
1 | import React, { Component } from 'react';
2 |
3 | export default (props) => {
4 | return (
5 | reply(props.id)}
11 | >
12 |
17 | {props.user}
18 |
19 |
20 |
25 | {props.timeAgo}
26 |
27 |
30 |
31 | );
32 | };
33 |
34 | const reply = (id) => {
35 | window.location.href = `https://news.ycombinator.com/reply?id=${id}`;
36 | };
37 |
--------------------------------------------------------------------------------
/src/reducers/comment.js:
--------------------------------------------------------------------------------
1 | import { NEWS_LIST, SHOW_LIST, ASK_LIST, NEWEST_LIST, JOBS_LIST } from '../const';
2 |
3 | const initState = {
4 | news: [],
5 | show: [],
6 | ask: [],
7 | newest: [],
8 | jobs: [],
9 | };
10 |
11 | export default (state = initState, action) => {
12 | switch (action.type) {
13 | case NEWS_LIST:
14 | return {
15 | ...state,
16 | news: action.data,
17 | };
18 | case SHOW_LIST:
19 | return {
20 | ...state,
21 | show: action.data,
22 | };
23 | case ASK_LIST:
24 | return {
25 | ...state,
26 | ask: action.data,
27 | };
28 | case NEWEST_LIST:
29 | return {
30 | ...state,
31 | newest: action.data,
32 | };
33 | case JOBS_LIST:
34 | return {
35 | ...state,
36 | jobs: action.data,
37 | };
38 | default:
39 | return state;
40 | }
41 | };
42 |
--------------------------------------------------------------------------------
/src/Jobs.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { connect } from 'react-redux';
3 | import { getData } from './actions/comment';
4 | import Scroll from './utils/scroll';
5 | import Container from './Container';
6 |
7 | class Jobs extends Container {
8 | componentWillMount() {
9 | const page = this.props.match.params.page;
10 | this.props.dispatch(getData('jobs', page));
11 | }
12 |
13 | componentWillReceiveProps(nextProps) {
14 | const newPage = nextProps.match.params.page;
15 | const page = this.props.match.params.page;
16 | if (newPage !== page) {
17 | this.props.dispatch(getData('jobs', newPage));
18 | }
19 | }
20 |
21 | render() {
22 | const { jobs } = this.props;
23 | return
24 | {this.renderList(jobs)}
25 | {this.renderPage('jobs')}
26 |
;
27 | }
28 | }
29 |
30 | const mapStateToProps = state => state.comment;
31 |
32 | export default connect(mapStateToProps)(Jobs);
33 |
--------------------------------------------------------------------------------
/src/Ask.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { connect } from 'react-redux';
3 | import { getData } from './actions/comment';
4 | import Scroll from './utils/scroll';
5 | import Container from './Container';
6 | import Item from './components/Item';
7 |
8 | class Ask extends Container {
9 | componentWillMount() {
10 | const page = this.props.match.params.page;
11 | this.props.dispatch(getData('ask', page));
12 | }
13 |
14 | componentWillReceiveProps(nextProps) {
15 | const newPage = nextProps.match.params.page;
16 | const page = this.props.match.params.page;
17 | if (newPage !== page) {
18 | this.props.dispatch(getData('ask', newPage));
19 | }
20 | }
21 |
22 | render() {
23 | const { ask } = this.props;
24 | return
25 | {this.renderList(ask)}
26 | {this.renderPage('ask')}
27 |
;
28 | }
29 | }
30 |
31 | const mapStateToProps = state => state.comment;
32 |
33 | export default connect(mapStateToProps)(Ask);
34 |
--------------------------------------------------------------------------------
/src/Show.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { connect } from 'react-redux';
3 | import { getData } from './actions/comment';
4 | import Scroll from './utils/scroll';
5 | import Container from './Container';
6 |
7 | class Show extends Container {
8 | componentWillMount() {
9 | const page = this.props.match.params.page;
10 | this.props.dispatch(getData('show', page));
11 | }
12 |
13 | componentWillReceiveProps(nextProps) {
14 | const newPage = nextProps.match.params.page;
15 | const page = this.props.match.params.page;
16 | if (newPage !== page) {
17 | this.props.dispatch(getData('show', newPage));
18 | }
19 | }
20 |
21 | render() {
22 | const { show, match } = this.props;
23 | const page = Number(match.params.page);
24 | return
25 | {this.renderList(show)}
26 | {this.renderPage('show')}
27 |
;
28 | }
29 | }
30 |
31 | const mapStateToProps = state => state.comment;
32 |
33 | export default connect(mapStateToProps)(Show);
34 |
--------------------------------------------------------------------------------
/src/Newest.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { connect } from 'react-redux';
3 | import { getData } from './actions/comment';
4 | import Scroll from './utils/scroll';
5 | import Container from './Container';
6 |
7 | class Newest extends Container {
8 | componentWillMount() {
9 | const page = this.props.match.params.page;
10 | this.props.dispatch(getData('newest', page));
11 | }
12 |
13 | componentWillReceiveProps(nextProps) {
14 | const newPage = nextProps.match.params.page;
15 | const page = this.props.match.params.page;
16 | if (newPage !== page) {
17 | this.props.dispatch(getData('newest', newPage));
18 | }
19 | }
20 |
21 | render() {
22 | const { newest, match } = this.props;
23 | const page = Number(match.params.page);
24 | return
25 | {this.renderList(newest)}
26 | {this.renderPage('newest')}
27 |
;
28 | }
29 | }
30 |
31 | const mapStateToProps = state => state.comment;
32 |
33 | export default connect(mapStateToProps)(Newest);
34 |
--------------------------------------------------------------------------------
/src/App.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { connect } from 'react-redux';
3 | import clone from 'lodash/clone';
4 | import { getData } from './actions/comment';
5 | import Scroll from './utils/scroll';
6 | import Container from './Container';
7 | import Item from './components/Item';
8 |
9 | class App extends Container {
10 | componentWillMount() {
11 | const page = this.props.match.params.page;
12 | this.props.dispatch(getData('news', page));
13 | }
14 |
15 | componentWillReceiveProps(nextProps, nextState, context) {
16 | const newPage = nextProps.match.params.page;
17 | const page = this.props.match.params.page;
18 | if (newPage !== page) {
19 | this.props.dispatch(getData('news', newPage));
20 | }
21 | }
22 |
23 | render() {
24 | console.log(this.props);
25 | const { news } = this.props;
26 | return
27 | {this.renderList(news)}
28 | {this.renderPage()}
29 |
;
30 | }
31 | }
32 |
33 | const mapStateToProps = state => {
34 | console.log(state);
35 | return state.comment
36 | };
37 |
38 | export default connect(mapStateToProps)(App);
39 |
--------------------------------------------------------------------------------
/src/components/Item.js:
--------------------------------------------------------------------------------
1 | import React, { Component } from 'react';
2 | import {
3 | Link,
4 | Redirect,
5 | } from 'react-router-dom'
6 |
7 | export default class Item extends Component {
8 | jumpUrl(event, url) {
9 | event.preventDefault();
10 | try {
11 | const newUrl = new URL(url);
12 | // window.location.href = url;
13 | window.open(url);
14 | } catch (e) {
15 | const id = url.match(/id=([0-9]*)/, 'g');
16 | }
17 | }
18 |
19 | render() {
20 | const { data, index, page } = this.props;
21 | return (
22 |
23 |
26 |
27 |
28 |
29 | { data.points ? `${data.points} by ${data.user} |` : ' ' }
30 | {' ' + data.time_ago} |
31 | {data.comments_count > 0 ? ` ${data.comments_count} comments` : ' discuss'}
32 |
33 |
34 |
35 |
36 | );
37 | }
38 | }
39 |
--------------------------------------------------------------------------------
/src/components/Comment.js:
--------------------------------------------------------------------------------
1 | import React, { Component } from 'react';
2 | import CommentItem from './CommentItem';
3 |
4 | export default class Comment extends Component {
5 | constructor() {
6 | super();
7 | this.element = [];
8 | }
9 |
10 | renderChildren(data, type = 1) {
11 | if (data.comments) {
12 | data.comments.forEach((ele, key) => {
13 | this.element.push(
14 |
22 | );
23 | this.renderChildren(ele, key + 2);
24 | });
25 | }
26 | }
27 |
28 | renderComponent(data) {
29 | this.element = [
30 |
37 | ];
38 | this.renderChildren(data);
39 | return this.element;
40 | }
41 |
42 | render() {
43 | return (
44 |
45 | {this.renderComponent(this.props.data)}
46 |
47 | );
48 | }
49 | }
50 |
51 | //
52 |
--------------------------------------------------------------------------------
/src/utils/scroll.js:
--------------------------------------------------------------------------------
1 | let isRunning = false;
2 | /**
3 | * @parma {func} callback - 执行的回调,一定要是promise
4 | * @parma {object} options - 执行scroll事件的一些配置
5 | * @parma {num} options.bottom - 当滑到距离底部的距离时执行回调
6 | */
7 | export default (cb, options = {}) => {
8 | const loadMore = () => {
9 | window.requestAnimationFrame(() => {
10 | let bottom = getWindowHeight() / 3;
11 | if (options && Object.hasOwnProperty.call(options, 'bottom')) {
12 | bottom = options.bottom;
13 | }
14 |
15 | if (getScrollTop() + getWindowHeight() >= getScrollHeight() - bottom) {
16 | cb();
17 | }
18 | });
19 | };
20 | window.addEventListener('scroll', loadMore, false);
21 | };
22 |
23 | const getScrollTop = () => {
24 | let scrollTop = 0;
25 | let bodyScrollTop = 0;
26 | let documentScrollTop = 0;
27 | if (document.body) {
28 | bodyScrollTop = document.body.scrollTop;
29 | }
30 | if (document.documentElement) {
31 | documentScrollTop = document.documentElement.scrollTop;
32 | }
33 | scrollTop = (bodyScrollTop - documentScrollTop > 0) ? bodyScrollTop : documentScrollTop;
34 | return scrollTop;
35 | };
36 |
37 | const getScrollHeight = () => {
38 | let scrollHeight = 0;
39 | let bodyScrollHeight = 0;
40 | let documentScrollHeight = 0;
41 | if (document.body) {
42 | bodyScrollHeight = document.body.scrollHeight;
43 | }
44 | if (document.documentElement) {
45 | documentScrollHeight = document.documentElement.scrollHeight;
46 | }
47 | scrollHeight = (bodyScrollHeight - documentScrollHeight > 0) ?
48 | bodyScrollHeight : documentScrollHeight;
49 | return scrollHeight;
50 | };
51 |
52 | const getWindowHeight = () => {
53 | let windowHeight = 0;
54 | if (document.compatMode === 'CSS1Compat') {
55 | windowHeight = document.documentElement.clientHeight;
56 | } else {
57 | windowHeight = document.body.clientHeight;
58 | }
59 | return windowHeight;
60 | };
61 |
--------------------------------------------------------------------------------
/src/Story.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { connect } from 'react-redux';
3 | import { getItem } from './actions/comment';
4 | import Comment from './components/Comment';
5 |
6 | class Story extends Component {
7 | componentWillMount() {
8 | const id = this.props.match.params.id;
9 | this.props.dispatch(getItem(id));
10 | }
11 |
12 | reply(id) {
13 | window.location.href = `https://news.ycombinator.com/item?id=${id}`;
14 | }
15 |
16 | renderDescription(data) {
17 | if (data && Object.hasOwnProperty.call(data, 'title')) {
18 | return (
19 |
20 |
23 |
this.reply(data.id)}>
24 | - points: {data.points}
25 | - comments_count: {data.comments_count}
26 | - time_ago: {data.time_ago}
27 | -
33 | reply
34 |
35 |
36 |
37 | );
38 | }
39 | }
40 |
41 | renderComments(data) {
42 | if (data && Object.hasOwnProperty.call(data, 'comments') && data.comments.length > 0) {
43 | return (
44 |
45 | {data.comments.map((ele, key) =>
46 | )}
47 |
48 | );
49 | }
50 | return null;
51 | }
52 |
53 | render() {
54 | const { state } = this.props;
55 | return (
56 |
57 | {this.renderDescription(state)}
58 | {this.renderComments(state)}
59 |
60 | );
61 | }
62 | }
63 |
64 | const mapStateToProps = state => state.story;
65 |
66 | export default connect(mapStateToProps)(Story);
67 |
--------------------------------------------------------------------------------
/src/Container.js:
--------------------------------------------------------------------------------
1 | import React, { Component, PropTypes } from 'react';
2 | import { Link } from 'react-router-dom';
3 | import Item from './components/Item';
4 | import { loadingPic } from './public/loading';
5 |
6 | export default class Container extends Component {
7 | constructor() {
8 | super();
9 | this.state = {
10 | page: 1,
11 | };
12 | }
13 |
14 | componentDidMount() {
15 | window.scrollTo(0, 0);
16 | }
17 |
18 | scrollTop() {
19 | window.scrollTo(0, 0);
20 | }
21 |
22 | renderItem(data) {
23 | if (data && data.length > 0) {
24 | return data.map((ele, index) => (
25 |
26 | ));
27 | }
28 | }
29 |
30 | renderLoading() {
31 | const { match } = this.props;
32 | const page = Number(match.params.page);
33 | let element;
34 | if (page > 1) {
35 | element = (
36 |
37 | content is null, please read prev page!
38 |
39 | );
40 | } else {
41 | element = (
42 |
43 |

44 |
45 | );
46 | }
47 | return element;
48 | }
49 |
50 | renderList(data) {
51 | return (
52 |
53 | {
54 | (data && data.length) > 0 ?
55 |
56 | {this.renderItem(data)}
57 |
:
58 | this.renderLoading()
59 | }
60 |
61 | );
62 | }
63 |
64 | renderPage(type) {
65 | const { match } = this.props;
66 | const page = Number(match.params.page);
67 | let newTypePrve;
68 | let newTypeNext;
69 | if (type) {
70 | newTypePrve = `/${type}/${(page - 1) > 1 ? page - 1 : 1}`
71 | newTypeNext = `/${type}/${page + 1}`;
72 | } else {
73 | newTypePrve = `/${(page - 1) > 1 ? page - 1 : 1}`
74 | newTypeNext = `/${page + 1}`;
75 | }
76 | return (
77 |
78 | this.scrollTop()}>{'< prev --- '}
79 | {page}
80 | this.scrollTop()}>{' --- next >'}
81 |
82 | );
83 | }
84 | }
85 |
--------------------------------------------------------------------------------
/src/registerServiceWorker.js:
--------------------------------------------------------------------------------
1 | // In production, we register a service worker to serve assets from local cache.
2 |
3 | // This lets the app load faster on subsequent visits in production, and gives
4 | // it offline capabilities. However, it also means that developers (and users)
5 | // will only see deployed updates on the "N+1" visit to a page, since previously
6 | // cached resources are updated in the background.
7 |
8 | // To learn more about the benefits of this model, read https://goo.gl/KwvDNy.
9 | // This link also includes instructions on opting out of this behavior.
10 |
11 | export default function register() {
12 | if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) {
13 | window.addEventListener('load', () => {
14 | const swUrl = `${process.env.PUBLIC_URL}/service-worker.js`;
15 | navigator.serviceWorker
16 | .register(swUrl)
17 | .then(registration => {
18 | registration.onupdatefound = () => {
19 | const installingWorker = registration.installing;
20 | installingWorker.onstatechange = () => {
21 | if (installingWorker.state === 'installed') {
22 | if (navigator.serviceWorker.controller) {
23 | // At this point, the old content will have been purged and
24 | // the fresh content will have been added to the cache.
25 | // It's the perfect time to display a "New content is
26 | // available; please refresh." message in your web app.
27 | console.log('New content is available; please refresh.');
28 | } else {
29 | // At this point, everything has been precached.
30 | // It's the perfect time to display a
31 | // "Content is cached for offline use." message.
32 | console.log('Content is cached for offline use.');
33 | }
34 | }
35 | };
36 | };
37 | })
38 | .catch(error => {
39 | console.error('Error during service worker registration:', error);
40 | });
41 | });
42 | }
43 | }
44 |
45 | export function unregister() {
46 | if ('serviceWorker' in navigator) {
47 | navigator.serviceWorker.ready.then(registration => {
48 | registration.unregister();
49 | });
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/src/logo.svg:
--------------------------------------------------------------------------------
1 |
8 |
--------------------------------------------------------------------------------
/src/index.css:
--------------------------------------------------------------------------------
1 | body {
2 | margin: 0;
3 | padding: 0;
4 | font-family: sans-serif;
5 | }
6 |
7 | a {
8 | text-decoration: none;
9 | max-width: 100%;
10 | word-break: break-all;
11 | }
12 |
13 | a:link {
14 | color: #333333;
15 | }
16 | a:active {
17 | color: #333333;
18 | }
19 | a:visited {
20 | color: #333333;
21 | }
22 | a:hover {
23 | color: #333333;
24 | }
25 |
26 | ul {
27 | list-style: none;
28 | }
29 |
30 | ol {
31 | display: block;
32 | list-style-type: decimal;
33 | -webkit-margin-before: 1em;
34 | -webkit-margin-after: 1em;
35 | -webkit-margin-start: 0px;
36 | -webkit-margin-end: 0px;
37 | -webkit-padding-start: 40px;
38 | }
39 |
40 | .topbar {
41 | position: fixed;
42 | width: 100%;
43 | top: 0;
44 | z-index: 100;
45 | height: 48px;
46 | padding: 0 16px;
47 | background-color: #222;
48 | display: flex;
49 | align-items: center;
50 | }
51 |
52 | .topbar > ul {
53 | list-style: none;
54 | display: flex;
55 | align-items: center;
56 | height: 100%;
57 | max-width: 1200px;
58 | overflow: hidden;
59 | margin-left: -20px;
60 | }
61 |
62 | .topbar > ul > li {
63 | margin-right: .6rem;
64 | color: #FFFFFF;
65 | }
66 |
67 | .topbar > ul > li > a {
68 | color: #FFFFFF;
69 | text-decoration: none;
70 | }
71 |
72 | .content {
73 | margin-top: 48px;
74 | }
75 |
76 | .content li {
77 | display: flex;
78 | flex-wrap: wrap;
79 | align-items: center;
80 | align-items: center;
81 | margin-left: -1.2rem;
82 | width: 100%;
83 | overflow: hidden;
84 | height: 4rem;
85 | border-bottom: 1px solid #ebebeb;
86 | padding: 0.3rem 0;
87 | }
88 |
89 | .content li div {
90 | width: 100%;
91 | }
92 |
93 | .content li > div:last-child {
94 | color: #999999;
95 | font-size: 0.8rem;
96 | }
97 |
98 | .item-footer {
99 | text-decoration: underline;
100 | }
101 |
102 | .story-top {
103 | border-bottom: 1px solid #ebebeb;
104 | }
105 |
106 | .story-title {
107 | font-size: 1.2rem;
108 | margin-left: 1.2rem;
109 | padding-top: 1rem;
110 | font-weight: bold;
111 | }
112 |
113 | .story-ol > li {
114 | height: 1.2rem;
115 | border-bottom: 0px;
116 | padding: 0px;
117 | color: #999999;
118 | }
119 |
120 | .comment-item {
121 | border-bottom: 1px solid #ebebeb;
122 | padding: 1rem;
123 | font-size: 0.8rem;
124 | max-width: 100%;
125 | }
126 |
127 | .footer-page {
128 | width: 100%;
129 | height: 32px;
130 | text-align: center;
131 | line-height: 16px;
132 | }
133 |
134 | .loading {
135 | height: 100%;
136 | width: 100%;
137 | display: flex;
138 | justify-content: center;
139 | align-items: center;
140 | }
141 |
142 | .loading > img {
143 | width: 200px;
144 | height: 200px;
145 | }
146 |
147 | .one-line-ellipsis {
148 | overflow: hidden;
149 | text-overflow: ellipsis;
150 | white-space: nowrap;
151 | }
152 |
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | import querystring from 'querystring';
2 |
3 | import React from 'react';
4 | import ReactDOM from 'react-dom';
5 |
6 | import { createStore, applyMiddleware } from 'redux'
7 | import { Provider } from 'react-redux'
8 | import thunk from 'redux-thunk'
9 | import reducer from './reducers'
10 |
11 | import {
12 | BrowserRouter as Router,
13 | Route,
14 | Link,
15 | Redirect
16 | } from 'react-router-dom'
17 |
18 | import Default from './Default';
19 | import App from './App';
20 | import Newest from './Newest';
21 | import Show from './Show';
22 | import Ask from './Ask';
23 | import Jobs from './Jobs';
24 | import Story from './Story';
25 |
26 | import registerServiceWorker from './registerServiceWorker';
27 |
28 | import './index.css';
29 |
30 | const middleware = [ thunk ]
31 |
32 | const store = createStore(
33 | reducer,
34 | applyMiddleware(...middleware)
35 | )
36 |
37 | const defaultChannelList = [
38 | 'news', 'newest', 'show', 'ask', 'jobs'
39 | ];
40 |
41 | const changeChannel = (event) => {
42 | const channelList = document.querySelectorAll('.channel > a');
43 | if (channelList && channelList.length > 0) {
44 | for (let i = 0; i < channelList.length; i += 1) {
45 | channelList[i].setAttribute('style', '');
46 | }
47 | }
48 | event.target.setAttribute('style', 'border-bottom: 4px solid rgb(255, 102, 0)');
49 | };
50 |
51 | const initChannel = () => {
52 | const path = window.location.pathname.match(/\/([a-z]*)\/([0-9]*)/, 'g');
53 | let channel = 'news';
54 | if (path) {
55 | channel = path[1];
56 | }
57 | const index = defaultChannelList.indexOf(channel);
58 | const channelList = document.querySelectorAll('.channel > a');
59 | index > -1 && channelList[index].setAttribute('style', 'border-bottom: 4px solid rgb(255, 102, 0)');
60 | };
61 |
62 | ReactDOM.render(
63 |
64 |
65 |
66 |
67 |

75 |
76 | - changeChannel(event)}>NEWS
77 | - changeChannel(event)}>NEW
78 | - changeChannel(event)}>SHOW
79 | - changeChannel(event)}>ASK
80 | - changeChannel(event)}>JOBS
81 |
82 |
83 |
84 |
85 |
86 |
87 |
88 |
89 |
90 |
91 |
92 |
93 | , document.getElementById('root'));
94 |
95 | window.addEventListener('popstate', () => initChannel());
96 |
97 | initChannel();
98 | registerServiceWorker();
99 |
--------------------------------------------------------------------------------
/src/public/loading.js:
--------------------------------------------------------------------------------
1 | export const loadingPic = '';
2 |
--------------------------------------------------------------------------------