選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。
 
 
 
 
 
 

1571 行
43 KiB

  1. // ==ClosureCompiler==
  2. // @compilation_level SIMPLE_OPTIMIZATIONS
  3. /**
  4. * @license Highcharts JS v4.1.9 (2015-10-07)
  5. *
  6. * (c) 2009-2013 Torstein Hønsi
  7. *
  8. * License: www.highcharts.com/license
  9. */
  10. // JSLint options:
  11. /*global Highcharts, HighchartsAdapter, document, window, navigator, setInterval, clearInterval, clearTimeout, setTimeout, location, jQuery, $, console */
  12. (function (Highcharts) {
  13. /**
  14. Shorthands for often used function
  15. */
  16. /**
  17. * Mathematical Functionility
  18. */
  19. var PI = Math.PI,
  20. deg2rad = (PI / 180), // degrees to radians
  21. sin = Math.sin,
  22. cos = Math.cos,
  23. pick = Highcharts.pick,
  24. round = Math.round;
  25. /**
  26. * Transforms a given array of points according to the angles in chart.options.
  27. * Parameters:
  28. * - points: the array of points
  29. * - chart: the chart
  30. * - insidePlotArea: wether to verifiy the points are inside the plotArea
  31. * Returns:
  32. * - an array of transformed points
  33. */
  34. function perspective(points, chart, insidePlotArea) {
  35. var options3d = chart.options.chart.options3d,
  36. inverted = false,
  37. origin;
  38. if (insidePlotArea) {
  39. inverted = chart.inverted;
  40. origin = {
  41. x: chart.plotWidth / 2,
  42. y: chart.plotHeight / 2,
  43. z: options3d.depth / 2,
  44. vd: pick(options3d.depth, 1) * pick(options3d.viewDistance, 0)
  45. };
  46. } else {
  47. origin = {
  48. x: chart.plotLeft + (chart.plotWidth / 2),
  49. y: chart.plotTop + (chart.plotHeight / 2),
  50. z: options3d.depth / 2,
  51. vd: pick(options3d.depth, 1) * pick(options3d.viewDistance, 0)
  52. };
  53. }
  54. var result = [],
  55. xe = origin.x,
  56. ye = origin.y,
  57. ze = origin.z,
  58. vd = origin.vd,
  59. angle1 = deg2rad * (inverted ? options3d.beta : -options3d.beta),
  60. angle2 = deg2rad * (inverted ? -options3d.alpha : options3d.alpha),
  61. s1 = sin(angle1),
  62. c1 = cos(angle1),
  63. s2 = sin(angle2),
  64. c2 = cos(angle2);
  65. var x, y, z, px, py, pz;
  66. // Transform each point
  67. Highcharts.each(points, function (point) {
  68. x = (inverted ? point.y : point.x) - xe;
  69. y = (inverted ? point.x : point.y) - ye;
  70. z = (point.z || 0) - ze;
  71. //Apply 3-D rotation
  72. px = c1 * x - s1 * z;
  73. py = -s1 * s2 * x - c1 * s2 * z + c2 * y;
  74. pz = s1 * c2 * x + c1 * c2 * z + s2 * y;
  75. //Apply perspective
  76. if ((vd > 0) && (vd < Number.POSITIVE_INFINITY)) {
  77. px = px * (vd / (pz + ze + vd));
  78. py = py * (vd / (pz + ze + vd));
  79. }
  80. //Apply translation
  81. px = px + xe;
  82. py = py + ye;
  83. pz = pz + ze;
  84. result.push({
  85. x: (inverted ? py : px),
  86. y: (inverted ? px : py),
  87. z: pz
  88. });
  89. });
  90. return result;
  91. }
  92. // Make function acessible to plugins
  93. Highcharts.perspective = perspective;
  94. /***
  95. EXTENSION TO THE SVG-RENDERER TO ENABLE 3D SHAPES
  96. ***/
  97. ////// HELPER METHODS //////
  98. var dFactor = (4 * (Math.sqrt(2) - 1) / 3) / (PI / 2);
  99. function defined(obj) {
  100. return obj !== undefined && obj !== null;
  101. }
  102. //Shoelace algorithm -- http://en.wikipedia.org/wiki/Shoelace_formula
  103. function shapeArea(vertexes) {
  104. var area = 0,
  105. i,
  106. j;
  107. for (i = 0; i < vertexes.length; i++) {
  108. j = (i + 1) % vertexes.length;
  109. area += vertexes[i].x * vertexes[j].y - vertexes[j].x * vertexes[i].y;
  110. }
  111. return area / 2;
  112. }
  113. function averageZ(vertexes) {
  114. var z = 0,
  115. i;
  116. for (i = 0; i < vertexes.length; i++) {
  117. z += vertexes[i].z;
  118. }
  119. return vertexes.length ? z / vertexes.length : 0;
  120. }
  121. /** Method to construct a curved path
  122. * Can 'wrap' around more then 180 degrees
  123. */
  124. function curveTo(cx, cy, rx, ry, start, end, dx, dy) {
  125. var result = [];
  126. if ((end > start) && (end - start > PI / 2 + 0.0001)) {
  127. result = result.concat(curveTo(cx, cy, rx, ry, start, start + (PI / 2), dx, dy));
  128. result = result.concat(curveTo(cx, cy, rx, ry, start + (PI / 2), end, dx, dy));
  129. return result;
  130. } else if ((end < start) && (start - end > PI / 2 + 0.0001)) {
  131. result = result.concat(curveTo(cx, cy, rx, ry, start, start - (PI / 2), dx, dy));
  132. result = result.concat(curveTo(cx, cy, rx, ry, start - (PI / 2), end, dx, dy));
  133. return result;
  134. } else {
  135. var arcAngle = end - start;
  136. return [
  137. 'C',
  138. cx + (rx * cos(start)) - ((rx * dFactor * arcAngle) * sin(start)) + dx,
  139. cy + (ry * sin(start)) + ((ry * dFactor * arcAngle) * cos(start)) + dy,
  140. cx + (rx * cos(end)) + ((rx * dFactor * arcAngle) * sin(end)) + dx,
  141. cy + (ry * sin(end)) - ((ry * dFactor * arcAngle) * cos(end)) + dy,
  142. cx + (rx * cos(end)) + dx,
  143. cy + (ry * sin(end)) + dy
  144. ];
  145. }
  146. }
  147. Highcharts.SVGRenderer.prototype.toLinePath = function (points, closed) {
  148. var result = [];
  149. // Put "L x y" for each point
  150. Highcharts.each(points, function (point) {
  151. result.push('L', point.x, point.y);
  152. });
  153. if (points.length) {
  154. // Set the first element to M
  155. result[0] = 'M';
  156. // If it is a closed line, add Z
  157. if (closed) {
  158. result.push('Z');
  159. }
  160. }
  161. return result;
  162. };
  163. ////// CUBOIDS //////
  164. Highcharts.SVGRenderer.prototype.cuboid = function (shapeArgs) {
  165. var result = this.g(),
  166. paths = this.cuboidPath(shapeArgs);
  167. // create the 3 sides
  168. result.front = this.path(paths[0]).attr({zIndex: paths[3], 'stroke-linejoin': 'round'}).add(result);
  169. result.top = this.path(paths[1]).attr({zIndex: paths[4], 'stroke-linejoin': 'round'}).add(result);
  170. result.side = this.path(paths[2]).attr({zIndex: paths[5], 'stroke-linejoin': 'round'}).add(result);
  171. // apply the fill everywhere, the top a bit brighter, the side a bit darker
  172. result.fillSetter = function (color) {
  173. var c0 = color,
  174. c1 = Highcharts.Color(color).brighten(0.1).get(),
  175. c2 = Highcharts.Color(color).brighten(-0.1).get();
  176. this.front.attr({fill: c0});
  177. this.top.attr({fill: c1});
  178. this.side.attr({fill: c2});
  179. this.color = color;
  180. return this;
  181. };
  182. // apply opacaity everywhere
  183. result.opacitySetter = function (opacity) {
  184. this.front.attr({opacity: opacity});
  185. this.top.attr({opacity: opacity});
  186. this.side.attr({opacity: opacity});
  187. return this;
  188. };
  189. result.attr = function (args) {
  190. if (args.shapeArgs || defined(args.x)) {
  191. var shapeArgs = args.shapeArgs || args;
  192. var paths = this.renderer.cuboidPath(shapeArgs);
  193. this.front.attr({d: paths[0], zIndex: paths[3]});
  194. this.top.attr({d: paths[1], zIndex: paths[4]});
  195. this.side.attr({d: paths[2], zIndex: paths[5]});
  196. } else {
  197. Highcharts.SVGElement.prototype.attr.call(this, args);
  198. }
  199. return this;
  200. };
  201. result.animate = function (args, duration, complete) {
  202. if (defined(args.x) && defined(args.y)) {
  203. var paths = this.renderer.cuboidPath(args);
  204. this.front.attr({zIndex: paths[3]}).animate({d: paths[0]}, duration, complete);
  205. this.top.attr({zIndex: paths[4]}).animate({d: paths[1]}, duration, complete);
  206. this.side.attr({zIndex: paths[5]}).animate({d: paths[2]}, duration, complete);
  207. } else if (args.opacity) {
  208. this.front.animate(args, duration, complete);
  209. this.top.animate(args, duration, complete);
  210. this.side.animate(args, duration, complete);
  211. } else {
  212. Highcharts.SVGElement.prototype.animate.call(this, args, duration, complete);
  213. }
  214. return this;
  215. };
  216. // destroy all children
  217. result.destroy = function () {
  218. this.front.destroy();
  219. this.top.destroy();
  220. this.side.destroy();
  221. return null;
  222. };
  223. // Apply the Z index to the cuboid group
  224. result.attr({ zIndex: -paths[3] });
  225. return result;
  226. };
  227. /**
  228. * Generates a cuboid
  229. */
  230. Highcharts.SVGRenderer.prototype.cuboidPath = function (shapeArgs) {
  231. var x = shapeArgs.x,
  232. y = shapeArgs.y,
  233. z = shapeArgs.z,
  234. h = shapeArgs.height,
  235. w = shapeArgs.width,
  236. d = shapeArgs.depth,
  237. chart = Highcharts.charts[this.chartIndex],
  238. map = Highcharts.map;
  239. // The 8 corners of the cube
  240. var pArr = [
  241. {x: x, y: y, z: z},
  242. {x: x + w, y: y, z: z},
  243. {x: x + w, y: y + h, z: z},
  244. {x: x, y: y + h, z: z},
  245. {x: x, y: y + h, z: z + d},
  246. {x: x + w, y: y + h, z: z + d},
  247. {x: x + w, y: y, z: z + d},
  248. {x: x, y: y, z: z + d}
  249. ];
  250. // apply perspective
  251. pArr = perspective(pArr, chart, shapeArgs.insidePlotArea);
  252. // helper method to decide which side is visible
  253. var pickShape = function (path1, path2) {
  254. path1 = map(path1, function (i) { return pArr[i]; });
  255. path2 = map(path2, function (i) { return pArr[i]; });
  256. if (shapeArea(path1) < 0) {
  257. return path1;
  258. } else if (shapeArea(path2) < 0) {
  259. return path2;
  260. } else {
  261. return [];
  262. }
  263. };
  264. // front or back
  265. var front = [3, 2, 1, 0];
  266. var back = [7, 6, 5, 4];
  267. var path1 = pickShape(front, back);
  268. // top or bottom
  269. var top = [1, 6, 7, 0];
  270. var bottom = [4, 5, 2, 3];
  271. var path2 = pickShape(top, bottom);
  272. // side
  273. var right = [1, 2, 5, 6];
  274. var left = [0, 7, 4, 3];
  275. var path3 = pickShape(right, left);
  276. return [this.toLinePath(path1, true), this.toLinePath(path2, true), this.toLinePath(path3, true), averageZ(path1), averageZ(path2), averageZ(path3)];
  277. };
  278. ////// SECTORS //////
  279. Highcharts.SVGRenderer.prototype.arc3d = function (shapeArgs) {
  280. shapeArgs.alpha *= deg2rad;
  281. shapeArgs.beta *= deg2rad;
  282. var result = this.g(),
  283. paths = this.arc3dPath(shapeArgs),
  284. renderer = result.renderer;
  285. var zIndex = paths.zTop * 100;
  286. result.shapeArgs = shapeArgs; // Store for later use
  287. // create the different sub sections of the shape
  288. result.top = renderer.path(paths.top).setRadialReference(shapeArgs.center).attr({zIndex: paths.zTop}).add(result);
  289. result.side1 = renderer.path(paths.side2).attr({zIndex: paths.zSide1});
  290. result.side2 = renderer.path(paths.side1).attr({zIndex: paths.zSide2});
  291. result.inn = renderer.path(paths.inn).attr({zIndex: paths.zInn});
  292. result.out = renderer.path(paths.out).attr({zIndex: paths.zOut});
  293. // apply the fill to the top and a darker shade to the sides
  294. result.fillSetter = function (color) {
  295. this.color = color;
  296. var c0 = color,
  297. c2 = Highcharts.Color(color).brighten(-0.1).get();
  298. this.side1.attr({fill: c2});
  299. this.side2.attr({fill: c2});
  300. this.inn.attr({fill: c2});
  301. this.out.attr({fill: c2});
  302. this.top.attr({fill: c0});
  303. return this;
  304. };
  305. // apply the translation to all
  306. result.translateXSetter = function (value) {
  307. this.out.attr({translateX: value});
  308. this.inn.attr({translateX: value});
  309. this.side1.attr({translateX: value});
  310. this.side2.attr({translateX: value});
  311. this.top.attr({translateX: value});
  312. };
  313. result.translateYSetter = function (value) {
  314. this.out.attr({translateY: value});
  315. this.inn.attr({translateY: value});
  316. this.side1.attr({translateY: value});
  317. this.side2.attr({translateY: value});
  318. this.top.attr({translateY: value});
  319. };
  320. result.animate = function (args, duration, complete) {
  321. if (defined(args.end) || defined(args.start)) {
  322. this._shapeArgs = this.shapeArgs;
  323. Highcharts.SVGElement.prototype.animate.call(this, {
  324. _args: args
  325. }, {
  326. duration: duration,
  327. start: function () {
  328. var args = arguments,
  329. fx = args[0],
  330. elem = fx.elem,
  331. end = elem._shapeArgs;
  332. if (end.fill !== elem.color) {
  333. elem.attr({
  334. fill: end.fill
  335. });
  336. }
  337. },
  338. step: function () {
  339. var args = arguments,
  340. fx = args[1],
  341. result = fx.elem,
  342. start = result._shapeArgs,
  343. end = fx.end,
  344. pos = fx.pos,
  345. sA = Highcharts.merge(start, {
  346. x: start.x + ((end.x - start.x) * pos),
  347. y: start.y + ((end.y - start.y) * pos),
  348. r: start.r + ((end.r - start.r) * pos),
  349. innerR: start.innerR + ((end.innerR - start.innerR) * pos),
  350. start: start.start + ((end.start - start.start) * pos),
  351. end: start.end + ((end.end - start.end) * pos)
  352. });
  353. var paths = result.renderer.arc3dPath(sA);
  354. result.shapeArgs = sA;
  355. result.top.attr({d: paths.top, zIndex: paths.zTop});
  356. result.inn.attr({d: paths.inn, zIndex: paths.zInn});
  357. result.out.attr({d: paths.out, zIndex: paths.zOut});
  358. result.side1.attr({d: paths.side1, zIndex: paths.zSide1});
  359. result.side2.attr({d: paths.side2, zIndex: paths.zSide2});
  360. }
  361. }, complete);
  362. } else {
  363. Highcharts.SVGElement.prototype.animate.call(this, args, duration, complete);
  364. }
  365. return this;
  366. };
  367. // destroy all children
  368. result.destroy = function () {
  369. this.top.destroy();
  370. this.out.destroy();
  371. this.inn.destroy();
  372. this.side1.destroy();
  373. this.side2.destroy();
  374. Highcharts.SVGElement.prototype.destroy.call(this);
  375. };
  376. // hide all children
  377. result.hide = function () {
  378. this.top.hide();
  379. this.out.hide();
  380. this.inn.hide();
  381. this.side1.hide();
  382. this.side2.hide();
  383. };
  384. result.show = function () {
  385. this.top.show();
  386. this.out.show();
  387. this.inn.show();
  388. this.side1.show();
  389. this.side2.show();
  390. };
  391. // show all children
  392. result.zIndex = zIndex;
  393. result.attr({zIndex: zIndex});
  394. return result;
  395. };
  396. /**
  397. * Generate the paths required to draw a 3D arc
  398. */
  399. Highcharts.SVGRenderer.prototype.arc3dPath = function (shapeArgs) {
  400. var cx = shapeArgs.x, // x coordinate of the center
  401. cy = shapeArgs.y, // y coordinate of the center
  402. start = shapeArgs.start, // start angle
  403. end = shapeArgs.end - 0.00001, // end angle
  404. r = shapeArgs.r, // radius
  405. ir = shapeArgs.innerR, // inner radius
  406. d = shapeArgs.depth, // depth
  407. alpha = shapeArgs.alpha, // alpha rotation of the chart
  408. beta = shapeArgs.beta; // beta rotation of the chart
  409. // Derived Variables
  410. var cs = cos(start), // cosinus of the start angle
  411. ss = sin(start), // sinus of the start angle
  412. ce = cos(end), // cosinus of the end angle
  413. se = sin(end), // sinus of the end angle
  414. rx = r * cos(beta), // x-radius
  415. ry = r * cos(alpha), // y-radius
  416. irx = ir * cos(beta), // x-radius (inner)
  417. iry = ir * cos(alpha), // y-radius (inner)
  418. dx = d * sin(beta), // distance between top and bottom in x
  419. dy = d * sin(alpha); // distance between top and bottom in y
  420. // TOP
  421. var top = ['M', cx + (rx * cs), cy + (ry * ss)];
  422. top = top.concat(curveTo(cx, cy, rx, ry, start, end, 0, 0));
  423. top = top.concat([
  424. 'L', cx + (irx * ce), cy + (iry * se)
  425. ]);
  426. top = top.concat(curveTo(cx, cy, irx, iry, end, start, 0, 0));
  427. top = top.concat(['Z']);
  428. // OUTSIDE
  429. var b = (beta > 0 ? PI / 2 : 0),
  430. a = (alpha > 0 ? 0 : PI / 2);
  431. var start2 = start > -b ? start : (end > -b ? -b : start),
  432. end2 = end < PI - a ? end : (start < PI - a ? PI - a : end);
  433. var out = ['M', cx + (rx * cos(start2)), cy + (ry * sin(start2))];
  434. out = out.concat(curveTo(cx, cy, rx, ry, start2, end2, 0, 0));
  435. // When slice goes over middle, need to add both, left and right outer side:
  436. if (end > PI - a && start < PI - a) {
  437. // Go to outer side
  438. out = out.concat([
  439. 'L', cx + (rx * cos(end2)) + dx, cy + (ry * sin(end2)) + dy
  440. ]);
  441. // Curve to the true end of the slice
  442. out = out.concat(curveTo(cx, cy, rx, ry, end2, end, dx, dy));
  443. // Go to the inner side
  444. out = out.concat([
  445. 'L', cx + (rx * cos(end)), cy + (ry * sin(end))
  446. ]);
  447. // Go back to the artifical end2
  448. out = out.concat(curveTo(cx, cy, rx, ry, end, end2, 0, 0));
  449. }
  450. out = out.concat([
  451. 'L', cx + (rx * cos(end2)) + dx, cy + (ry * sin(end2)) + dy
  452. ]);
  453. out = out.concat(curveTo(cx, cy, rx, ry, end2, start2, dx, dy));
  454. out = out.concat(['Z']);
  455. // INSIDE
  456. var inn = ['M', cx + (irx * cs), cy + (iry * ss)];
  457. inn = inn.concat(curveTo(cx, cy, irx, iry, start, end, 0, 0));
  458. inn = inn.concat([
  459. 'L', cx + (irx * cos(end)) + dx, cy + (iry * sin(end)) + dy
  460. ]);
  461. inn = inn.concat(curveTo(cx, cy, irx, iry, end, start, dx, dy));
  462. inn = inn.concat(['Z']);
  463. // SIDES
  464. var side1 = [
  465. 'M', cx + (rx * cs), cy + (ry * ss),
  466. 'L', cx + (rx * cs) + dx, cy + (ry * ss) + dy,
  467. 'L', cx + (irx * cs) + dx, cy + (iry * ss) + dy,
  468. 'L', cx + (irx * cs), cy + (iry * ss),
  469. 'Z'
  470. ];
  471. var side2 = [
  472. 'M', cx + (rx * ce), cy + (ry * se),
  473. 'L', cx + (rx * ce) + dx, cy + (ry * se) + dy,
  474. 'L', cx + (irx * ce) + dx, cy + (iry * se) + dy,
  475. 'L', cx + (irx * ce), cy + (iry * se),
  476. 'Z'
  477. ];
  478. // correction for changed position of vanishing point caused by alpha and beta rotations
  479. var angleCorr = Math.atan2(dy, -dx),
  480. angleEnd = Math.abs(end + angleCorr),
  481. angleStart = Math.abs(start + angleCorr),
  482. angleMid = Math.abs((start + end) / 2 + angleCorr);
  483. // set to 0-PI range
  484. function toZeroPIRange(angle) {
  485. angle = angle % (2 * PI);
  486. if (angle > PI) {
  487. angle = 2 * PI - angle;
  488. }
  489. return angle;
  490. }
  491. angleEnd = toZeroPIRange(angleEnd);
  492. angleStart = toZeroPIRange(angleStart);
  493. angleMid = toZeroPIRange(angleMid);
  494. // *1e5 is to compensate pInt in zIndexSetter
  495. var incPrecision = 1e5,
  496. a1 = angleMid * incPrecision,
  497. a2 = angleStart * incPrecision,
  498. a3 = angleEnd * incPrecision;
  499. return {
  500. top: top,
  501. zTop: PI * incPrecision + 1, // max angle is PI, so this is allways higher
  502. out: out,
  503. zOut: Math.max(a1, a2, a3),
  504. inn: inn,
  505. zInn: Math.max(a1, a2, a3),
  506. side1: side1,
  507. zSide1: a3 * 0.99, // to keep below zOut and zInn in case of same values
  508. side2: side2,
  509. zSide2: a2 * 0.99
  510. };
  511. };
  512. /***
  513. EXTENSION FOR 3D CHARTS
  514. ***/
  515. // Shorthand to check the is3d flag
  516. Highcharts.Chart.prototype.is3d = function () {
  517. return this.options.chart.options3d && this.options.chart.options3d.enabled; // #4280
  518. };
  519. Highcharts.wrap(Highcharts.Chart.prototype, 'isInsidePlot', function (proceed) {
  520. if (this.is3d()) {
  521. return true;
  522. } else {
  523. return proceed.apply(this, [].slice.call(arguments, 1));
  524. }
  525. });
  526. var defaultChartOptions = Highcharts.getOptions();
  527. defaultChartOptions.chart.options3d = {
  528. enabled: false,
  529. alpha: 0,
  530. beta: 0,
  531. depth: 100,
  532. viewDistance: 25,
  533. frame: {
  534. bottom: { size: 1, color: 'rgba(255,255,255,0)' },
  535. side: { size: 1, color: 'rgba(255,255,255,0)' },
  536. back: { size: 1, color: 'rgba(255,255,255,0)' }
  537. }
  538. };
  539. Highcharts.wrap(Highcharts.Chart.prototype, 'init', function (proceed) {
  540. var args = [].slice.call(arguments, 1),
  541. plotOptions,
  542. pieOptions;
  543. if (args[0].chart.options3d && args[0].chart.options3d.enabled) {
  544. // Normalize alpha and beta to (-360, 360) range
  545. args[0].chart.options3d.alpha = (args[0].chart.options3d.alpha || 0) % 360;
  546. args[0].chart.options3d.beta = (args[0].chart.options3d.beta || 0) % 360;
  547. plotOptions = args[0].plotOptions || {};
  548. pieOptions = plotOptions.pie || {};
  549. pieOptions.borderColor = Highcharts.pick(pieOptions.borderColor, undefined);
  550. }
  551. proceed.apply(this, args);
  552. });
  553. Highcharts.wrap(Highcharts.Chart.prototype, 'setChartSize', function (proceed) {
  554. proceed.apply(this, [].slice.call(arguments, 1));
  555. if (this.is3d()) {
  556. var inverted = this.inverted,
  557. clipBox = this.clipBox,
  558. margin = this.margin,
  559. x = inverted ? 'y' : 'x',
  560. y = inverted ? 'x' : 'y',
  561. w = inverted ? 'height' : 'width',
  562. h = inverted ? 'width' : 'height';
  563. clipBox[x] = -(margin[3] || 0);
  564. clipBox[y] = -(margin[0] || 0);
  565. clipBox[w] = this.chartWidth + (margin[3] || 0) + (margin[1] || 0);
  566. clipBox[h] = this.chartHeight + (margin[0] || 0) + (margin[2] || 0);
  567. }
  568. });
  569. Highcharts.wrap(Highcharts.Chart.prototype, 'redraw', function (proceed) {
  570. if (this.is3d()) {
  571. // Set to force a redraw of all elements
  572. this.isDirtyBox = true;
  573. }
  574. proceed.apply(this, [].slice.call(arguments, 1));
  575. });
  576. // Draw the series in the reverse order (#3803, #3917)
  577. Highcharts.wrap(Highcharts.Chart.prototype, 'renderSeries', function (proceed) {
  578. var series,
  579. i = this.series.length;
  580. if (this.is3d()) {
  581. while (i--) {
  582. series = this.series[i];
  583. series.translate();
  584. series.render();
  585. }
  586. } else {
  587. proceed.call(this);
  588. }
  589. });
  590. Highcharts.Chart.prototype.retrieveStacks = function (stacking) {
  591. var series = this.series,
  592. stacks = {},
  593. stackNumber,
  594. i = 1;
  595. Highcharts.each(this.series, function (s) {
  596. stackNumber = pick(s.options.stack, (stacking ? 0 : series.length - 1 - s.index)); // #3841, #4532
  597. if (!stacks[stackNumber]) {
  598. stacks[stackNumber] = { series: [s], position: i};
  599. i++;
  600. } else {
  601. stacks[stackNumber].series.push(s);
  602. }
  603. });
  604. stacks.totalStacks = i + 1;
  605. return stacks;
  606. };
  607. /***
  608. EXTENSION TO THE AXIS
  609. ***/
  610. Highcharts.wrap(Highcharts.Axis.prototype, 'setOptions', function (proceed, userOptions) {
  611. var options;
  612. proceed.call(this, userOptions);
  613. if (this.chart.is3d()) {
  614. options = this.options;
  615. options.tickWidth = Highcharts.pick(options.tickWidth, 0);
  616. options.gridLineWidth = Highcharts.pick(options.gridLineWidth, 1);
  617. }
  618. });
  619. Highcharts.wrap(Highcharts.Axis.prototype, 'render', function (proceed) {
  620. proceed.apply(this, [].slice.call(arguments, 1));
  621. // Do not do this if the chart is not 3D
  622. if (!this.chart.is3d()) {
  623. return;
  624. }
  625. var chart = this.chart,
  626. renderer = chart.renderer,
  627. options3d = chart.options.chart.options3d,
  628. frame = options3d.frame,
  629. fbottom = frame.bottom,
  630. fback = frame.back,
  631. fside = frame.side,
  632. depth = options3d.depth,
  633. height = this.height,
  634. width = this.width,
  635. left = this.left,
  636. top = this.top;
  637. if (this.isZAxis) {
  638. return;
  639. } else if (this.horiz) {
  640. var bottomShape = {
  641. x: left,
  642. y: top + (chart.xAxis[0].opposite ? -fbottom.size : height),
  643. z: 0,
  644. width: width,
  645. height: fbottom.size,
  646. depth: depth,
  647. insidePlotArea: false
  648. };
  649. if (!this.bottomFrame) {
  650. this.bottomFrame = renderer.cuboid(bottomShape).attr({
  651. fill: fbottom.color,
  652. zIndex: (chart.yAxis[0].reversed && options3d.alpha > 0 ? 4 : -1)
  653. })
  654. .css({
  655. stroke: fbottom.color
  656. }).add();
  657. } else {
  658. this.bottomFrame.animate(bottomShape);
  659. }
  660. } else {
  661. // BACK
  662. var backShape = {
  663. x: left + (chart.yAxis[0].opposite ? 0 : -fside.size),
  664. y: top + (chart.xAxis[0].opposite ? -fbottom.size : 0),
  665. z: depth,
  666. width: width + fside.size,
  667. height: height + fbottom.size,
  668. depth: fback.size,
  669. insidePlotArea: false
  670. };
  671. if (!this.backFrame) {
  672. this.backFrame = renderer.cuboid(backShape).attr({
  673. fill: fback.color,
  674. zIndex: -3
  675. }).css({
  676. stroke: fback.color
  677. }).add();
  678. } else {
  679. this.backFrame.animate(backShape);
  680. }
  681. var sideShape = {
  682. x: left + (chart.yAxis[0].opposite ? width : -fside.size),
  683. y: top + (chart.xAxis[0].opposite ? -fbottom.size : 0),
  684. z: 0,
  685. width: fside.size,
  686. height: height + fbottom.size,
  687. depth: depth,
  688. insidePlotArea: false
  689. };
  690. if (!this.sideFrame) {
  691. this.sideFrame = renderer.cuboid(sideShape).attr({
  692. fill: fside.color,
  693. zIndex: -2
  694. }).css({
  695. stroke: fside.color
  696. }).add();
  697. } else {
  698. this.sideFrame.animate(sideShape);
  699. }
  700. }
  701. });
  702. Highcharts.wrap(Highcharts.Axis.prototype, 'getPlotLinePath', function (proceed) {
  703. var path = proceed.apply(this, [].slice.call(arguments, 1));
  704. // Do not do this if the chart is not 3D
  705. if (!this.chart.is3d()) {
  706. return path;
  707. }
  708. if (path === null) { return path; }
  709. var chart = this.chart,
  710. options3d = chart.options.chart.options3d;
  711. var d = this.isZAxis ? this.chart.plotWidth : options3d.depth,
  712. opposite = this.opposite;
  713. if (this.horiz) {
  714. opposite = !opposite;
  715. }
  716. var pArr = [
  717. this.swapZ({ x: path[1], y: path[2], z: (opposite ? d : 0)}),
  718. this.swapZ({ x: path[1], y: path[2], z: d }),
  719. this.swapZ({ x: path[4], y: path[5], z: d }),
  720. this.swapZ({ x: path[4], y: path[5], z: (opposite ? 0 : d)})
  721. ];
  722. pArr = perspective(pArr, this.chart, false);
  723. path = this.chart.renderer.toLinePath(pArr, false);
  724. return path;
  725. });
  726. // Do not draw axislines in 3D
  727. Highcharts.wrap(Highcharts.Axis.prototype, 'getLinePath', function (proceed) {
  728. return this.chart.is3d() ? [] : proceed.apply(this, [].slice.call(arguments, 1));
  729. });
  730. Highcharts.wrap(Highcharts.Axis.prototype, 'getPlotBandPath', function (proceed) {
  731. // Do not do this if the chart is not 3D
  732. if (!this.chart.is3d()) {
  733. return proceed.apply(this, [].slice.call(arguments, 1));
  734. } else {
  735. var args = arguments,
  736. from = args[1],
  737. to = args[2];
  738. var toPath = this.getPlotLinePath(to),
  739. path = this.getPlotLinePath(from);
  740. if (path && toPath) {
  741. path.push(
  742. 'L',
  743. toPath[10], // These two do not exist in the regular getPlotLine
  744. toPath[11], // ---- # 3005
  745. 'L',
  746. toPath[7],
  747. toPath[8],
  748. 'L',
  749. toPath[4],
  750. toPath[5],
  751. 'L',
  752. toPath[1],
  753. toPath[2]
  754. );
  755. } else { // outside the axis area
  756. path = null;
  757. }
  758. return path;
  759. }
  760. });
  761. /***
  762. EXTENSION TO THE TICKS
  763. ***/
  764. Highcharts.wrap(Highcharts.Tick.prototype, 'getMarkPath', function (proceed) {
  765. var path = proceed.apply(this, [].slice.call(arguments, 1));
  766. // Do not do this if the chart is not 3D
  767. if (!this.axis.chart.is3d()) {
  768. return path;
  769. }
  770. var pArr = [
  771. this.axis.swapZ({x: path[1], y: path[2], z: 0}),
  772. this.axis.swapZ({x: path[4], y: path[5], z: 0})
  773. ];
  774. pArr = perspective(pArr, this.axis.chart, false);
  775. path = [
  776. 'M', pArr[0].x, pArr[0].y,
  777. 'L', pArr[1].x, pArr[1].y
  778. ];
  779. return path;
  780. });
  781. Highcharts.wrap(Highcharts.Tick.prototype, 'getLabelPosition', function (proceed) {
  782. var pos = proceed.apply(this, [].slice.call(arguments, 1));
  783. // Do not do this if the chart is not 3D
  784. if (!this.axis.chart.is3d()) {
  785. return pos;
  786. }
  787. var new_pos = perspective([this.axis.swapZ({x: pos.x, y: pos.y, z: 0})], this.axis.chart, false)[0];
  788. new_pos.x = new_pos.x - (!this.axis.horiz && this.axis.opposite ? this.axis.transA : 0); //#3788
  789. new_pos.old = pos;
  790. return new_pos;
  791. });
  792. Highcharts.wrap(Highcharts.Tick.prototype, 'handleOverflow', function (proceed, xy) {
  793. if (this.axis.chart.is3d()) {
  794. xy = xy.old;
  795. }
  796. return proceed.call(this, xy);
  797. });
  798. Highcharts.wrap(Highcharts.Axis.prototype, 'getTitlePosition', function (proceed) {
  799. var pos = proceed.apply(this, [].slice.call(arguments, 1));
  800. // Do not do this if the chart is not 3D
  801. if (!this.chart.is3d()) {
  802. return pos;
  803. }
  804. pos = perspective([this.swapZ({x: pos.x, y: pos.y, z: 0})], this.chart, false)[0];
  805. return pos;
  806. });
  807. Highcharts.wrap(Highcharts.Axis.prototype, 'drawCrosshair', function (proceed) {
  808. var args = arguments;
  809. if (this.chart.is3d()) {
  810. if (args[2]) {
  811. args[2] = {
  812. plotX: args[2].plotXold || args[2].plotX,
  813. plotY: args[2].plotYold || args[2].plotY
  814. };
  815. }
  816. }
  817. proceed.apply(this, [].slice.call(args, 1));
  818. });
  819. /***
  820. Z-AXIS
  821. ***/
  822. Highcharts.Axis.prototype.swapZ = function (p, insidePlotArea) {
  823. if (this.isZAxis) {
  824. var plotLeft = insidePlotArea ? 0 : this.chart.plotLeft;
  825. var chart = this.chart;
  826. return {
  827. x: plotLeft + (chart.yAxis[0].opposite ? p.z : chart.xAxis[0].width - p.z),
  828. y: p.y,
  829. z: p.x - plotLeft
  830. };
  831. } else {
  832. return p;
  833. }
  834. };
  835. var ZAxis = Highcharts.ZAxis = function () {
  836. this.isZAxis = true;
  837. this.init.apply(this, arguments);
  838. };
  839. Highcharts.extend(ZAxis.prototype, Highcharts.Axis.prototype);
  840. Highcharts.extend(ZAxis.prototype, {
  841. setOptions: function (userOptions) {
  842. userOptions = Highcharts.merge({
  843. offset: 0,
  844. lineWidth: 0
  845. }, userOptions);
  846. Highcharts.Axis.prototype.setOptions.call(this, userOptions);
  847. this.coll = 'zAxis';
  848. },
  849. setAxisSize: function () {
  850. Highcharts.Axis.prototype.setAxisSize.call(this);
  851. this.width = this.len = this.chart.options.chart.options3d.depth;
  852. this.right = this.chart.chartWidth - this.width - this.left;
  853. },
  854. getSeriesExtremes: function () {
  855. var axis = this,
  856. chart = axis.chart;
  857. axis.hasVisibleSeries = false;
  858. // Reset properties in case we're redrawing (#3353)
  859. axis.dataMin = axis.dataMax = axis.ignoreMinPadding = axis.ignoreMaxPadding = null;
  860. if (axis.buildStacks) {
  861. axis.buildStacks();
  862. }
  863. // loop through this axis' series
  864. Highcharts.each(axis.series, function (series) {
  865. if (series.visible || !chart.options.chart.ignoreHiddenSeries) {
  866. var seriesOptions = series.options,
  867. zData,
  868. threshold = seriesOptions.threshold;
  869. axis.hasVisibleSeries = true;
  870. // Validate threshold in logarithmic axes
  871. if (axis.isLog && threshold <= 0) {
  872. threshold = null;
  873. }
  874. zData = series.zData;
  875. if (zData.length) {
  876. axis.dataMin = Math.min(pick(axis.dataMin, zData[0]), Math.min.apply(null, zData));
  877. axis.dataMax = Math.max(pick(axis.dataMax, zData[0]), Math.max.apply(null, zData));
  878. }
  879. }
  880. });
  881. }
  882. });
  883. /**
  884. * Extend the chart getAxes method to also get the color axis
  885. */
  886. Highcharts.wrap(Highcharts.Chart.prototype, 'getAxes', function (proceed) {
  887. var chart = this,
  888. options = this.options,
  889. zAxisOptions = options.zAxis = Highcharts.splat(options.zAxis || {});
  890. proceed.call(this);
  891. if (!chart.is3d()) {
  892. return;
  893. }
  894. this.zAxis = [];
  895. Highcharts.each(zAxisOptions, function (axisOptions, i) {
  896. axisOptions.index = i;
  897. axisOptions.isX = true; //Z-Axis is shown horizontally, so it's kind of a X-Axis
  898. var zAxis = new ZAxis(chart, axisOptions);
  899. zAxis.setScale();
  900. });
  901. });
  902. /***
  903. EXTENSION FOR 3D COLUMNS
  904. ***/
  905. Highcharts.wrap(Highcharts.seriesTypes.column.prototype, 'translate', function (proceed) {
  906. proceed.apply(this, [].slice.call(arguments, 1));
  907. // Do not do this if the chart is not 3D
  908. if (!this.chart.is3d()) {
  909. return;
  910. }
  911. var series = this,
  912. chart = series.chart,
  913. seriesOptions = series.options,
  914. depth = seriesOptions.depth || 25;
  915. var stack = seriesOptions.stacking ? (seriesOptions.stack || 0) : series._i;
  916. var z = stack * (depth + (seriesOptions.groupZPadding || 1));
  917. if (seriesOptions.grouping !== false) { z = 0; }
  918. z += (seriesOptions.groupZPadding || 1);
  919. Highcharts.each(series.data, function (point) {
  920. if (point.y !== null) {
  921. var shapeArgs = point.shapeArgs,
  922. tooltipPos = point.tooltipPos;
  923. point.shapeType = 'cuboid';
  924. shapeArgs.z = z;
  925. shapeArgs.depth = depth;
  926. shapeArgs.insidePlotArea = true;
  927. // Translate the tooltip position in 3d space
  928. tooltipPos = perspective([{ x: tooltipPos[0], y: tooltipPos[1], z: z }], chart, false)[0];
  929. point.tooltipPos = [tooltipPos.x, tooltipPos.y];
  930. }
  931. });
  932. // store for later use #4067
  933. series.z = z;
  934. });
  935. Highcharts.wrap(Highcharts.seriesTypes.column.prototype, 'animate', function (proceed) {
  936. if (!this.chart.is3d()) {
  937. proceed.apply(this, [].slice.call(arguments, 1));
  938. } else {
  939. var args = arguments,
  940. init = args[1],
  941. yAxis = this.yAxis,
  942. series = this,
  943. reversed = this.yAxis.reversed;
  944. if (Highcharts.svg) { // VML is too slow anyway
  945. if (init) {
  946. Highcharts.each(series.data, function (point) {
  947. if (point.y !== null) {
  948. point.height = point.shapeArgs.height;
  949. point.shapey = point.shapeArgs.y; //#2968
  950. point.shapeArgs.height = 1;
  951. if (!reversed) {
  952. if (point.stackY) {
  953. point.shapeArgs.y = point.plotY + yAxis.translate(point.stackY);
  954. } else {
  955. point.shapeArgs.y = point.plotY + (point.negative ? -point.height : point.height);
  956. }
  957. }
  958. }
  959. });
  960. } else { // run the animation
  961. Highcharts.each(series.data, function (point) {
  962. if (point.y !== null) {
  963. point.shapeArgs.height = point.height;
  964. point.shapeArgs.y = point.shapey; //#2968
  965. // null value do not have a graphic
  966. if (point.graphic) {
  967. point.graphic.animate(point.shapeArgs, series.options.animation);
  968. }
  969. }
  970. });
  971. // redraw datalabels to the correct position
  972. this.drawDataLabels();
  973. // delete this function to allow it only once
  974. series.animate = null;
  975. }
  976. }
  977. }
  978. });
  979. Highcharts.wrap(Highcharts.seriesTypes.column.prototype, 'init', function (proceed) {
  980. proceed.apply(this, [].slice.call(arguments, 1));
  981. if (this.chart.is3d()) {
  982. var seriesOptions = this.options,
  983. grouping = seriesOptions.grouping,
  984. stacking = seriesOptions.stacking,
  985. z = 0;
  986. if (!(grouping !== undefined && !grouping)) {
  987. var stacks = this.chart.retrieveStacks(stacking),
  988. stack = seriesOptions.stack || 0,
  989. i; // position within the stack
  990. for (i = 0; i < stacks[stack].series.length; i++) {
  991. if (stacks[stack].series[i] === this) {
  992. break;
  993. }
  994. }
  995. z = (stacks.totalStacks * 10) - (10 * (stacks.totalStacks - stacks[stack].position)) - i;
  996. }
  997. seriesOptions.zIndex = z;
  998. }
  999. });
  1000. function draw3DPoints(proceed) {
  1001. // Do not do this if the chart is not 3D
  1002. if (this.chart.is3d()) {
  1003. var grouping = this.chart.options.plotOptions.column.grouping;
  1004. if (grouping !== undefined && !grouping && this.group.zIndex !== undefined && !this.zIndexSet) {
  1005. this.group.attr({zIndex : (this.group.zIndex * 10)});
  1006. this.zIndexSet = true; // #4062 set zindex only once
  1007. }
  1008. var options = this.options,
  1009. states = this.options.states;
  1010. this.borderWidth = options.borderWidth = defined(options.edgeWidth) ? options.edgeWidth : 1; //#4055
  1011. Highcharts.each(this.data, function (point) {
  1012. if (point.y !== null) {
  1013. var pointAttr = point.pointAttr;
  1014. // Set the border color to the fill color to provide a smooth edge
  1015. this.borderColor = Highcharts.pick(options.edgeColor, pointAttr[''].fill);
  1016. pointAttr[''].stroke = this.borderColor;
  1017. pointAttr.hover.stroke = Highcharts.pick(states.hover.edgeColor, this.borderColor);
  1018. pointAttr.select.stroke = Highcharts.pick(states.select.edgeColor, this.borderColor);
  1019. }
  1020. });
  1021. }
  1022. proceed.apply(this, [].slice.call(arguments, 1));
  1023. }
  1024. Highcharts.wrap(Highcharts.Series.prototype, 'alignDataLabel', function (proceed) {
  1025. // Only do this for 3D columns and columnranges
  1026. if (this.chart.is3d() && (this.type === 'column' || this.type === 'columnrange')) {
  1027. var series = this,
  1028. chart = series.chart;
  1029. var args = arguments,
  1030. alignTo = args[4];
  1031. var pos = ({x: alignTo.x, y: alignTo.y, z: series.z});
  1032. pos = perspective([pos], chart, true)[0];
  1033. alignTo.x = pos.x;
  1034. alignTo.y = pos.y;
  1035. }
  1036. proceed.apply(this, [].slice.call(arguments, 1));
  1037. });
  1038. if (Highcharts.seriesTypes.columnrange) {
  1039. Highcharts.wrap(Highcharts.seriesTypes.columnrange.prototype, 'drawPoints', draw3DPoints);
  1040. }
  1041. Highcharts.wrap(Highcharts.seriesTypes.column.prototype, 'drawPoints', draw3DPoints);
  1042. /***
  1043. EXTENSION FOR 3D CYLINDRICAL COLUMNS
  1044. Not supported
  1045. ***/
  1046. /*
  1047. var defaultOptions = Highcharts.getOptions();
  1048. defaultOptions.plotOptions.cylinder = Highcharts.merge(defaultOptions.plotOptions.column);
  1049. var CylinderSeries = Highcharts.extendClass(Highcharts.seriesTypes.column, {
  1050. type: 'cylinder'
  1051. });
  1052. Highcharts.seriesTypes.cylinder = CylinderSeries;
  1053. Highcharts.wrap(Highcharts.seriesTypes.cylinder.prototype, 'translate', function (proceed) {
  1054. proceed.apply(this, [].slice.call(arguments, 1));
  1055. // Do not do this if the chart is not 3D
  1056. if (!this.chart.is3d()) {
  1057. return;
  1058. }
  1059. var series = this,
  1060. chart = series.chart,
  1061. options = chart.options,
  1062. cylOptions = options.plotOptions.cylinder,
  1063. options3d = options.chart.options3d,
  1064. depth = cylOptions.depth || 0,
  1065. origin = {
  1066. x: chart.inverted ? chart.plotHeight / 2 : chart.plotWidth / 2,
  1067. y: chart.inverted ? chart.plotWidth / 2 : chart.plotHeight / 2,
  1068. z: options3d.depth,
  1069. vd: options3d.viewDistance
  1070. },
  1071. alpha = options3d.alpha;
  1072. var z = cylOptions.stacking ? (this.options.stack || 0) * depth : series._i * depth;
  1073. z += depth / 2;
  1074. if (cylOptions.grouping !== false) { z = 0; }
  1075. Highcharts.each(series.data, function (point) {
  1076. var shapeArgs = point.shapeArgs;
  1077. point.shapeType = 'arc3d';
  1078. shapeArgs.x += depth / 2;
  1079. shapeArgs.z = z;
  1080. shapeArgs.start = 0;
  1081. shapeArgs.end = 2 * PI;
  1082. shapeArgs.r = depth * 0.95;
  1083. shapeArgs.innerR = 0;
  1084. shapeArgs.depth = shapeArgs.height * (1 / sin((90 - alpha) * deg2rad)) - z;
  1085. shapeArgs.alpha = 90 - alpha;
  1086. shapeArgs.beta = 0;
  1087. shapeArgs.origin = origin;
  1088. });
  1089. });
  1090. *//***
  1091. EXTENSION FOR 3D PIES
  1092. ***/
  1093. Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'translate', function (proceed) {
  1094. proceed.apply(this, [].slice.call(arguments, 1));
  1095. // Do not do this if the chart is not 3D
  1096. if (!this.chart.is3d()) {
  1097. return;
  1098. }
  1099. var series = this,
  1100. chart = series.chart,
  1101. options = chart.options,
  1102. seriesOptions = series.options,
  1103. depth = seriesOptions.depth || 0,
  1104. options3d = options.chart.options3d,
  1105. origin = {
  1106. x: chart.plotWidth / 2,
  1107. y: chart.plotHeight / 2,
  1108. z: options3d.depth
  1109. },
  1110. alpha = options3d.alpha,
  1111. beta = options3d.beta,
  1112. z = seriesOptions.stacking ? (seriesOptions.stack || 0) * depth : series._i * depth;
  1113. z += depth / 2;
  1114. if (seriesOptions.grouping !== false) { z = 0; }
  1115. Highcharts.each(series.data, function (point) {
  1116. var shapeArgs = point.shapeArgs,
  1117. angle;
  1118. point.shapeType = 'arc3d';
  1119. shapeArgs.z = z;
  1120. shapeArgs.depth = depth * 0.75;
  1121. shapeArgs.origin = origin;
  1122. shapeArgs.alpha = alpha;
  1123. shapeArgs.beta = beta;
  1124. shapeArgs.center = series.center;
  1125. angle = (shapeArgs.end + shapeArgs.start) / 2;
  1126. point.slicedTranslation = {
  1127. translateX : round(cos(angle) * seriesOptions.slicedOffset * cos(alpha * deg2rad)),
  1128. translateY : round(sin(angle) * seriesOptions.slicedOffset * cos(alpha * deg2rad))
  1129. };
  1130. });
  1131. });
  1132. Highcharts.wrap(Highcharts.seriesTypes.pie.prototype.pointClass.prototype, 'haloPath', function (proceed) {
  1133. var args = arguments;
  1134. return this.series.chart.is3d() ? [] : proceed.call(this, args[1]);
  1135. });
  1136. Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'drawPoints', function (proceed) {
  1137. var seriesGroup = this.group,
  1138. options = this.options,
  1139. states = options.states;
  1140. // Do not do this if the chart is not 3D
  1141. if (this.chart.is3d()) {
  1142. // Set the border color to the fill color to provide a smooth edge
  1143. this.borderWidth = options.borderWidth = options.edgeWidth || 1;
  1144. this.borderColor = options.edgeColor = Highcharts.pick(options.edgeColor, options.borderColor, undefined);
  1145. states.hover.borderColor = Highcharts.pick(states.hover.edgeColor, this.borderColor);
  1146. states.hover.borderWidth = Highcharts.pick(states.hover.edgeWidth, this.borderWidth);
  1147. states.select.borderColor = Highcharts.pick(states.select.edgeColor, this.borderColor);
  1148. states.select.borderWidth = Highcharts.pick(states.select.edgeWidth, this.borderWidth);
  1149. Highcharts.each(this.data, function (point) {
  1150. var pointAttr = point.pointAttr;
  1151. pointAttr[''].stroke = point.series.borderColor || point.color;
  1152. pointAttr['']['stroke-width'] = point.series.borderWidth;
  1153. pointAttr.hover.stroke = states.hover.borderColor;
  1154. pointAttr.hover['stroke-width'] = states.hover.borderWidth;
  1155. pointAttr.select.stroke = states.select.borderColor;
  1156. pointAttr.select['stroke-width'] = states.select.borderWidth;
  1157. });
  1158. }
  1159. proceed.apply(this, [].slice.call(arguments, 1));
  1160. if (this.chart.is3d()) {
  1161. Highcharts.each(this.points, function (point) {
  1162. var graphic = point.graphic;
  1163. graphic.out.add(seriesGroup);
  1164. graphic.inn.add(seriesGroup);
  1165. graphic.side1.add(seriesGroup);
  1166. graphic.side2.add(seriesGroup);
  1167. // Hide null or 0 points (#3006, 3650)
  1168. graphic[point.y ? 'show' : 'hide']();
  1169. });
  1170. }
  1171. });
  1172. Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'drawDataLabels', function (proceed) {
  1173. if (this.chart.is3d()) {
  1174. var series = this;
  1175. Highcharts.each(series.data, function (point) {
  1176. var shapeArgs = point.shapeArgs,
  1177. r = shapeArgs.r,
  1178. d = shapeArgs.depth,
  1179. a1 = (shapeArgs.alpha || series.chart.options.chart.options3d.alpha) * deg2rad, //#3240 issue with datalabels for 0 and null values
  1180. a2 = (shapeArgs.start + shapeArgs.end) / 2,
  1181. labelPos = point.labelPos;
  1182. labelPos[1] += (-r * (1 - cos(a1)) * sin(a2)) + (sin(a2) > 0 ? sin(a1) * d : 0);
  1183. labelPos[3] += (-r * (1 - cos(a1)) * sin(a2)) + (sin(a2) > 0 ? sin(a1) * d : 0);
  1184. labelPos[5] += (-r * (1 - cos(a1)) * sin(a2)) + (sin(a2) > 0 ? sin(a1) * d : 0);
  1185. });
  1186. }
  1187. proceed.apply(this, [].slice.call(arguments, 1));
  1188. });
  1189. Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'addPoint', function (proceed) {
  1190. proceed.apply(this, [].slice.call(arguments, 1));
  1191. if (this.chart.is3d()) {
  1192. // destroy (and rebuild) everything!!!
  1193. this.update(this.userOptions, true); // #3845 pass the old options
  1194. }
  1195. });
  1196. Highcharts.wrap(Highcharts.seriesTypes.pie.prototype, 'animate', function (proceed) {
  1197. if (!this.chart.is3d()) {
  1198. proceed.apply(this, [].slice.call(arguments, 1));
  1199. } else {
  1200. var args = arguments,
  1201. init = args[1],
  1202. animation = this.options.animation,
  1203. attribs,
  1204. center = this.center,
  1205. group = this.group,
  1206. markerGroup = this.markerGroup;
  1207. if (Highcharts.svg) { // VML is too slow anyway
  1208. if (animation === true) {
  1209. animation = {};
  1210. }
  1211. // Initialize the animation
  1212. if (init) {
  1213. // Scale down the group and place it in the center
  1214. group.oldtranslateX = group.translateX;
  1215. group.oldtranslateY = group.translateY;
  1216. attribs = {
  1217. translateX: center[0],
  1218. translateY: center[1],
  1219. scaleX: 0.001, // #1499
  1220. scaleY: 0.001
  1221. };
  1222. group.attr(attribs);
  1223. if (markerGroup) {
  1224. markerGroup.attrSetters = group.attrSetters;
  1225. markerGroup.attr(attribs);
  1226. }
  1227. // Run the animation
  1228. } else {
  1229. attribs = {
  1230. translateX: group.oldtranslateX,
  1231. translateY: group.oldtranslateY,
  1232. scaleX: 1,
  1233. scaleY: 1
  1234. };
  1235. group.animate(attribs, animation);
  1236. if (markerGroup) {
  1237. markerGroup.animate(attribs, animation);
  1238. }
  1239. // Delete this function to allow it only once
  1240. this.animate = null;
  1241. }
  1242. }
  1243. }
  1244. });/***
  1245. EXTENSION FOR 3D SCATTER CHART
  1246. ***/
  1247. Highcharts.wrap(Highcharts.seriesTypes.scatter.prototype, 'translate', function (proceed) {
  1248. //function translate3d(proceed) {
  1249. proceed.apply(this, [].slice.call(arguments, 1));
  1250. if (!this.chart.is3d()) {
  1251. return;
  1252. }
  1253. var series = this,
  1254. chart = series.chart,
  1255. zAxis = Highcharts.pick(series.zAxis, chart.options.zAxis[0]),
  1256. rawPoints = [],
  1257. rawPoint,
  1258. projectedPoints,
  1259. projectedPoint,
  1260. zValue,
  1261. i;
  1262. for (i = 0; i < series.data.length; i++) {
  1263. rawPoint = series.data[i];
  1264. zValue = zAxis.isLog && zAxis.val2lin ? zAxis.val2lin(rawPoint.z) : rawPoint.z; // #4562
  1265. rawPoint.plotZ = zAxis.translate(zValue);
  1266. rawPoint.isInside = rawPoint.isInside ? (zValue >= zAxis.min && zValue <= zAxis.max) : false;
  1267. rawPoints.push({
  1268. x: rawPoint.plotX,
  1269. y: rawPoint.plotY,
  1270. z: rawPoint.plotZ
  1271. });
  1272. }
  1273. projectedPoints = perspective(rawPoints, chart, true);
  1274. for (i = 0; i < series.data.length; i++) {
  1275. rawPoint = series.data[i];
  1276. projectedPoint = projectedPoints[i];
  1277. rawPoint.plotXold = rawPoint.plotX;
  1278. rawPoint.plotYold = rawPoint.plotY;
  1279. rawPoint.plotX = projectedPoint.x;
  1280. rawPoint.plotY = projectedPoint.y;
  1281. rawPoint.plotZ = projectedPoint.z;
  1282. }
  1283. });
  1284. Highcharts.wrap(Highcharts.seriesTypes.scatter.prototype, 'init', function (proceed, chart, options) {
  1285. if (chart.is3d()) {
  1286. // add a third coordinate
  1287. this.axisTypes = ['xAxis', 'yAxis', 'zAxis'];
  1288. this.pointArrayMap = ['x', 'y', 'z'];
  1289. this.parallelArrays = ['x', 'y', 'z'];
  1290. }
  1291. var result = proceed.apply(this, [chart, options]);
  1292. if (this.chart.is3d()) {
  1293. // Set a new default tooltip formatter
  1294. var default3dScatterTooltip = 'x: <b>{point.x}</b><br/>y: <b>{point.y}</b><br/>z: <b>{point.z}</b><br/>';
  1295. if (this.userOptions.tooltip) {
  1296. this.tooltipOptions.pointFormat = this.userOptions.tooltip.pointFormat || default3dScatterTooltip;
  1297. } else {
  1298. this.tooltipOptions.pointFormat = default3dScatterTooltip;
  1299. }
  1300. }
  1301. return result;
  1302. });
  1303. /**
  1304. * Extension to the VML Renderer
  1305. */
  1306. if (Highcharts.VMLRenderer) {
  1307. Highcharts.setOptions({animate: false});
  1308. Highcharts.VMLRenderer.prototype.cuboid = Highcharts.SVGRenderer.prototype.cuboid;
  1309. Highcharts.VMLRenderer.prototype.cuboidPath = Highcharts.SVGRenderer.prototype.cuboidPath;
  1310. Highcharts.VMLRenderer.prototype.toLinePath = Highcharts.SVGRenderer.prototype.toLinePath;
  1311. Highcharts.VMLRenderer.prototype.createElement3D = Highcharts.SVGRenderer.prototype.createElement3D;
  1312. Highcharts.VMLRenderer.prototype.arc3d = function (shapeArgs) {
  1313. var result = Highcharts.SVGRenderer.prototype.arc3d.call(this, shapeArgs);
  1314. result.css({zIndex: result.zIndex});
  1315. return result;
  1316. };
  1317. Highcharts.VMLRenderer.prototype.arc3dPath = Highcharts.SVGRenderer.prototype.arc3dPath;
  1318. Highcharts.wrap(Highcharts.Axis.prototype, 'render', function (proceed) {
  1319. proceed.apply(this, [].slice.call(arguments, 1));
  1320. // VML doesn't support a negative z-index
  1321. if (this.sideFrame) {
  1322. this.sideFrame.css({zIndex: 0});
  1323. this.sideFrame.front.attr({fill: this.sideFrame.color});
  1324. }
  1325. if (this.bottomFrame) {
  1326. this.bottomFrame.css({zIndex: 1});
  1327. this.bottomFrame.front.attr({fill: this.bottomFrame.color});
  1328. }
  1329. if (this.backFrame) {
  1330. this.backFrame.css({zIndex: 0});
  1331. this.backFrame.front.attr({fill: this.backFrame.color});
  1332. }
  1333. });
  1334. }
  1335. }(Highcharts));