无关风月
2024-07-24 327be533479a547211684d326b7f6b5aec05cfa7
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
/**
 * @license Highcharts JS v2.2.5 (2012-06-08)
 * MooTools adapter
 *
 * (c) 2010-2011 Torstein Hønsi
 *
 * License: www.highcharts.com/license
 */
 
// JSLint options:
/*global Fx, $, $extend, $each, $merge, Events, Event, DOMEvent */
 
(function () {
 
var win = window,
    doc = document,
    mooVersion = win.MooTools.version.substring(0, 3), // Get the first three characters of the version number
    legacy = mooVersion === '1.2' || mooVersion === '1.1', // 1.1 && 1.2 considered legacy, 1.3 is not.
    legacyEvent = legacy || mooVersion === '1.3', // In versions 1.1 - 1.3 the event class is named Event, in newer versions it is named DOMEvent.
    $extend = win.$extend || function () {
        return Object.append.apply(Object, arguments);
    };
 
win.HighchartsAdapter = {
    /**
     * Initialize the adapter. This is run once as Highcharts is first run.
     * @param {Object} pathAnim The helper object to do animations across adapters.
     */
    init: function (pathAnim) {
        var fxProto = Fx.prototype,
            fxStart = fxProto.start,
            morphProto = Fx.Morph.prototype,
            morphCompute = morphProto.compute;
 
        // override Fx.start to allow animation of SVG element wrappers
        /*jslint unparam: true*//* allow unused parameters in fx functions */
        fxProto.start = function (from, to) {
            var fx = this,
                elem = fx.element;
 
            // special for animating paths
            if (from.d) {
                //this.fromD = this.element.d.split(' ');
                fx.paths = pathAnim.init(
                    elem,
                    elem.d,
                    fx.toD
                );
            }
            fxStart.apply(fx, arguments);
 
            return this; // chainable
        };
 
        // override Fx.step to allow animation of SVG element wrappers
        morphProto.compute = function (from, to, delta) {
            var fx = this,
                paths = fx.paths;
 
            if (paths) {
                fx.element.attr(
                    'd',
                    pathAnim.step(paths[0], paths[1], delta, fx.toD)
                );
            } else {
                return morphCompute.apply(fx, arguments);
            }
        };
        /*jslint unparam: false*/
    },
    
    /**
     * Run a general method on the framework, following jQuery syntax
     * @param {Object} el The HTML element
     * @param {String} method Which method to run on the wrapped element
     */
    adapterRun: function (el, method) {
        
        // This currently works for getting inner width and height. If adding
        // more methods later, we need a conditional implementation for each.
        return $(el).getStyle(method).toInt();
        
    },
 
    /**
     * Downloads a script and executes a callback when done.
     * @param {String} scriptLocation
     * @param {Function} callback
     */
    getScript: function (scriptLocation, callback) {
        // We cannot assume that Assets class from mootools-more is available so instead insert a script tag to download script.
        var head = doc.getElementsByTagName('head')[0];
        var script = doc.createElement('script');
 
        script.type = 'text/javascript';
        script.src = scriptLocation;
        script.onload = callback;
 
        head.appendChild(script);
    },
 
    /**
     * Animate a HTML element or SVG element wrapper
     * @param {Object} el
     * @param {Object} params
     * @param {Object} options jQuery-like animation options: duration, easing, callback
     */
    animate: function (el, params, options) {
        var isSVGElement = el.attr,
            effect,
            complete = options && options.complete;
 
        if (isSVGElement && !el.setStyle) {
            // add setStyle and getStyle methods for internal use in Moo
            el.getStyle = el.attr;
            el.setStyle = function () { // property value is given as array in Moo - break it down
                var args = arguments;
                el.attr.call(el, args[0], args[1][0]);
            };
            // dirty hack to trick Moo into handling el as an element wrapper
            el.$family = function () { return true; };
        }
 
        // stop running animations
        win.HighchartsAdapter.stop(el);
 
        // define and run the effect
        effect = new Fx.Morph(
            isSVGElement ? el : $(el),
            $extend({
                transition: Fx.Transitions.Quad.easeInOut
            }, options)
        );
 
        // Make sure that the element reference is set when animating svg elements
        if (isSVGElement) {
            effect.element = el;
        }
 
        // special treatment for paths
        if (params.d) {
            effect.toD = params.d;
        }
 
        // jQuery-like events
        if (complete) {
            effect.addEvent('complete', complete);
        }
 
        // run
        effect.start(params);
 
        // record for use in stop method
        el.fx = effect;
    },
 
    /**
     * MooTool's each function
     *
     */
    each: function (arr, fn) {
        return legacy ?
            $each(arr, fn) :
            Array.each(arr, fn);
    },
 
    /**
     * Map an array
     * @param {Array} arr
     * @param {Function} fn
     */
    map: function (arr, fn) {
        return arr.map(fn);
    },
 
    /**
     * Grep or filter an array
     * @param {Array} arr
     * @param {Function} fn
     */
    grep: function (arr, fn) {
        return arr.filter(fn);
    },
 
    /**
     * Deep merge two objects and return a third
     */
    merge: function () {
        var args = arguments,
            args13 = [{}], // MooTools 1.3+
            i = args.length,
            ret;
 
        if (legacy) {
            ret = $merge.apply(null, args);
        } else {
            while (i--) {
                // Boolean argumens should not be merged.
                // JQuery explicitly skips this, so we do it here as well.
                if (typeof args[i] !== 'boolean') {
                    args13[i + 1] = args[i];
                }
            }
            ret = Object.merge.apply(Object, args13);
        }
 
        return ret;
    },
 
    /**
     * Get the offset of an element relative to the top left corner of the web page
     */
    offset: function (el) {
        var offsets = $(el).getOffsets();
        return {
            left: offsets.x,
            top: offsets.y
        };
    },
 
    /**
     * Extends an object with Events, if its not done
     */
    extendWithEvents: function (el) {
        // if the addEvent method is not defined, el is a custom Highcharts object
        // like series or point
        if (!el.addEvent) {
            if (el.nodeName) {
                el = $(el); // a dynamically generated node
            } else {
                $extend(el, new Events()); // a custom object
            }
        }
    },
 
    /**
     * Add an event listener
     * @param {Object} el HTML element or custom object
     * @param {String} type Event type
     * @param {Function} fn Event handler
     */
    addEvent: function (el, type, fn) {
        if (typeof type === 'string') { // chart broke due to el being string, type function
 
            if (type === 'unload') { // Moo self destructs before custom unload events
                type = 'beforeunload';
            }
 
            win.HighchartsAdapter.extendWithEvents(el);
 
            el.addEvent(type, fn);
        }
    },
 
    removeEvent: function (el, type, fn) {
        if (typeof el === 'string') {
            // el.removeEvents below apperantly calls this method again. Do not quite understand why, so for now just bail out.
            return;
        }
        
        win.HighchartsAdapter.extendWithEvents(el);
        if (type) {
            if (type === 'unload') { // Moo self destructs before custom unload events
                type = 'beforeunload';
            }
 
            if (fn) {
                el.removeEvent(type, fn);
            } else if (el.removeEvents) { // #958
                el.removeEvents(type);
            }
        } else {
            el.removeEvents();
        }
    },
 
    fireEvent: function (el, event, eventArguments, defaultFunction) {
        var eventArgs = {
            type: event,
            target: el
        };
        // create an event object that keeps all functions
        event = legacyEvent ? new Event(eventArgs) : new DOMEvent(eventArgs);
        event = $extend(event, eventArguments);
        // override the preventDefault function to be able to use
        // this for custom events
        event.preventDefault = function () {
            defaultFunction = null;
        };
        // if fireEvent is not available on the object, there hasn't been added
        // any events to it above
        if (el.fireEvent) {
            el.fireEvent(event.type, event);
        }
 
        // fire the default if it is passed and it is not prevented above
        if (defaultFunction) {
            defaultFunction(event);
        }
    },
    
    /**
     * Set back e.pageX and e.pageY that MooTools has abstracted away
     */
    washMouseEvent: function (e) {
        e.pageX = e.page.x;
        e.pageY = e.page.y;
        return e;
    },
 
    /**
     * Stop running animations on the object
     */
    stop: function (el) {
        if (el.fx) {
            el.fx.cancel();
        }
    }
};
 
}());