VexFlow - Copyright (c) Mohit Muthanna 2010. Author: Cyril Silverman
This file implement TextBrackets which extend between two notes.
The octave transposition markings (8va, 8vb, 15va, 15vb) can be created
using this class.
import { Vex } from './vex';
import { Flow } from './tables';
import { Renderer } from './renderer';To enable logging for this class. Set Vex.Flow.TextBracket.DEBUG to true.
function L() { if (TextBracket.DEBUG) Vex.L('Vex.Flow.TextBracket', arguments); }
export class TextBracket {
static get Positions() {
return {
TOP: 1,
BOTTOM: -1,
};
}
constructor(bracket_data) {
this.start = bracket_data.start;
this.stop = bracket_data.stop;
this.text = bracket_data.text || '';
this.superscript = bracket_data.superscript || '';
this.position = bracket_data.position || TextBracket.Positions.TOP;
this.line = 1;
this.font = {
family: 'Serif',
size: 15,
weight: 'italic',
};
this.render_options = {
dashed: true,
dash: [5],
color: 'black',
line_width: 1,
show_bracket: true,
bracket_height: 8,In the BOTTOM position, the bracket line can extend under the superscript.
underline_superscript: true,
};
}Apply the text backet styling to the provided context
applyStyle(context) {Apply style for the octave bracket
context.setFont(this.font.family, this.font.size, this.font.weight);
context.setStrokeStyle(this.render_options.color);
context.setFillStyle(this.render_options.color);
context.setLineWidth(this.render_options.line_width);
return this;
}Set whether the bracket line should be dashed. You can also
optionally set the dash pattern by passing in an array of numbers
setDashed(dashed, dash) {
this.render_options.dashed = dashed;
if (dash) this.render_options.dash = dash;
return this;
}Set the font for the text
setFont(font) { this.font = font; return this; }Set the rendering context for the octave bracket
setContext(context) { this.context = context; return this; }Set the staff line to render the bracket on
setLine(line) { this.line = line; return this; }Draw the octave bracket on the rendering context
draw() {
const ctx = this.context;
let y = 0;
switch (this.position) {
case TextBracket.Positions.TOP:
y = this.start.getStave().getYForTopText(this.line);
break;
case TextBracket.Positions.BOTTOM:
y = this.start.getStave().getYForBottomText(this.line);
break;
}Get the preliminary start and stop coordintates for the bracket
const start = { x: this.start.getAbsoluteX(), y };
const stop = { x: this.stop.getAbsoluteX(), y };
L('Rendering TextBracket: start:', start, 'stop:', stop, 'y:', y);
const bracket_height = this.render_options.bracket_height * this.position;
ctx.save();
this.applyStyle(ctx);Draw text
ctx.fillText(this.text, start.x, start.y);Get the width and height for the octave number
const main_width = ctx.measureText(this.text).width;
const main_height = ctx.measureText('M').width;Calculate the y position for the super script
const super_y = start.y - (main_height / 2.5);Draw the superscript
ctx.setFont(this.font.family, this.font.size / 1.4, this.font.weight);
ctx.fillText(this.superscript, start.x + main_width + 1, super_y);Determine width and height of the superscript
const superscript_width = ctx.measureText(this.superscript).width;
const super_height = ctx.measureText('M').width;Setup initial coordinates for the bracket line
let start_x = start.x;
let line_y = super_y;
const end_x = stop.x + this.stop.getGlyph().head_width;Adjust x and y coordinates based on position
if (this.position === TextBracket.Positions.TOP) {
start_x += main_width + superscript_width + 5;
line_y -= super_height / 2.7;
} else if (this.position === TextBracket.Positions.BOTTOM) {
line_y += super_height / 2.7;
start_x += main_width + 2;
if (!this.render_options.underline_superscript) {
start_x += superscript_width;
}
}
if (this.render_options.dashed) {Main line
Renderer.drawDashedLine(ctx, start_x, line_y, end_x, line_y,
this.render_options.dash);Ending Bracket
if (this.render_options.show_bracket) {
Renderer.drawDashedLine(ctx, end_x, line_y + (1 * this.position),
end_x, line_y + bracket_height, this.render_options.dash);
}
} else {
ctx.beginPath();
ctx.moveTo(start_x, line_y);Main line
ctx.lineTo(end_x, line_y);
if (this.render_options.show_bracket) {Ending bracket
ctx.lineTo(end_x, line_y + bracket_height);
}
ctx.stroke();
ctx.closePath();
}
ctx.restore();
}
}