Source: components/data_layer/annotation_track.js

  1. import BaseDataLayer from './base';
  2. import {merge} from '../../helpers/layouts';
  3. /**
  4. * @memberof module:LocusZoom_DataLayers~annotation_track
  5. */
  6. const default_layout = {
  7. color: '#000000',
  8. filters: null,
  9. tooltip_positioning: 'vertical',
  10. hitarea_width: 8,
  11. };
  12. /**
  13. * Create a single continuous 2D track that provides information about each datapoint
  14. *
  15. * For example, this can be used to mark items by membership in a group, alongside information in other panels
  16. * @alias module:LocusZoom_DataLayers~annotation_track
  17. * @see {@link module:LocusZoom_DataLayers~BaseDataLayer} for additional layout options
  18. */
  19. class AnnotationTrack extends BaseDataLayer {
  20. /**
  21. * @param {String|module:LocusZoom_DataLayers~ScalableParameter[]} [layout.color] Specify how to choose the fill color for each tick mark
  22. * @param {number} [layout.hitarea_width=8] The width (in pixels) of hitareas. Annotation marks are typically 1 px wide,
  23. * so a hit area of 4px on each side can make it much easier to select an item for a tooltip. Hitareas will not interfere
  24. * with selecting adjacent points.
  25. * @param {'horizontal'|'vertical'|'top'|'bottom'|'left'|'right'} [layout.tooltip_positioning='vertical'] Where to draw the tooltip relative to the datum.
  26. */
  27. constructor(layout) {
  28. if (!Array.isArray(layout.filters)) {
  29. throw new Error('Annotation track must specify array of filters for selecting points to annotate');
  30. }
  31. merge(layout, default_layout);
  32. super(...arguments);
  33. }
  34. initialize() {
  35. super.initialize();
  36. this._hitareas_group = this.svg.group.append('g')
  37. .attr('class', `lz-data_layer-${this.layout.type}-hit_areas`);
  38. this._visible_lines_group = this.svg.group.append('g')
  39. .attr('class', `lz-data_layer-${this.layout.type}-visible_lines`);
  40. }
  41. render() {
  42. // Apply filters to only render a specified set of points
  43. const track_data = this._applyFilters();
  44. const hit_areas_selection = this._hitareas_group.selectAll(`rect.lz-data_layer-${this.layout.type}`)
  45. .data(track_data, (d) => d[this.layout.id_field]);
  46. const _getX = (d, i) => {
  47. // Helper for hitarea position calcs: ensures that a hitarea never overlaps the space allocated
  48. // for a real data element. Helps to avoid mouse jitter when selecting tooltips in crowded areas.
  49. const x_center = this.parent['x_scale'](d[this.layout.x_axis.field]);
  50. let x_left = x_center - this.layout.hitarea_width / 2;
  51. if (i >= 1) {
  52. // This assumes that the data are in sorted order.
  53. const left_node = track_data[i - 1];
  54. const left_node_x_center = this.parent['x_scale'](left_node[this.layout.x_axis.field]);
  55. x_left = Math.max(x_left, (x_center + left_node_x_center) / 2);
  56. }
  57. return [x_left, x_center];
  58. };
  59. // Draw hitareas under real data elements, so that real data elements always take precedence
  60. hit_areas_selection.enter()
  61. .append('rect')
  62. .attr('class', `lz-data_layer-${this.layout.type}`)
  63. // Update the set of elements to reflect new data
  64. .merge(hit_areas_selection)
  65. .attr('id', (d) => this.getElementId(d))
  66. .attr('height', this.parent.layout.height)
  67. .attr('opacity', 0)
  68. .attr('x', (d, i) => {
  69. const crds = _getX(d, i);
  70. return crds[0];
  71. })
  72. .attr('width', (d, i) => {
  73. const crds = _getX(d, i);
  74. return (crds[1] - crds[0]) + this.layout.hitarea_width / 2;
  75. });
  76. const width = 1;
  77. const selection = this._visible_lines_group.selectAll(`rect.lz-data_layer-${this.layout.type}`)
  78. .data(track_data, (d) => d[this.layout.id_field]);
  79. // Draw rectangles (visual and tooltip positioning)
  80. selection.enter()
  81. .append('rect')
  82. .attr('class', `lz-data_layer-${this.layout.type}`)
  83. .merge(selection)
  84. .attr('id', (d) => this.getElementId(d))
  85. .attr('x', (d) => this.parent['x_scale'](d[this.layout.x_axis.field]) - width / 2)
  86. .attr('width', width)
  87. .attr('height', this.parent.layout.height)
  88. .attr('fill', (d, i) => this.resolveScalableParameter(this.layout.color, d, i));
  89. // Remove unused elements
  90. selection.exit()
  91. .remove();
  92. // Set up tooltips and mouse interaction
  93. this.svg.group
  94. .call(this.applyBehaviors.bind(this));
  95. // Remove unused elements
  96. hit_areas_selection.exit()
  97. .remove();
  98. }
  99. /**
  100. * Render tooltip at the center of each tick mark
  101. * @param tooltip
  102. * @return {{y_min: number, x_max: *, y_max: *, x_min: number}}
  103. * @private
  104. */
  105. _getTooltipPosition(tooltip) {
  106. const panel = this.parent;
  107. const data_layer_height = panel.layout.height - (panel.layout.margin.top + panel.layout.margin.bottom);
  108. const stroke_width = 1; // as defined in the default stylesheet
  109. const x_center = panel.x_scale(tooltip.data[this.layout.x_axis.field]);
  110. const y_center = data_layer_height / 2;
  111. return {
  112. x_min: x_center - stroke_width,
  113. x_max: x_center + stroke_width,
  114. y_min: y_center - panel.layout.margin.top,
  115. y_max: y_center + panel.layout.margin.bottom,
  116. };
  117. }
  118. }
  119. export {AnnotationTrack as default};