multimodalart's picture
Squashing commit
4450790 verified
import { app } from "scripts/app.js";
import type {
IWidget,
LGraphNode,
LGraphCanvas as TLGraphCanvas,
Vector2,
AdjustedMouseEvent,
Vector4,
} from "../typings/litegraph.js";
import { drawNodeWidget, drawRoundedRectangle, fitString, isLowQuality } from "./utils_canvas.js";
/**
* Draws a label on teft, and a value on the right, ellipsizing when out of space.
*/
export function drawLabelAndValue(
ctx: CanvasRenderingContext2D,
label: string,
value: string,
width: number,
posY: number,
height: number,
options?: { offsetLeft: number },
) {
const outerMargin = 15;
const innerMargin = 10;
const midY = posY + height / 2;
ctx.save();
ctx.textAlign = "left";
ctx.textBaseline = "middle";
ctx.fillStyle = LiteGraph.WIDGET_SECONDARY_TEXT_COLOR;
const labelX = outerMargin + innerMargin + (options?.offsetLeft ?? 0);
ctx.fillText(label, labelX, midY);
const valueXLeft = labelX + ctx.measureText(label).width + 7;
const valueXRight = width - (outerMargin + innerMargin);
ctx.fillStyle = LiteGraph.WIDGET_TEXT_COLOR;
ctx.textAlign = "right";
ctx.fillText(fitString(ctx, value, valueXRight - valueXLeft), valueXRight, midY);
ctx.restore();
}
export type RgthreeBaseWidgetBounds = {
/** The bounds, either [x, width] assuming the full height, or [x, y, width, height] if height. */
bounds: Vector2 | Vector4;
onDown?(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode): boolean | void;
onDown?(
event: AdjustedMouseEvent,
pos: Vector2,
node: LGraphNode,
bounds: RgthreeBaseWidgetBounds,
): boolean | void;
onUp?(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode): boolean | void;
onUp?(
event: AdjustedMouseEvent,
pos: Vector2,
node: LGraphNode,
bounds: RgthreeBaseWidgetBounds,
): boolean | void;
onMove?(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode): boolean | void;
onMove?(
event: AdjustedMouseEvent,
pos: Vector2,
node: LGraphNode,
bounds: RgthreeBaseWidgetBounds,
): boolean | void;
data?: any;
};
export type RgthreeBaseHitAreas<Keys extends string> = {
[K in Keys]: RgthreeBaseWidgetBounds;
};
/**
* A base widget that handles mouse events more properly.
*/
export abstract class RgthreeBaseWidget<T> implements IWidget<T, any> {
// We don't want our value to be an array as a widget will be serialized as an "input" for the API
// which uses an array value to represent a link. To keep things simpler, we'll avoid using an
// array at all.
abstract value: T extends Array<any> ? never : T;
name: string;
last_y: number = 0;
protected mouseDowned: Vector2 | null = null;
protected isMouseDownedAndOver: boolean = false;
// protected hitAreas: {[key: string]: RgthreeBaseWidgetBounds} = {};
protected readonly hitAreas: RgthreeBaseHitAreas<any> = {};
private downedHitAreasForMove: RgthreeBaseWidgetBounds[] = [];
constructor(name: string) {
this.name = name;
}
private clickWasWithinBounds(pos: Vector2, bounds: Vector2 | Vector4) {
let xStart = bounds[0];
let xEnd = xStart + (bounds.length > 2 ? bounds[2]! : bounds[1]!);
const clickedX = pos[0] >= xStart && pos[0] <= xEnd;
if (bounds.length === 2) {
return clickedX;
}
return clickedX && pos[1] >= bounds[1] && pos[1] <= bounds[1] + bounds[3];
}
mouse(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode) {
const canvas = app.canvas as TLGraphCanvas;
if (event.type == "pointerdown") {
this.mouseDowned = [...pos];
this.isMouseDownedAndOver = true;
this.downedHitAreasForMove.length = 0;
// Loop over out bounds data and call any specifics.
let anyHandled = false;
for (const part of Object.values(this.hitAreas)) {
if ((part.onDown || part.onMove) && this.clickWasWithinBounds(pos, part.bounds)) {
if (part.onMove) {
this.downedHitAreasForMove.push(part);
}
if (part.onDown) {
const thisHandled = part.onDown.apply(this, [event, pos, node, part]);
anyHandled = anyHandled || thisHandled == true;
}
}
}
return this.onMouseDown(event, pos, node) ?? anyHandled;
}
// This only fires when LiteGraph has a node_widget (meaning it's pressed), but we may not be
// the original widget pressed, so we still need `mouseDowned`.
if (event.type == "pointerup") {
if (!this.mouseDowned) return true;
this.downedHitAreasForMove.length = 0;
this.cancelMouseDown();
let anyHandled = false;
for (const part of Object.values(this.hitAreas)) {
if (part.onUp && this.clickWasWithinBounds(pos, part.bounds)) {
const thisHandled = part.onUp.apply(this, [event, pos, node, part]);
anyHandled = anyHandled || thisHandled == true;
}
}
return this.onMouseUp(event, pos, node) ?? anyHandled;
}
// This only fires when LiteGraph has a node_widget (meaning it's pressed).
if (event.type == "pointermove") {
this.isMouseDownedAndOver = !!this.mouseDowned;
// If we've moved off the button while pressing, then consider us no longer pressing.
if (
this.mouseDowned &&
(pos[0] < 15 ||
pos[0] > node.size[0] - 15 ||
pos[1] < this.last_y ||
pos[1] > this.last_y + LiteGraph.NODE_WIDGET_HEIGHT)
) {
this.isMouseDownedAndOver = false;
}
for (const part of this.downedHitAreasForMove) {
part.onMove!.apply(this, [event, pos, node, part]);
}
return this.onMouseMove(event, pos, node) ?? true;
}
return false;
}
/** Sometimes we want to cancel a mouse down, so that an up/move aren't fired. */
cancelMouseDown() {
this.mouseDowned = null;
this.isMouseDownedAndOver = false;
this.downedHitAreasForMove.length = 0;
}
/** An event that fires when the pointer is pressed down (once). */
onMouseDown(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode): boolean | void {
return;
}
/**
* An event that fires when the pointer is let go. Only fires if this was the widget that was
* originally pressed down.
*/
onMouseUp(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode): boolean | void {
return;
}
/**
* An event that fires when the pointer is moving after pressing down. Will fire both on and off
* of the widget. Check `isMouseDownedAndOver` to determine if the mouse is currently over the
* widget or not.
*/
onMouseMove(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode): boolean | void {
return;
}
}
/**
* A better implementation of the LiteGraph button widget.
*/
export class RgthreeBetterButtonWidget extends RgthreeBaseWidget<string> {
value: string = "";
mouseUpCallback: (event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode) => boolean | void;
constructor(
name: string,
mouseUpCallback: (event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode) => boolean | void,
) {
super(name);
this.mouseUpCallback = mouseUpCallback;
}
draw(ctx: CanvasRenderingContext2D, node: LGraphNode, width: number, y: number, height: number) {
drawWidgetButton({ctx, node, width, height, y}, this.name, this.isMouseDownedAndOver);
}
override onMouseUp(event: AdjustedMouseEvent, pos: Vector2, node: LGraphNode) {
return this.mouseUpCallback(event, pos, node);
}
}
/**
* A better implementation of the LiteGraph text widget, including auto ellipsis.
*/
export class RgthreeBetterTextWidget implements IWidget<string> {
name: string;
value: string;
constructor(name: string, value: string) {
this.name = name;
this.value = value;
}
draw(ctx: CanvasRenderingContext2D, node: LGraphNode, width: number, y: number, height: number) {
const widgetData = drawNodeWidget(ctx, { width, height, posY: y });
if (!widgetData.lowQuality) {
drawLabelAndValue(ctx, this.name, this.value, width, y, height);
}
}
mouse(event: MouseEvent, pos: Vector2, node: LGraphNode) {
const canvas = app.canvas as TLGraphCanvas;
if (event.type == "pointerdown") {
canvas.prompt("Label", this.value, (v: string) => (this.value = v), event);
return true;
}
return false;
}
}
/**
* Options for the Divider Widget.
*/
type RgthreeDividerWidgetOptions = {
marginTop: number;
marginBottom: number;
marginLeft: number;
marginRight: number;
color: string;
thickness: number;
};
/**
* A divider widget; can also be used as a spacer if fed a 0 thickness.
*/
export class RgthreeDividerWidget implements IWidget<null> {
options = { serialize: false };
value = null;
name = "divider";
private readonly widgetOptions: RgthreeDividerWidgetOptions = {
marginTop: 7,
marginBottom: 7,
marginLeft: 15,
marginRight: 15,
color: LiteGraph.WIDGET_OUTLINE_COLOR,
thickness: 1,
};
constructor(widgetOptions?: Partial<RgthreeDividerWidgetOptions>) {
Object.assign(this.widgetOptions, widgetOptions || {});
}
draw(ctx: CanvasRenderingContext2D, node: LGraphNode, width: number, posY: number, h: number) {
if (this.widgetOptions.thickness) {
ctx.strokeStyle = this.widgetOptions.color;
const x = this.widgetOptions.marginLeft;
const y = posY + this.widgetOptions.marginTop;
const w = width - this.widgetOptions.marginLeft - this.widgetOptions.marginRight;
ctx.stroke(new Path2D(`M ${x} ${y} h ${w}`));
}
}
computeSize(width: number): [number, number] {
return [
width,
this.widgetOptions.marginTop + this.widgetOptions.marginBottom + this.widgetOptions.thickness,
];
}
}
/**
* Options for the Label Widget.
*/
export type RgthreeLabelWidgetOptions = {
align?: "left" | "center" | "right";
color?: string;
italic?: boolean;
size?: number;
/** A label to put on the right side. */
actionLabel?: "__PLUS_ICON__" | string;
actionCallback?: (event: PointerEvent) => void;
};
/**
* A simple label widget, drawn with no background.
*/
export class RgthreeLabelWidget implements IWidget<null> {
options = { serialize: false };
value = null;
name: string;
private readonly widgetOptions: RgthreeLabelWidgetOptions = {};
private posY: number = 0;
constructor(name: string, widgetOptions?: RgthreeLabelWidgetOptions) {
this.name = name;
Object.assign(this.widgetOptions, widgetOptions);
}
draw(
ctx: CanvasRenderingContext2D,
node: LGraphNode,
width: number,
posY: number,
height: number,
) {
this.posY = posY;
ctx.save();
ctx.textAlign = this.widgetOptions.align || "left";
ctx.fillStyle = this.widgetOptions.color || LiteGraph.WIDGET_TEXT_COLOR;
const oldFont = ctx.font;
if (this.widgetOptions.italic) {
ctx.font = "italic " + ctx.font;
}
if (this.widgetOptions.size) {
ctx.font = ctx.font.replace(/\d+px/, `${this.widgetOptions.size}px`);
}
const midY = posY + height / 2;
ctx.textBaseline = "middle";
if (this.widgetOptions.align === "center") {
ctx.fillText(this.name, node.size[0] / 2, midY);
} else {
ctx.fillText(this.name, 15, midY);
} // TODO(right);
ctx.font = oldFont;
if (this.widgetOptions.actionLabel === "__PLUS_ICON__") {
const plus = new Path2D(
`M${node.size[0] - 15 - 2} ${posY + 7} v4 h-4 v4 h-4 v-4 h-4 v-4 h4 v-4 h4 v4 h4 z`,
);
ctx.lineJoin = "round";
ctx.lineCap = "round";
ctx.fillStyle = "#3a3";
ctx.strokeStyle = "#383";
ctx.fill(plus);
ctx.stroke(plus);
}
ctx.restore();
}
mouse(event: PointerEvent, nodePos: Vector2, node: LGraphNode) {
if (
event.type !== "pointerdown" ||
isLowQuality() ||
!this.widgetOptions.actionLabel ||
!this.widgetOptions.actionCallback
) {
return false;
}
const pos: Vector2 = [nodePos[0], nodePos[1] - this.posY];
const rightX = node.size[0] - 15;
if (pos[0] > rightX || pos[0] < rightX - 16) {
return false;
}
this.widgetOptions.actionCallback(event);
return true;
}
}
/** An invisible widget. */
export class RgthreeInvisibleWidget<T> implements IWidget<T> {
name: string;
type: string;
value: T;
serializeValue: IWidget['serializeValue'] = undefined;
constructor(name: string, type: string, value: T, serializeValueFn: ()=> T) {
this.name = name;
this.type = type;
this.value = value;
if (serializeValueFn) {
this.serializeValue = serializeValueFn
}
}
draw() { return; }
computeSize(width: number) : Vector2 { return [0, 0]; }
}
type DrawContext = {
ctx: CanvasRenderingContext2D,
node: LGraphNode,
width: number,
y: number,
height: number,
}
/**
* Draws a better button.
*/
export function drawWidgetButton(drawCtx: DrawContext, text: string, isMouseDownedAndOver: boolean = false) {
// First, add a shadow if we're not down or lowquality.
if (!isLowQuality() && !isMouseDownedAndOver) {
drawRoundedRectangle(drawCtx.ctx, {
width: drawCtx.width - 30 - 2,
height: drawCtx.height,
posY: drawCtx.y + 1,
posX: 15 + 1,
borderRadius: 4,
colorBackground: "#000000aa",
colorStroke: "#000000aa",
});
}
drawRoundedRectangle(drawCtx.ctx, {
width: drawCtx.width - 30,
height: drawCtx.height,
posY: drawCtx.y + (isMouseDownedAndOver ? 1 : 0),
posX: 15,
borderRadius: isLowQuality() ? 0 : 4,
colorBackground: isMouseDownedAndOver ? "#444" : LiteGraph.WIDGET_BGCOLOR,
});
if (!isLowQuality()) {
drawCtx.ctx.textBaseline = "middle";
drawCtx.ctx.textAlign = "center";
drawCtx.ctx.fillStyle = LiteGraph.WIDGET_TEXT_COLOR;
drawCtx.ctx.fillText(
text,
drawCtx.node.size[0] / 2,
drawCtx.y + drawCtx.height / 2 + (isMouseDownedAndOver ? 1 : 0),
);
}
}