├── 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 |
    24 | this.jumpUrl(event,data.url)} target="_blank">{index + 1 + (page - 1) * 30}.{data.title} 25 |
    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 |
    21 | {data.title} 22 |
    23 |
      this.reply(data.id)}> 24 |
    1. points: {data.points}
    2. 25 |
    3. comments_count: {data.comments_count}
    4. 26 |
    5. time_ago: {data.time_ago}
    6. 27 |
    7. 33 | reply 34 |
    8. 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 | loading 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 | 2 | 3 | 4 | 5 | 6 | 7 | 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 | logo 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 | --------------------------------------------------------------------------------