VexFlow - Music Engraving for HTML5 Copyright Mohit Muthanna 2010

This class implements dot modifiers for notes.

import { Vex } from './vex';
import { Modifier } from './modifier';

export class Dot extends Modifier {
  static get CATEGORY() { return 'dots'; }

Arrange dots inside a ModifierContext.

  static format(dots, state) {
    const right_shift = state.right_shift;
    const dot_spacing = 1;

    if (!dots || dots.length === 0) return false;

    let i, dot, note, shift;
    const dot_list = [];
    for (i = 0; i < dots.length; ++i) {
      dot = dots[i];
      note = dot.getNote();

      let props;

Only StaveNote has .getKeyProps()

      if (typeof note.getKeyProps === 'function') {
        props = note.getKeyProps()[dot.getIndex()];
        shift = (props.displaced ? note.getExtraRightPx() : 0);
      } else { // Else it's a TabNote
        props = { line: 0.5 }; // Shim key props for dot placement
        shift = 0;
      }

      dot_list.push({ line: props.line, shift, note, dot });
    }

Sort dots by line number.

    dot_list.sort((a, b) => b.line - a.line);

    let dot_shift = right_shift;
    let x_width = 0;
    let last_line = null;
    let last_note = null;
    let prev_dotted_space = null;
    let half_shiftY = 0;

    for (i = 0; i < dot_list.length; ++i) {
      dot = dot_list[i].dot;
      note = dot_list[i].note;
      shift = dot_list[i].shift;
      const line = dot_list[i].line;

Reset the position of the dot every line.

      if (line != last_line || note != last_note) {
        dot_shift = shift;
      }

      if (!note.isRest() && line != last_line) {
        if (Math.abs(line % 1) == 0.5) {

note is on a space, so no dot shift

          half_shiftY = 0;
        } else if (!note.isRest()) {

note is on a line, so shift dot to space above the line

          half_shiftY = 0.5;
          if (last_note != null &&
              !last_note.isRest() && last_line - line == 0.5) {

previous note on a space, so shift dot to space below the line

            half_shiftY = -0.5;
          } else if (line + half_shiftY == prev_dotted_space) {

previous space is dotted, so shift dot to space below the line

            half_shiftY = -0.5;
          }
        }
      }

convert half_shiftY to a multiplier for dots.draw()

      dot.dot_shiftY = (-half_shiftY);
      prev_dotted_space = line + half_shiftY;

      dot.setXShift(dot_shift);
      dot_shift += dot.getWidth() + dot_spacing; // spacing
      x_width = (dot_shift > x_width) ? dot_shift : x_width;
      last_line = line;
      last_note = note;
    }

Update state.

    state.right_shift += x_width;
  }

  /**
   * @constructor
   */
  constructor() {
    super();

    this.note = null;
    this.index = null;
    this.position = Modifier.Position.RIGHT;

    this.radius = 2;
    this.setWidth(5);
    this.dot_shiftY = 0;
  }
  getCategory() { return Dot.CATEGORY; }
  setNote(note) {
    this.note = note;

    if (this.note.getCategory() === 'gracenotes') {
      this.radius *= 0.50;
      this.setWidth(3);
    }
  }
  setDotShiftY(y) { this.dot_shiftY = y; return this; }
  draw() {
    if (!this.context) throw new Vex.RERR('NoContext',
      "Can't draw dot without a context.");
    if (!(this.note && (this.index != null))) throw new Vex.RERR('NoAttachedNote',
      "Can't draw dot without a note and index.");

    const line_space = this.note.stave.options.spacing_between_lines_px;

    const start = this.note.getModifierStartXY(this.position, this.index);

Set the starting y coordinate to the base of the stem for TabNotes

    if (this.note.getCategory() === 'tabnotes') {
      start.y = this.note.getStemExtents().baseY;
    }

    const dot_x = (start.x + this.x_shift) + this.width - this.radius;
    const dot_y = start.y + this.y_shift + (this.dot_shiftY * line_space);
    const ctx = this.context;

    ctx.beginPath();
    ctx.arc(dot_x, dot_y, this.radius, 0, Math.PI * 2, false);
    ctx.fill();
  }
}
h