1 | /**
|
2 | * Analog clock.
|
3 | * @constructor ApAnalogClock
|
4 | */
|
5 |
|
6 | ;
|
7 |
|
8 | const 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 */
|
20 | let 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 |
|
171 | module.exports = ApAnalogClock;
|
172 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["jsx/ap_analog_clock.jsx"],"names":[],"mappings":";;;;;AAKA,YAAY,CAAC;;AAEb,MAAM,KAAK,GAAG,OAAO,CAAC,OAAO,CAAC;MAC1B,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC;MAC/B,UAAU,GAAG,OAAO,CAAC,YAAY,CAAC;MAClC,OAAO,GAAG,OAAO,CAAC,YAAY,CAAC;MAC/B,eAAe,GAAG,OAAO,CAAC,gCAAgC,CAAC;MAC3D,OAAO,GAAG,OAAO,CAAC,SAAS,CAAC;MAC5B,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;MAC1B,KAAK,GAAG,KAAK,CAAC,SAAS;MACvB,iBAAiB,GAAG,OAAO,CAAC,wBAAwB,CAAC;MACrD,mBAAmB,GAAG,OAAO,CAAC,0BAA0B,CAAC;;;AAAC,AAG9D,IAAI,aAAa,GAAG,KAAK,CAAC,WAAW,CAAC;;;;;;;AAMlC,aAAS,EAAE;AACP,oBAAY,EAAE,KAAK,CAAC,KAAK;KAC5B;;AAED,UAAM,EAAE,CACJ,eAAe,CAClB;;AAED,WAAO,EAAE;AACL,sBAAc,EAAE,UAAU,KAAK,EAAE,GAAG,EAAE;AAClC,gBAAI,IAAI,GAAG,AAAC,KAAK,GAAG,GAAG,GAAI,GAAG,CAAC;AAC/B,mBAAO,OAAO,CAAC,KAAK,CAAC,IAAI,GAAG,GAAG,EAAE,GAAG,CAAC,CAAC;SACzC;AACD,qBAAa,EAAE,UAAU,IAAI,EAAE;AAC3B,gBAAI,KAAK,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;AAC5B,mBAAO,aAAa,CAAC,cAAc,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;SAClD;AACD,uBAAe,EAAE,UAAU,IAAI,EAAE;AAC7B,gBAAI,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;AAChC,mBAAO,aAAa,CAAC,cAAc,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;SACpD;AACD,uBAAe,EAAE,UAAU,IAAI,EAAE;AAC7B,gBAAI,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;AAChC,mBAAO,aAAa,CAAC,cAAc,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;SACpD;AACD,mBAAW,EAAE,UAAU,CAAC,EAAE,KAAK,EAAE;AAC7B,mBAAO,aAAa,CAAC,cAAc,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;SACjD;KACJ;;AAED,mBAAe,EAAE,YAAY;AACzB,eAAO;AACH,mBAAO,EAAE,KAAK;AACd,gBAAI,EAAE,CAAC;AACP,kBAAM,EAAE,CAAC;AACT,kBAAM,EAAE,CAAC;AACT,gBAAI,EAAE,GAAG;SACZ,CAAC;KACL;;AAED,mBAAe,EAAE,YAAY;AACzB,eAAO;AACH,wBAAY,EAAE,4BAA4B,CAAC,KAAK,CAAC,GAAG,CAAC;SACxD,CAAC;KACL;;AAED,UAAM,EAAE,YAAY;AAChB,YAAI,CAAC,GAAG,IAAI;YACR,KAAK,GAAG,CAAC,CAAC,KAAK;YACf,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC;;AAEpB,YAAI,OAAO,GAAG,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,CAAC,EAAE,OAAO,KAAI;AACxD,gBAAI,KAAK,GAAG,aAAa,CAAC,WAAW,CAAC,CAAC,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;AACzD,mBACI,oBAAC,mBAAmB,IAAC,GAAG,EAAE,mBAAmB,GAAG,CAAC,AAAC;AAC7B,sBAAM,EAAE,MAAM,AAAC;AACf,qBAAK,EAAE,KAAK,AAAC,GAAuB,CAC3D;SACL,CAAC,CAAC;;AAEH,eACI;AAAC,mBAAO;cAAC,SAAS,EAAE,UAAU,CAAC,iBAAiB,EAAE,KAAK,CAAC,SAAS,CAAC,AAAC;YAC/D;;kBAAK,SAAS,EAAC,uBAAuB,EAAC,KAAK,EAAE,EAAC,KAAK,EAAC,KAAK,CAAC,IAAI,EAAE,MAAM,EAAC,KAAK,CAAC,IAAI,EAAC,AAAC;gBAChF;;sBAAK,SAAS,EAAC,6BAA6B;oBACxC,oBAAC,iBAAiB,IAAC,KAAK,EAAE,CAAC,AAAC,EAAC,UAAU,EAAE,GAAG,AAAC,EAAC,KAAK,EAAE,KAAK,CAAC,IAAI,AAAC,GAAqB;oBACrF,oBAAC,iBAAiB,IAAC,KAAK,EAAE,CAAC,AAAC,EAAC,UAAU,EAAE,CAAC,AAAC,EAAC,KAAK,EAAE,KAAK,CAAC,MAAM,AAAC,GAAqB;oBACrF,oBAAC,iBAAiB,IAAC,KAAK,EAAE,CAAC,AAAC,EAAC,UAAU,EAAE,CAAC,AAAC,EAAC,KAAK,EAAE,KAAK,CAAC,MAAM,AAAC,GAAqB;iBACnF;gBACN;;;oBACK,OAAO;iBACN;aACJ;SACA,CACZ;KACL;;;;;;AAOD,sBAAkB,EAAE,YAAY;AAC5B,YAAI,CAAC,GAAG,IAAI,CAAC;AACb,SAAC,CAAC,QAAQ,GAAG,IAAI,CAAC;KACrB;;AAED,qBAAiB,EAAE,YAAY;AAC3B,YAAI,CAAC,GAAG,IAAI;YACR,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC;;AAEpB,iBAAS,KAAK,GAAG;AACb,gBAAI,CAAC,CAAC,CAAC,QAAQ,EAAE;AACb,uBAAO;aACV;AACD,gBAAI,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;AACrB,aAAC,CAAC,QAAQ,CAAC;AACP,oBAAI,EAAE,aAAa,CAAC,aAAa,CAAC,GAAG,CAAC;AACtC,sBAAM,EAAE,aAAa,CAAC,eAAe,CAAC,GAAG,CAAC;AAC1C,sBAAM,EAAE,aAAa,CAAC,eAAe,CAAC,GAAG,CAAC;aAC7C,CAAC,CAAC;AACH,kBAAM,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;SACvC;;AAED,cAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,CAAC,CAAC,WAAW,CAAC,CAAC;AACjD,aAAK,EAAE,CAAC;AACR,SAAC,CAAC,WAAW,EAAE,CAAC;KAGnB;;AAED,6BAAyB,EAAE,UAAU,SAAS,EAAE;AAC5C,YAAI,CAAC,GAAG,IAAI,CAAC;KAChB;;AAED,uBAAmB,EAAE,UAAU,SAAS,EAAE,SAAS,EAAE;AACjD,YAAI,CAAC,GAAG,IAAI,CAAC;KAChB;;AAED,sBAAkB,EAAE,UAAU,SAAS,EAAE,SAAS,EAAE;AAChD,YAAI,CAAC,GAAG,IAAI,CAAC;KAChB;;AAED,wBAAoB,EAAE,YAAY;AAC9B,YAAI,CAAC,GAAG,IAAI,CAAC;AACb,cAAM,CAAC,mBAAmB,CAAC,QAAQ,EAAE,CAAC,CAAC,WAAW,CAAC,CAAC;AACpD,SAAC,CAAC,QAAQ,GAAG,KAAK,CAAC;KACtB;;;;;;AAMD,eAAW,EAAE,YAAY;AACrB,YAAI,CAAC,GAAG,IAAI;YACR,GAAG,GAAG,QAAQ,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC;AAClC,YAAI,IAAI,GAAG,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,WAAW,EAAE,GAAG,CAAC,YAAY,CAAC,CAAC;AACzD,SAAC,CAAC,QAAQ,CAAC;AACP,gBAAI,EAAE,IAAI;SACb,CAAC,CAAC;KACN;;;;AAAA,CAIJ,CAAC,CAAC;;AAEH,MAAM,CAAC,OAAO,GAAG,aAAa,CAAC","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 |