UNPKG

16.8 kBJavaScriptView Raw
1/**
2 * Analog clock.
3 * @constructor ApAnalogClock
4 */
5
6"use strict";
7
8const React = require('react'),
9 ReactDOM = require('react-dom'),
10 classnames = require('classnames'),
11 ApClock = require('./ap_clock'),
12 PureRenderMixin = require('react-addons-pure-render-mixin'),
13 chopcal = require('chopcal'),
14 numcal = require('numcal'),
15 types = React.PropTypes,
16 ApAnalogClockHand = require('./ap_analog_clock_hand'),
17 ApAnalogClockLetter = require('./ap_analog_clock_letter');
18
19/** @lends ApAnalogClock */
20let ApAnalogClock = React.createClass({
21 displayName: 'ApAnalogClock',
22
23 //--------------------
24 // Specs
25 //--------------------
26
27 propTypes: {
28 boardLetters: types.array
29 },
30
31 mixins: [PureRenderMixin],
32
33 statics: {
34 _angleForValue: function (value, max) {
35 let rate = value % max / max;
36 return chopcal.round(rate * 360, 0.1);
37 },
38 hourHandAngle: function (date) {
39 let hours = date.getHours();
40 return ApAnalogClock._angleForValue(hours, 12);
41 },
42 minuteHandAngle: function (date) {
43 let minutes = date.getMinutes();
44 return ApAnalogClock._angleForValue(minutes, 60);
45 },
46 secondHandAngle: function (date) {
47 let seconds = date.getSeconds();
48 return ApAnalogClock._angleForValue(seconds, 60);
49 },
50 letterAngle: function (i, count) {
51 return ApAnalogClock._angleForValue(i, count);
52 }
53 },
54
55 getInitialState: function () {
56 return {
57 looping: false,
58 hour: 0,
59 minute: 0,
60 second: 0,
61 size: 256
62 };
63 },
64
65 getDefaultProps: function () {
66 return {
67 boardLetters: '12,1,2,3,4,5,6,7,8,9,10,11'.split(',')
68 };
69 },
70
71 render: function () {
72 let s = this,
73 state = s.state,
74 props = s.props;
75
76 let letters = props.boardLetters.map((letter, i, letters) => {
77 let angle = ApAnalogClock.letterAngle(i, letters.length);
78 return React.createElement(ApAnalogClockLetter, { key: "ap-analog-letter-" + i,
79 letter: letter,
80 angle: angle });
81 });
82
83 return React.createElement(
84 ApClock,
85 { className: classnames("ap-analog-clock", props.className) },
86 React.createElement(
87 'div',
88 { className: 'ap-analog-clock-board', style: { width: state.size, height: state.size } },
89 React.createElement(
90 'div',
91 { className: 'ap-analog-clock-board-inner' },
92 React.createElement(ApAnalogClockHand, { width: 4, heightRate: 0.8, angle: state.hour }),
93 React.createElement(ApAnalogClockHand, { width: 4, heightRate: 1, angle: state.minute }),
94 React.createElement(ApAnalogClockHand, { width: 2, heightRate: 1, angle: state.second })
95 ),
96 React.createElement(
97 'div',
98 null,
99 letters
100 )
101 )
102 );
103 },
104
105 //--------------------
106 // Lifecycle
107 //--------------------
108
109 componentWillMount: function () {
110 let s = this;
111 s._looping = true;
112 },
113
114 componentDidMount: function () {
115 let s = this,
116 props = s.props;
117
118 function _loop() {
119 if (!s._looping) {
120 return;
121 }
122 let now = new Date();
123 s.setState({
124 hour: ApAnalogClock.hourHandAngle(now),
125 minute: ApAnalogClock.minuteHandAngle(now),
126 second: ApAnalogClock.secondHandAngle(now)
127 });
128 window.requestAnimationFrame(_loop);
129 }
130
131 window.addEventListener('resize', s.resizeClock);
132 _loop();
133 s.resizeClock();
134 },
135
136 componentWillReceiveProps: function (nextProps) {
137 let s = this;
138 },
139
140 componentWillUpdate: function (nextProps, nextState) {
141 let s = this;
142 },
143
144 componentDidUpdate: function (prevProps, prevState) {
145 let s = this;
146 },
147
148 componentWillUnmount: function () {
149 let s = this;
150 window.removeEventListener('resize', s.resizeClock);
151 s._looping = false;
152 },
153
154 //------------------
155 // Helper
156 //------------------
157
158 resizeClock: function () {
159 let s = this,
160 elm = ReactDOM.findDOMNode(s);
161 let size = numcal.min(elm.offsetWidth, elm.offsetHeight);
162 s.setState({
163 size: size
164 });
165 }
166 //------------------
167 // Private
168 //------------------
169});
170
171module.exports = ApAnalogClock;
172//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["jsx/ap_analog_clock.jsx"],"names":[],"mappings":"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","file":"ap_analog_clock.js","sourceRoot":"/Users/okunishitaka/projects/apeman-react-labo/apeman-react-clock/lib/jsx","sourcesContent":["/**\n * Analog clock.\n * @constructor ApAnalogClock\n */\n\n\"use strict\";\n\nconst React = require('react'),\n    ReactDOM = require('react-dom'),\n    classnames = require('classnames'),\n    ApClock = require('./ap_clock'),\n    PureRenderMixin = require('react-addons-pure-render-mixin'),\n    chopcal = require('chopcal'),\n    numcal = require('numcal'),\n    types = React.PropTypes,\n    ApAnalogClockHand = require('./ap_analog_clock_hand'),\n    ApAnalogClockLetter = require('./ap_analog_clock_letter');\n\n/** @lends ApAnalogClock */\nlet ApAnalogClock = React.createClass({\n\n    //--------------------\n    // Specs\n    //--------------------\n\n    propTypes: {\n        boardLetters: types.array\n    },\n\n    mixins: [\n        PureRenderMixin\n    ],\n\n    statics: {\n        _angleForValue: function (value, max) {\n            let rate = (value % max) / max;\n            return chopcal.round(rate * 360, 0.1);\n        },\n        hourHandAngle: function (date) {\n            let hours = date.getHours();\n            return ApAnalogClock._angleForValue(hours, 12);\n        },\n        minuteHandAngle: function (date) {\n            let minutes = date.getMinutes();\n            return ApAnalogClock._angleForValue(minutes, 60);\n        },\n        secondHandAngle: function (date) {\n            let seconds = date.getSeconds();\n            return ApAnalogClock._angleForValue(seconds, 60);\n        },\n        letterAngle: function (i, count) {\n            return ApAnalogClock._angleForValue(i, count);\n        }\n    },\n\n    getInitialState: function () {\n        return {\n            looping: false,\n            hour: 0,\n            minute: 0,\n            second: 0,\n            size: 256\n        };\n    },\n\n    getDefaultProps: function () {\n        return {\n            boardLetters: '12,1,2,3,4,5,6,7,8,9,10,11'.split(',')\n        };\n    },\n\n    render: function () {\n        let s = this,\n            state = s.state,\n            props = s.props;\n\n        let letters = props.boardLetters.map((letter, i, letters)=> {\n            let angle = ApAnalogClock.letterAngle(i, letters.length);\n            return (\n                <ApAnalogClockLetter key={\"ap-analog-letter-\" + i}\n                                     letter={letter}\n                                     angle={angle}></ApAnalogClockLetter>\n            );\n        });\n\n        return (\n            <ApClock className={classnames(\"ap-analog-clock\", props.className)}>\n                <div className=\"ap-analog-clock-board\" style={{width:state.size, height:state.size}}>\n                    <div className=\"ap-analog-clock-board-inner\">\n                        <ApAnalogClockHand width={4} heightRate={0.8} angle={state.hour}></ApAnalogClockHand>\n                        <ApAnalogClockHand width={4} heightRate={1} angle={state.minute}></ApAnalogClockHand>\n                        <ApAnalogClockHand width={2} heightRate={1} angle={state.second}></ApAnalogClockHand>\n                    </div>\n                    <div>\n                        {letters}\n                    </div>\n                </div>\n            </ApClock>\n        );\n    },\n\n\n    //--------------------\n    // Lifecycle\n    //--------------------\n\n    componentWillMount: function () {\n        let s = this;\n        s._looping = true;\n    },\n\n    componentDidMount: function () {\n        let s = this,\n            props = s.props;\n\n        function _loop() {\n            if (!s._looping) {\n                return;\n            }\n            let now = new Date();\n            s.setState({\n                hour: ApAnalogClock.hourHandAngle(now),\n                minute: ApAnalogClock.minuteHandAngle(now),\n                second: ApAnalogClock.secondHandAngle(now)\n            });\n            window.requestAnimationFrame(_loop);\n        }\n\n        window.addEventListener('resize', s.resizeClock);\n        _loop();\n        s.resizeClock();\n\n\n    },\n\n    componentWillReceiveProps: function (nextProps) {\n        let s = this;\n    },\n\n    componentWillUpdate: function (nextProps, nextState) {\n        let s = this;\n    },\n\n    componentDidUpdate: function (prevProps, prevState) {\n        let s = this;\n    },\n\n    componentWillUnmount: function () {\n        let s = this;\n        window.removeEventListener('resize', s.resizeClock);\n        s._looping = false;\n    },\n\n    //------------------\n    // Helper\n    //------------------\n\n    resizeClock: function () {\n        let s = this,\n            elm = ReactDOM.findDOMNode(s);\n        let size = numcal.min(elm.offsetWidth, elm.offsetHeight);\n        s.setState({\n            size: size\n        });\n    }\n    //------------------\n    // Private\n    //------------------\n});\n\nmodule.exports = ApAnalogClock;\n"]}
\No newline at end of file