mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-21 15:24:09 +00:00
refactor: improve Vue widget type safety and runtime prop handling
- Add proper type guards for all widget input specs (Color, TreeSelect, MultiSelect, FileUpload, Galleria) - Enhance schemas with missing properties (format, placeholder, accept, extensions, tooltip) - Fix widgets to honor runtime props like disabled while accessing spec metadata - Eliminate all 'as any' usage in widget components with proper TypeScript types - Clean separation: widget.spec.options for metadata, widget.options for runtime state - Refactor devtools into modular structure with vue_widgets showcase nodes
This commit is contained in:
@@ -6,11 +6,13 @@ This directory contains development tools and test utilities for ComfyUI, previo
|
||||
|
||||
- `__init__.py` - Server endpoints for development tools (`/api/devtools/*`)
|
||||
- `dev_nodes.py` - Development and testing nodes for ComfyUI
|
||||
- `nodes/vue_widgets.py` - Widget showcase nodes used to exercise new Vue-based widgets
|
||||
- `fake_model.safetensors` - Test fixture for model loading tests
|
||||
|
||||
## Purpose
|
||||
|
||||
These tools provide:
|
||||
|
||||
- Test endpoints for browser automation
|
||||
- Development nodes for testing various UI features
|
||||
- Mock data for consistent testing environments
|
||||
@@ -25,4 +27,4 @@ cp -r tools/devtools/* /path/to/your/ComfyUI/custom_nodes/ComfyUI_devtools/
|
||||
|
||||
## Migration
|
||||
|
||||
This directory was created as part of issue #4683 to merge the ComfyUI_devtools repository into the main frontend repository, eliminating the need for separate versioning and simplifying the development workflow.
|
||||
This directory was created as part of issue #4683 to merge the ComfyUI_devtools repository into the main frontend repository, eliminating the need for separate versioning and simplifying the development workflow.
|
||||
|
||||
@@ -1,13 +1,20 @@
|
||||
from __future__ import annotations
|
||||
|
||||
from .nodes import (
|
||||
VueAudioPreviewComboNode,
|
||||
VueAudioRecordWidgetNode,
|
||||
VueChartWidgetNode,
|
||||
DeprecatedNode,
|
||||
DummyPatch,
|
||||
ErrorRaiseNode,
|
||||
ErrorRaiseNodeWithMessage,
|
||||
ExperimentalNode,
|
||||
VueFileUploadWidgetNode,
|
||||
LoadAnimatedImageTest,
|
||||
LongComboDropdown,
|
||||
VueMarkdownWidgetNode,
|
||||
VueGalleriaWidgetNode,
|
||||
VueImageCompareWidgetNode,
|
||||
MultiSelectNode,
|
||||
NodeWithBooleanInput,
|
||||
NodeWithDefaultInput,
|
||||
@@ -23,24 +30,36 @@ from .nodes import (
|
||||
NodeWithValidation,
|
||||
NodeWithV2ComboInput,
|
||||
ObjectPatchNode,
|
||||
VueSelectButtonWidgetNode,
|
||||
VueTextareaWidgetNode,
|
||||
VueTreeSelectMultiWidgetNode,
|
||||
VueTreeSelectWidgetNode,
|
||||
RemoteWidgetNode,
|
||||
RemoteWidgetNodeWithControlAfterRefresh,
|
||||
RemoteWidgetNodeWithParams,
|
||||
RemoteWidgetNodeWithRefresh,
|
||||
RemoteWidgetNodeWithRefreshButton,
|
||||
SimpleSlider,
|
||||
VueColorWidgetNode,
|
||||
NODE_CLASS_MAPPINGS,
|
||||
NODE_DISPLAY_NAME_MAPPINGS,
|
||||
)
|
||||
|
||||
__all__ = [
|
||||
"VueAudioPreviewComboNode",
|
||||
"VueAudioRecordWidgetNode",
|
||||
"VueChartWidgetNode",
|
||||
"DeprecatedNode",
|
||||
"DummyPatch",
|
||||
"ErrorRaiseNode",
|
||||
"ErrorRaiseNodeWithMessage",
|
||||
"ExperimentalNode",
|
||||
"VueFileUploadWidgetNode",
|
||||
"LoadAnimatedImageTest",
|
||||
"LongComboDropdown",
|
||||
"VueMarkdownWidgetNode",
|
||||
"VueGalleriaWidgetNode",
|
||||
"VueImageCompareWidgetNode",
|
||||
"MultiSelectNode",
|
||||
"NodeWithBooleanInput",
|
||||
"NodeWithDefaultInput",
|
||||
@@ -56,12 +75,17 @@ __all__ = [
|
||||
"NodeWithValidation",
|
||||
"NodeWithV2ComboInput",
|
||||
"ObjectPatchNode",
|
||||
"VueSelectButtonWidgetNode",
|
||||
"VueTextareaWidgetNode",
|
||||
"VueTreeSelectMultiWidgetNode",
|
||||
"VueTreeSelectWidgetNode",
|
||||
"RemoteWidgetNode",
|
||||
"RemoteWidgetNodeWithControlAfterRefresh",
|
||||
"RemoteWidgetNodeWithParams",
|
||||
"RemoteWidgetNodeWithRefresh",
|
||||
"RemoteWidgetNodeWithRefreshButton",
|
||||
"SimpleSlider",
|
||||
"VueColorWidgetNode",
|
||||
"NODE_CLASS_MAPPINGS",
|
||||
"NODE_DISPLAY_NAME_MAPPINGS",
|
||||
]
|
||||
|
||||
@@ -44,12 +44,29 @@ from .remote import (
|
||||
NODE_CLASS_MAPPINGS as remote_class_mappings,
|
||||
NODE_DISPLAY_NAME_MAPPINGS as remote_display_name_mappings,
|
||||
)
|
||||
from .vue_widgets import (
|
||||
VueAudioPreviewComboNode,
|
||||
VueAudioRecordWidgetNode,
|
||||
VueChartWidgetNode,
|
||||
VueColorWidgetNode,
|
||||
VueFileUploadWidgetNode,
|
||||
VueGalleriaWidgetNode,
|
||||
VueImageCompareWidgetNode,
|
||||
VueMarkdownWidgetNode,
|
||||
VueSelectButtonWidgetNode,
|
||||
VueTextareaWidgetNode,
|
||||
VueTreeSelectMultiWidgetNode,
|
||||
VueTreeSelectWidgetNode,
|
||||
NODE_CLASS_MAPPINGS as vue_widgets_class_mappings,
|
||||
NODE_DISPLAY_NAME_MAPPINGS as vue_widgets_display_name_mappings,
|
||||
)
|
||||
|
||||
NODE_CLASS_MAPPINGS = {
|
||||
**errors_class_mappings,
|
||||
**inputs_class_mappings,
|
||||
**remote_class_mappings,
|
||||
**models_class_mappings,
|
||||
**vue_widgets_class_mappings,
|
||||
}
|
||||
|
||||
NODE_DISPLAY_NAME_MAPPINGS = {
|
||||
@@ -57,6 +74,7 @@ NODE_DISPLAY_NAME_MAPPINGS = {
|
||||
**inputs_display_name_mappings,
|
||||
**remote_display_name_mappings,
|
||||
**models_display_name_mappings,
|
||||
**vue_widgets_display_name_mappings,
|
||||
}
|
||||
|
||||
__all__ = [
|
||||
@@ -88,6 +106,18 @@ __all__ = [
|
||||
"RemoteWidgetNodeWithRefresh",
|
||||
"RemoteWidgetNodeWithRefreshButton",
|
||||
"SimpleSlider",
|
||||
"VueAudioPreviewComboNode",
|
||||
"VueAudioRecordWidgetNode",
|
||||
"VueChartWidgetNode",
|
||||
"VueColorWidgetNode",
|
||||
"VueFileUploadWidgetNode",
|
||||
"VueGalleriaWidgetNode",
|
||||
"VueImageCompareWidgetNode",
|
||||
"VueMarkdownWidgetNode",
|
||||
"VueSelectButtonWidgetNode",
|
||||
"VueTextareaWidgetNode",
|
||||
"VueTreeSelectMultiWidgetNode",
|
||||
"VueTreeSelectWidgetNode",
|
||||
"NODE_CLASS_MAPPINGS",
|
||||
"NODE_DISPLAY_NAME_MAPPINGS",
|
||||
]
|
||||
|
||||
477
tools/devtools/nodes/vue_widgets.py
Normal file
477
tools/devtools/nodes/vue_widgets.py
Normal file
@@ -0,0 +1,477 @@
|
||||
from __future__ import annotations
|
||||
|
||||
SAMPLE_IMAGE_DATA_URI = (
|
||||
""
|
||||
)
|
||||
SAMPLE_IMAGE_DATA_URI_ALT = (
|
||||
""
|
||||
)
|
||||
SAMPLE_IMAGE_DATA_URI_THIRD = (
|
||||
""
|
||||
)
|
||||
|
||||
|
||||
class VueFileUploadWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"uploaded_file": (
|
||||
"FILEUPLOAD",
|
||||
{
|
||||
"default": [],
|
||||
"options": {
|
||||
"extensions": ["png", "jpg", "jpeg", "webp"],
|
||||
"accept": "image/png,image/jpeg,image/webp",
|
||||
"tooltip": "Upload an image file",
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_file"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the FILEUPLOAD widget"
|
||||
|
||||
def return_file(self, uploaded_file: str | None):
|
||||
return (uploaded_file or "",)
|
||||
|
||||
|
||||
class VueImageCompareWidgetNode:
|
||||
BEFORE_IMAGE = ""
|
||||
AFTER_IMAGE = ""
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"comparison": (
|
||||
"IMAGECOMPARE",
|
||||
{
|
||||
"default": {
|
||||
"before": BEFORE_IMAGE,
|
||||
"after": AFTER_IMAGE,
|
||||
},
|
||||
"options": {
|
||||
"beforeAlt": "Before",
|
||||
"afterAlt": "After",
|
||||
"initialPosition": 40,
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ()
|
||||
FUNCTION = "noop"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the IMAGECOMPARE widget"
|
||||
|
||||
def noop(self, comparison):
|
||||
return tuple()
|
||||
|
||||
|
||||
class VueTreeSelectWidgetNode:
|
||||
TREE_DATA = [
|
||||
{
|
||||
"key": "root",
|
||||
"label": "Root",
|
||||
"children": [
|
||||
{
|
||||
"key": "section-a",
|
||||
"label": "Section A",
|
||||
"children": [
|
||||
{"key": "item-a1", "label": "Item A1"},
|
||||
{"key": "item-a2", "label": "Item A2"},
|
||||
],
|
||||
},
|
||||
{
|
||||
"key": "section-b",
|
||||
"label": "Section B",
|
||||
"children": [
|
||||
{"key": "item-b1", "label": "Item B1"},
|
||||
{"key": "item-b2", "label": "Item B2"},
|
||||
],
|
||||
},
|
||||
],
|
||||
}
|
||||
]
|
||||
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"selection": (
|
||||
"TREESELECT",
|
||||
{
|
||||
"default": "item-a1",
|
||||
"options": {
|
||||
"values": cls.TREE_DATA,
|
||||
"multiple": False,
|
||||
"placeholder": "Select an item",
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_selection"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the TREESELECT widget"
|
||||
|
||||
def return_selection(self, selection: str):
|
||||
return (selection,)
|
||||
|
||||
|
||||
class VueTreeSelectMultiWidgetNode(VueTreeSelectWidgetNode):
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"selection": (
|
||||
"TREESELECT",
|
||||
{
|
||||
"default": ["item-a1", "item-b1"],
|
||||
"options": {
|
||||
"values": cls.TREE_DATA,
|
||||
"multiple": True,
|
||||
"placeholder": "Select items",
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
OUTPUT_IS_LIST = (True,)
|
||||
FUNCTION = "return_selection"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the TREESELECT widget in multi-select mode"
|
||||
|
||||
def return_selection(self, selection: list[str]):
|
||||
return (selection,)
|
||||
|
||||
|
||||
class VueSelectButtonWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
options = [
|
||||
{"label": "Low", "value": "low"},
|
||||
{"label": "Medium", "value": "medium"},
|
||||
{"label": "High", "value": "high"},
|
||||
]
|
||||
|
||||
return {
|
||||
"required": {
|
||||
"mode": (
|
||||
"SELECTBUTTON",
|
||||
{
|
||||
"default": "Medium",
|
||||
"options": {
|
||||
"values": ["Low", "Medium", "High"],
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_mode"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the SELECTBUTTON widget"
|
||||
|
||||
def return_mode(self, mode: str):
|
||||
return (mode,)
|
||||
|
||||
|
||||
class VueTextareaWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"notes": (
|
||||
"TEXTAREA",
|
||||
{
|
||||
"default": "This is a DevTools textarea widget.\nFeel free to edit me!",
|
||||
"options": {
|
||||
"rows": 4,
|
||||
"cols": 40,
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_notes"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the TEXTAREA widget"
|
||||
|
||||
def return_notes(self, notes: str):
|
||||
return (notes,)
|
||||
|
||||
|
||||
class VueChartWidgetNode:
|
||||
CHART_DATA = {
|
||||
"labels": ["Iteration 1", "Iteration 2", "Iteration 3"],
|
||||
"datasets": [
|
||||
{
|
||||
"label": "Loss",
|
||||
"data": [0.95, 0.62, 0.31],
|
||||
"borderColor": "#339AF0",
|
||||
"backgroundColor": "rgba(51, 154, 240, 0.2)",
|
||||
"fill": True,
|
||||
"tension": 0.35,
|
||||
}
|
||||
],
|
||||
}
|
||||
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"chart": (
|
||||
"CHART",
|
||||
{
|
||||
"options": {
|
||||
"type": "line",
|
||||
"data": cls.CHART_DATA,
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("DICT",)
|
||||
FUNCTION = "return_chart"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the CHART widget"
|
||||
|
||||
def return_chart(self, chart):
|
||||
return (chart,)
|
||||
|
||||
|
||||
class VueGalleriaWidgetNode:
|
||||
GALLERIA_IMAGES = [
|
||||
{
|
||||
"itemImageSrc": SAMPLE_IMAGE_DATA_URI,
|
||||
"thumbnailImageSrc": SAMPLE_IMAGE_DATA_URI,
|
||||
"alt": "Warm gradient",
|
||||
},
|
||||
{
|
||||
"itemImageSrc": SAMPLE_IMAGE_DATA_URI_ALT,
|
||||
"thumbnailImageSrc": SAMPLE_IMAGE_DATA_URI_ALT,
|
||||
"alt": "Cool gradient",
|
||||
},
|
||||
{
|
||||
"itemImageSrc": SAMPLE_IMAGE_DATA_URI_THIRD,
|
||||
"thumbnailImageSrc": SAMPLE_IMAGE_DATA_URI_THIRD,
|
||||
"alt": "Fresh gradient",
|
||||
},
|
||||
]
|
||||
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"gallery": (
|
||||
"GALLERIA",
|
||||
{
|
||||
"default": cls.GALLERIA_IMAGES,
|
||||
"options": {
|
||||
"images": cls.GALLERIA_IMAGES,
|
||||
"showThumbnails": True,
|
||||
"showItemNavigators": True,
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ()
|
||||
FUNCTION = "noop"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the GALLERIA widget"
|
||||
|
||||
def noop(self, gallery):
|
||||
return tuple()
|
||||
|
||||
|
||||
class VueMarkdownWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"markdown": (
|
||||
"MARKDOWN",
|
||||
{
|
||||
"default": "# DevTools Markdown\nThis widget renders **Markdown** content.",
|
||||
"options": {
|
||||
"content": "# DevTools Markdown\nThis widget renders **Markdown** content.",
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ()
|
||||
FUNCTION = "noop"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the MARKDOWN widget"
|
||||
|
||||
def noop(self, markdown):
|
||||
return tuple()
|
||||
|
||||
|
||||
class VueAudioRecordWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"recording": (
|
||||
"AUDIORECORD",
|
||||
{
|
||||
"default": "",
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_recording"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the AUDIORECORD widget"
|
||||
|
||||
def return_recording(self, recording: str):
|
||||
return (recording,)
|
||||
|
||||
|
||||
class VueMultiSelectWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"selection": (
|
||||
"MULTISELECT",
|
||||
{
|
||||
"default": ["option1", "option3"],
|
||||
"options": {
|
||||
"values": ["option1", "option2", "option3", "option4", "option5"],
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
OUTPUT_IS_LIST = (True,)
|
||||
FUNCTION = "return_selection"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the MULTISELECT widget"
|
||||
|
||||
def return_selection(self, selection: list[str]):
|
||||
return (selection,)
|
||||
|
||||
|
||||
class VueColorWidgetNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"color": (
|
||||
"COLOR",
|
||||
{
|
||||
"default": "#ff6b6b",
|
||||
"options": {
|
||||
"tooltip": "Pick a color",
|
||||
},
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_color"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the COLOR widget"
|
||||
|
||||
def return_color(self, color: str):
|
||||
return (color,)
|
||||
|
||||
|
||||
class VueAudioPreviewComboNode:
|
||||
@classmethod
|
||||
def INPUT_TYPES(cls):
|
||||
return {
|
||||
"required": {
|
||||
"audio": (
|
||||
"COMBO",
|
||||
{
|
||||
"options": ["ambient.wav", "dialog.wav"],
|
||||
"default": "ambient.wav",
|
||||
"tooltip": "Pick an audio clip",
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
RETURN_TYPES = ("STRING",)
|
||||
FUNCTION = "return_audio"
|
||||
CATEGORY = "DevTools/Vue Widgets"
|
||||
DESCRIPTION = "Showcases the COMBO widget rendered as Audio UI"
|
||||
|
||||
def return_audio(self, audio: str):
|
||||
return (audio,)
|
||||
|
||||
|
||||
NODE_CLASS_MAPPINGS = {
|
||||
"DevToolsVueFileUploadWidgetNode": VueFileUploadWidgetNode,
|
||||
"DevToolsVueImageCompareWidgetNode": VueImageCompareWidgetNode,
|
||||
"DevToolsVueTreeSelectWidgetNode": VueTreeSelectWidgetNode,
|
||||
"DevToolsVueTreeSelectMultiWidgetNode": VueTreeSelectMultiWidgetNode,
|
||||
"DevToolsVueSelectButtonWidgetNode": VueSelectButtonWidgetNode,
|
||||
"DevToolsVueMultiSelectWidgetNode": VueMultiSelectWidgetNode,
|
||||
"DevToolsVueTextareaWidgetNode": VueTextareaWidgetNode,
|
||||
"DevToolsVueChartWidgetNode": VueChartWidgetNode,
|
||||
"DevToolsVueGalleriaWidgetNode": VueGalleriaWidgetNode,
|
||||
"DevToolsVueMarkdownWidgetNode": VueMarkdownWidgetNode,
|
||||
"DevToolsVueAudioRecordWidgetNode": VueAudioRecordWidgetNode,
|
||||
"DevToolsVueColorWidgetNode": VueColorWidgetNode,
|
||||
"DevToolsVueAudioPreviewComboNode": VueAudioPreviewComboNode,
|
||||
}
|
||||
|
||||
NODE_DISPLAY_NAME_MAPPINGS = {
|
||||
"DevToolsVueFileUploadWidgetNode": "Vue File Upload Widget",
|
||||
"DevToolsVueImageCompareWidgetNode": "Vue Image Compare Widget",
|
||||
"DevToolsVueTreeSelectWidgetNode": "Vue Tree Select Widget",
|
||||
"DevToolsVueTreeSelectMultiWidgetNode": "Vue Tree Select (Multi) Widget",
|
||||
"DevToolsVueSelectButtonWidgetNode": "Vue Select Button Widget",
|
||||
"DevToolsVueMultiSelectWidgetNode": "Vue Multi Select Widget",
|
||||
"DevToolsVueTextareaWidgetNode": "Vue Textarea Widget",
|
||||
"DevToolsVueChartWidgetNode": "Vue Chart Widget",
|
||||
"DevToolsVueGalleriaWidgetNode": "Vue Galleria Widget",
|
||||
"DevToolsVueMarkdownWidgetNode": "Vue Markdown Widget",
|
||||
"DevToolsVueAudioRecordWidgetNode": "Vue Audio Record Widget",
|
||||
"DevToolsVueColorWidgetNode": "Vue Color Widget",
|
||||
"DevToolsVueAudioPreviewComboNode": "Vue Audio Combo Widget",
|
||||
}
|
||||
|
||||
__all__ = [
|
||||
"VueFileUploadWidgetNode",
|
||||
"VueImageCompareWidgetNode",
|
||||
"VueTreeSelectWidgetNode",
|
||||
"VueTreeSelectMultiWidgetNode",
|
||||
"VueSelectButtonWidgetNode",
|
||||
"VueMultiSelectWidgetNode",
|
||||
"VueTextareaWidgetNode",
|
||||
"VueChartWidgetNode",
|
||||
"VueGalleriaWidgetNode",
|
||||
"VueMarkdownWidgetNode",
|
||||
"VueAudioRecordWidgetNode",
|
||||
"VueColorWidgetNode",
|
||||
"VueAudioPreviewComboNode",
|
||||
"NODE_CLASS_MAPPINGS",
|
||||
"NODE_DISPLAY_NAME_MAPPINGS",
|
||||
]
|
||||
Reference in New Issue
Block a user