Source: animate.js

  1. define([
  2. 'underscore',
  3. 'trajectory'
  4. ],
  5. function(_, trajectory) {
  6. var getSampleNamesAndDataForSortedTrajectories =
  7. trajectory.getSampleNamesAndDataForSortedTrajectories;
  8. var getMinimumDelta = trajectory.getMinimumDelta;
  9. var TrajectoryOfSamples = trajectory.TrajectoryOfSamples;
  10. /**
  11. *
  12. * @class AnimationDirector
  13. *
  14. * This object represents an animation director, as the name implies,
  15. * is an object that manages an animation. Takes the for a plot (mapping file
  16. * and coordinates) as well as the metadata categories we want to animate
  17. * over. This object gets called in the main emperor module when an
  18. * animation starts and an instance will only be alive for one animation
  19. * cycle i. e. until the cycle hits the final frame of the animation.
  20. *
  21. * @param {String[]} mappingFileHeaders an Array of strings containing
  22. * metadata mapping file headers.
  23. * @param {Object[]} mappingFileData an Array where the indices are sample
  24. * identifiers and each of the contained elements is an Array of strings where
  25. * the first element corresponds to the first data for the first column in the
  26. * mapping file (mappingFileHeaders).
  27. * @param {Object[]} coordinatesData an Array of Objects where the indices are
  28. * the sample identifiers and each of the objects has the following
  29. * properties: x, y, z, name, color, P1, P2, P3, ... PN where N is the number
  30. * of dimensions in this dataset.
  31. * @param {String} gradientCategory a string with the name of the mapping file
  32. * header where the data that spreads the samples over a gradient is
  33. * contained, usually time or days_since_epoch. Note that this should be an
  34. * all numeric category.
  35. * @param {String} trajectoryCategory a string with the name of the mapping
  36. * file header where the data that groups the samples is contained, this will
  37. * usually be BODY_SITE, HOST_SUBJECT_ID, etc..
  38. * @param {speed} Positive real number determining the speed of an animation,
  39. * this is reflected in the number of frames produced for each time interval.
  40. *
  41. * @return {AnimationDirector} returns an animation director if the parameters
  42. * passed in were all valid.
  43. *
  44. * @throws {Error} Note that this class will raise an Error in any of the
  45. * following cases:
  46. * - One of the input arguments is undefined.
  47. * - If gradientCategory is not in the mappingFileHeaders.
  48. * - If trajectoryCategory is not in the mappingFileHeaders.
  49. * @constructs AnimationDirector
  50. */
  51. function AnimationDirector(mappingFileHeaders, mappingFileData,
  52. coordinatesData, gradientCategory,
  53. trajectoryCategory, speed) {
  54. // all arguments are required
  55. if (mappingFileHeaders === undefined || mappingFileData === undefined ||
  56. coordinatesData === undefined || gradientCategory === undefined ||
  57. trajectoryCategory === undefined || speed === undefined) {
  58. throw new Error('All arguments are required');
  59. }
  60. var index;
  61. index = mappingFileHeaders.indexOf(gradientCategory);
  62. if (index == -1) {
  63. throw new Error('Could not find the gradient category in the mapping' +
  64. ' file');
  65. }
  66. index = mappingFileHeaders.indexOf(trajectoryCategory);
  67. if (index == -1) {
  68. throw new Error('Could not find the trajectory category in the mapping' +
  69. ' file');
  70. }
  71. // guard against logical problems with the trajectory object
  72. if (speed <= 0) {
  73. throw new Error('The animation speed cannot be less than or equal to ' +
  74. 'zero');
  75. }
  76. /**
  77. * @type {String[]}
  78. mappingFileHeaders an Array of strings containing metadata mapping file
  79. headers.
  80. */
  81. this.mappingFileHeaders = mappingFileHeaders;
  82. /**
  83. * @type {Object[]}
  84. *an Array where the indices are sample identifiers
  85. * and each of the contained elements is an Array of strings where the first
  86. * element corresponds to the first data for the first column in the mapping
  87. * file (mappingFileHeaders).
  88. */
  89. this.mappingFileData = mappingFileData;
  90. /**
  91. * @type {Object[]}
  92. * an Array of Objects where the indices are the
  93. * sample identifiers and each of the objects has the following properties:
  94. * x, y, z, name, color, P1, P2, P3, ... PN where N is the number of
  95. * dimensions in this dataset.
  96. */
  97. this.coordinatesData = coordinatesData;
  98. /**
  99. * @type {String}
  100. *a string with the name of the mapping file
  101. * header where the data that spreads the samples over a gradient is
  102. * contained, usually time or days_since_epoch. Note that this should be an
  103. * all numeric category
  104. */
  105. this.gradientCategory = gradientCategory;
  106. /**
  107. * @type {String}
  108. * a string with the name of the mapping file
  109. * header where the data that groups the samples is contained, this will
  110. * usually be BODY_SITE, HOST_SUBJECT_ID, etc..
  111. */
  112. this.trajectoryCategory = trajectoryCategory;
  113. /**
  114. * @type {Float}
  115. * A floating point value determining what the minimum separation between
  116. * samples along the gradients is. Will be null until it is initialized to
  117. * the values according to the input data.
  118. * @default null
  119. */
  120. this.minimumDelta = null;
  121. /**
  122. * @type {Integer}
  123. * Maximum length the groups of samples have along a gradient.
  124. * @default null
  125. */
  126. this.maximumTrajectoryLength = null;
  127. /*
  128. * @type {Integer}
  129. * The current frame being served by the director
  130. * @default -1
  131. */
  132. this.currentFrame = -1;
  133. /**
  134. * @type {Integer}
  135. * The previous frame served by the director
  136. */
  137. this.previousFrame = -1;
  138. /**
  139. * @type {Array}
  140. * Array where each element in the trajectory is a trajectory with the
  141. * interpolated points in it.
  142. */
  143. this.trajectories = [];
  144. /**
  145. * @type {Float}
  146. * How fast should the animation run, has to be a postive non-zero value.
  147. */
  148. this.speed = speed;
  149. this.initializeTrajectories();
  150. this.getMaximumTrajectoryLength();
  151. return this;
  152. }
  153. /**
  154. *
  155. * Initializes the trajectories that the director manages.
  156. *
  157. */
  158. AnimationDirector.prototype.initializeTrajectories = function() {
  159. var chewedData = null, trajectoryBuffer = null, minimumDelta;
  160. var sampleNamesBuffer = [], gradientPointsBuffer = [];
  161. var coordinatesBuffer = [];
  162. var chewedDataBuffer = null;
  163. // frames we want projected in the trajectory's interval
  164. var n = Math.floor((1 / (this.speed)) * 10);
  165. // compute a dictionary from where we will extract the germane data
  166. chewedData = getSampleNamesAndDataForSortedTrajectories(
  167. this.mappingFileHeaders, this.mappingFileData, this.coordinatesData,
  168. this.trajectoryCategory, this.gradientCategory);
  169. if (chewedData === null) {
  170. throw new Error('Error initializing the trajectories, could not ' +
  171. 'compute the data');
  172. }
  173. // calculate the minimum delta per step
  174. this.minimumDelta = getMinimumDelta(chewedData);
  175. // we have to iterate over the keys because chewedData is a dictionary-like
  176. // object, if possible this should be changed in the future to be an Array
  177. for (var key in chewedData) {
  178. // re-initalize the arrays, essentially dropping all the previously
  179. // existing information
  180. sampleNamesBuffer = [];
  181. gradientPointsBuffer = [];
  182. coordinatesBuffer = [];
  183. // buffer this to avoid the multiple look-ups below
  184. chewedDataBuffer = chewedData[key];
  185. // each of the keys is a trajectory name i. e. CONTROL, TREATMENT, etc
  186. // we are going to generate buffers so we can initialize the trajectory
  187. for (var index = 0; index < chewedDataBuffer.length; index++) {
  188. // list of sample identifiers
  189. sampleNamesBuffer.push(chewedDataBuffer[index]['name']);
  190. // list of the value each sample has in the gradient
  191. gradientPointsBuffer.push(chewedDataBuffer[index]['value']);
  192. // x, y and z values for the coordinates data
  193. coordinatesBuffer.push({'x': chewedDataBuffer[index]['x'],
  194. 'y': chewedDataBuffer[index]['y'],
  195. 'z': chewedDataBuffer[index]['z']});
  196. }
  197. // Don't add a trajectory unless it has more than one sample in the
  198. // gradient. For example, there's no reason why we should animate a
  199. // trajectory that has 3 samples at timepoint 0 ([0, 0, 0]) or a
  200. // trajectory that has just one sample at timepoint 0 ([0])
  201. if (sampleNamesBuffer.length <= 1 ||
  202. _.uniq(gradientPointsBuffer).length <= 1) {
  203. continue;
  204. }
  205. // create the trajectory object, we use Infinity to draw as many frames
  206. // as they may be needed
  207. trajectoryBuffer = new TrajectoryOfSamples(sampleNamesBuffer, key,
  208. gradientPointsBuffer, coordinatesBuffer, this.minimumDelta, n,
  209. Infinity);
  210. this.trajectories.push(trajectoryBuffer);
  211. }
  212. return;
  213. };
  214. /**
  215. *
  216. * Retrieves the lengths of all the trajectories and figures out which of
  217. * them is the longest one, then assigns that value to the
  218. * maximumTrajectoryLength property.
  219. * @return {Integer} Maximum trajectory length
  220. *
  221. */
  222. AnimationDirector.prototype.getMaximumTrajectoryLength = function() {
  223. if (this.maximumTrajectoryLength === null) {
  224. this._computeN();
  225. }
  226. return this.maximumTrajectoryLength;
  227. };
  228. /**
  229. *
  230. * Helper function to compute the maximum length of the trajectories that the
  231. * director is in charge of.
  232. * @private
  233. *
  234. */
  235. AnimationDirector.prototype._computeN = function() {
  236. var arrayOfLengths = [];
  237. // retrieve the length of all the trajectories
  238. for (var index = 0; index < this.trajectories.length; index++) {
  239. arrayOfLengths.push(
  240. this.trajectories[index].interpolatedCoordinates.length);
  241. }
  242. // assign the value of the maximum value for these lengths
  243. this.maximumTrajectoryLength = _.max(arrayOfLengths);
  244. };
  245. /**
  246. *
  247. * Helper method to update the value of the currentFrame property.
  248. *
  249. */
  250. AnimationDirector.prototype.updateFrame = function() {
  251. if (this.animationCycleFinished() === false) {
  252. this.previousFrame = this.currentFrame;
  253. this.currentFrame = this.currentFrame + 1;
  254. }
  255. };
  256. /**
  257. *
  258. * Check whether or not the animation cycle has finished for this object.
  259. * @return {bool} True if the animation has reached it's end and False if the
  260. * animation still has frames to go.
  261. *
  262. */
  263. AnimationDirector.prototype.animationCycleFinished = function() {
  264. return this.currentFrame > this.getMaximumTrajectoryLength();
  265. };
  266. return AnimationDirector;
  267. });