|
|
|
|
@ -477,12 +477,7 @@ class ByteDanceTextToVideoNode(IO.ComfyNode):
|
|
|
|
|
inputs=[
|
|
|
|
|
IO.Combo.Input(
|
|
|
|
|
"model",
|
|
|
|
|
options=[
|
|
|
|
|
"seedance-1-5-pro-251215",
|
|
|
|
|
"seedance-1-0-pro-250528",
|
|
|
|
|
"seedance-1-0-lite-t2v-250428",
|
|
|
|
|
"seedance-1-0-pro-fast-251015",
|
|
|
|
|
],
|
|
|
|
|
options=["seedance-1-0-pro-250528", "seedance-1-0-lite-t2v-250428", "seedance-1-0-pro-fast-251015"],
|
|
|
|
|
default="seedance-1-0-pro-fast-251015",
|
|
|
|
|
),
|
|
|
|
|
IO.String.Input(
|
|
|
|
|
@ -533,12 +528,6 @@ class ByteDanceTextToVideoNode(IO.ComfyNode):
|
|
|
|
|
tooltip='Whether to add an "AI generated" watermark to the video.',
|
|
|
|
|
optional=True,
|
|
|
|
|
),
|
|
|
|
|
IO.Boolean.Input(
|
|
|
|
|
"generate_audio",
|
|
|
|
|
default=False,
|
|
|
|
|
tooltip="This parameter is ignored for any model except seedance-1-5-pro.",
|
|
|
|
|
optional=True,
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
outputs=[
|
|
|
|
|
IO.Video.Output(),
|
|
|
|
|
@ -563,10 +552,7 @@ class ByteDanceTextToVideoNode(IO.ComfyNode):
|
|
|
|
|
seed: int,
|
|
|
|
|
camera_fixed: bool,
|
|
|
|
|
watermark: bool,
|
|
|
|
|
generate_audio: bool = False,
|
|
|
|
|
) -> IO.NodeOutput:
|
|
|
|
|
if model == "seedance-1-5-pro-251215" and duration < 4:
|
|
|
|
|
raise ValueError("Minimum supported duration for Seedance 1.5 Pro is 4 seconds.")
|
|
|
|
|
validate_string(prompt, strip_whitespace=True, min_length=1)
|
|
|
|
|
raise_if_text_params(prompt, ["resolution", "ratio", "duration", "seed", "camerafixed", "watermark"])
|
|
|
|
|
|
|
|
|
|
@ -581,11 +567,7 @@ class ByteDanceTextToVideoNode(IO.ComfyNode):
|
|
|
|
|
)
|
|
|
|
|
return await process_video_task(
|
|
|
|
|
cls,
|
|
|
|
|
payload=Text2VideoTaskCreationRequest(
|
|
|
|
|
model=model,
|
|
|
|
|
content=[TaskTextContent(text=prompt)],
|
|
|
|
|
generate_audio=generate_audio if model == "seedance-1-5-pro-251215" else None,
|
|
|
|
|
),
|
|
|
|
|
payload=Text2VideoTaskCreationRequest(model=model, content=[TaskTextContent(text=prompt)]),
|
|
|
|
|
estimated_duration=max(1, math.ceil(VIDEO_TASKS_EXECUTION_TIME[model][resolution] * (duration / 10.0))),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@ -602,12 +584,7 @@ class ByteDanceImageToVideoNode(IO.ComfyNode):
|
|
|
|
|
inputs=[
|
|
|
|
|
IO.Combo.Input(
|
|
|
|
|
"model",
|
|
|
|
|
options=[
|
|
|
|
|
"seedance-1-5-pro-251215",
|
|
|
|
|
"seedance-1-0-pro-250528",
|
|
|
|
|
"seedance-1-0-lite-i2v-250428",
|
|
|
|
|
"seedance-1-0-pro-fast-251015",
|
|
|
|
|
],
|
|
|
|
|
options=["seedance-1-0-pro-250528", "seedance-1-0-lite-t2v-250428", "seedance-1-0-pro-fast-251015"],
|
|
|
|
|
default="seedance-1-0-pro-fast-251015",
|
|
|
|
|
),
|
|
|
|
|
IO.String.Input(
|
|
|
|
|
@ -662,12 +639,6 @@ class ByteDanceImageToVideoNode(IO.ComfyNode):
|
|
|
|
|
tooltip='Whether to add an "AI generated" watermark to the video.',
|
|
|
|
|
optional=True,
|
|
|
|
|
),
|
|
|
|
|
IO.Boolean.Input(
|
|
|
|
|
"generate_audio",
|
|
|
|
|
default=False,
|
|
|
|
|
tooltip="This parameter is ignored for any model except seedance-1-5-pro.",
|
|
|
|
|
optional=True,
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
outputs=[
|
|
|
|
|
IO.Video.Output(),
|
|
|
|
|
@ -693,10 +664,7 @@ class ByteDanceImageToVideoNode(IO.ComfyNode):
|
|
|
|
|
seed: int,
|
|
|
|
|
camera_fixed: bool,
|
|
|
|
|
watermark: bool,
|
|
|
|
|
generate_audio: bool = False,
|
|
|
|
|
) -> IO.NodeOutput:
|
|
|
|
|
if model == "seedance-1-5-pro-251215" and duration < 4:
|
|
|
|
|
raise ValueError("Minimum supported duration for Seedance 1.5 Pro is 4 seconds.")
|
|
|
|
|
validate_string(prompt, strip_whitespace=True, min_length=1)
|
|
|
|
|
raise_if_text_params(prompt, ["resolution", "ratio", "duration", "seed", "camerafixed", "watermark"])
|
|
|
|
|
validate_image_dimensions(image, min_width=300, min_height=300, max_width=6000, max_height=6000)
|
|
|
|
|
@ -718,7 +686,6 @@ class ByteDanceImageToVideoNode(IO.ComfyNode):
|
|
|
|
|
payload=Image2VideoTaskCreationRequest(
|
|
|
|
|
model=model,
|
|
|
|
|
content=[TaskTextContent(text=prompt), TaskImageContent(image_url=TaskImageContentUrl(url=image_url))],
|
|
|
|
|
generate_audio=generate_audio if model == "seedance-1-5-pro-251215" else None,
|
|
|
|
|
),
|
|
|
|
|
estimated_duration=max(1, math.ceil(VIDEO_TASKS_EXECUTION_TIME[model][resolution] * (duration / 10.0))),
|
|
|
|
|
)
|
|
|
|
|
@ -736,7 +703,7 @@ class ByteDanceFirstLastFrameNode(IO.ComfyNode):
|
|
|
|
|
inputs=[
|
|
|
|
|
IO.Combo.Input(
|
|
|
|
|
"model",
|
|
|
|
|
options=["seedance-1-5-pro-251215", "seedance-1-0-pro-250528", "seedance-1-0-lite-i2v-250428"],
|
|
|
|
|
options=["seedance-1-0-pro-250528", "seedance-1-0-lite-i2v-250428"],
|
|
|
|
|
default="seedance-1-0-lite-i2v-250428",
|
|
|
|
|
),
|
|
|
|
|
IO.String.Input(
|
|
|
|
|
@ -795,12 +762,6 @@ class ByteDanceFirstLastFrameNode(IO.ComfyNode):
|
|
|
|
|
tooltip='Whether to add an "AI generated" watermark to the video.',
|
|
|
|
|
optional=True,
|
|
|
|
|
),
|
|
|
|
|
IO.Boolean.Input(
|
|
|
|
|
"generate_audio",
|
|
|
|
|
default=False,
|
|
|
|
|
tooltip="This parameter is ignored for any model except seedance-1-5-pro.",
|
|
|
|
|
optional=True,
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
outputs=[
|
|
|
|
|
IO.Video.Output(),
|
|
|
|
|
@ -827,10 +788,7 @@ class ByteDanceFirstLastFrameNode(IO.ComfyNode):
|
|
|
|
|
seed: int,
|
|
|
|
|
camera_fixed: bool,
|
|
|
|
|
watermark: bool,
|
|
|
|
|
generate_audio: bool = False,
|
|
|
|
|
) -> IO.NodeOutput:
|
|
|
|
|
if model == "seedance-1-5-pro-251215" and duration < 4:
|
|
|
|
|
raise ValueError("Minimum supported duration for Seedance 1.5 Pro is 4 seconds.")
|
|
|
|
|
validate_string(prompt, strip_whitespace=True, min_length=1)
|
|
|
|
|
raise_if_text_params(prompt, ["resolution", "ratio", "duration", "seed", "camerafixed", "watermark"])
|
|
|
|
|
for i in (first_frame, last_frame):
|
|
|
|
|
@ -863,7 +821,6 @@ class ByteDanceFirstLastFrameNode(IO.ComfyNode):
|
|
|
|
|
TaskImageContent(image_url=TaskImageContentUrl(url=str(download_urls[0])), role="first_frame"),
|
|
|
|
|
TaskImageContent(image_url=TaskImageContentUrl(url=str(download_urls[1])), role="last_frame"),
|
|
|
|
|
],
|
|
|
|
|
generate_audio=generate_audio if model == "seedance-1-5-pro-251215" else None,
|
|
|
|
|
),
|
|
|
|
|
estimated_duration=max(1, math.ceil(VIDEO_TASKS_EXECUTION_TIME[model][resolution] * (duration / 10.0))),
|
|
|
|
|
)
|
|
|
|
|
@ -939,41 +896,7 @@ class ByteDanceImageReferenceNode(IO.ComfyNode):
|
|
|
|
|
IO.Hidden.unique_id,
|
|
|
|
|
],
|
|
|
|
|
is_api_node=True,
|
|
|
|
|
price_badge=IO.PriceBadge(
|
|
|
|
|
depends_on=IO.PriceBadgeDepends(widgets=["model", "duration", "resolution"]),
|
|
|
|
|
expr="""
|
|
|
|
|
(
|
|
|
|
|
$priceByModel := {
|
|
|
|
|
"seedance-1-0-pro": {
|
|
|
|
|
"480p":[0.23,0.24],
|
|
|
|
|
"720p":[0.51,0.56]
|
|
|
|
|
},
|
|
|
|
|
"seedance-1-0-lite": {
|
|
|
|
|
"480p":[0.17,0.18],
|
|
|
|
|
"720p":[0.37,0.41]
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
$model := widgets.model;
|
|
|
|
|
$modelKey :=
|
|
|
|
|
$contains($model, "seedance-1-0-pro") ? "seedance-1-0-pro" :
|
|
|
|
|
"seedance-1-0-lite";
|
|
|
|
|
$resolution := widgets.resolution;
|
|
|
|
|
$resKey :=
|
|
|
|
|
$contains($resolution, "720") ? "720p" :
|
|
|
|
|
"480p";
|
|
|
|
|
$modelPrices := $lookup($priceByModel, $modelKey);
|
|
|
|
|
$baseRange := $lookup($modelPrices, $resKey);
|
|
|
|
|
$min10s := $baseRange[0];
|
|
|
|
|
$max10s := $baseRange[1];
|
|
|
|
|
$scale := widgets.duration / 10;
|
|
|
|
|
$minCost := $min10s * $scale;
|
|
|
|
|
$maxCost := $max10s * $scale;
|
|
|
|
|
($minCost = $maxCost)
|
|
|
|
|
? {"type":"usd","usd": $minCost}
|
|
|
|
|
: {"type":"range_usd","min_usd": $minCost, "max_usd": $maxCost}
|
|
|
|
|
)
|
|
|
|
|
""",
|
|
|
|
|
),
|
|
|
|
|
price_badge=PRICE_BADGE_VIDEO,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
@ -1044,15 +967,10 @@ def raise_if_text_params(prompt: str, text_params: list[str]) -> None:
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
PRICE_BADGE_VIDEO = IO.PriceBadge(
|
|
|
|
|
depends_on=IO.PriceBadgeDepends(widgets=["model", "duration", "resolution", "generate_audio"]),
|
|
|
|
|
depends_on=IO.PriceBadgeDepends(widgets=["model", "duration", "resolution"]),
|
|
|
|
|
expr="""
|
|
|
|
|
(
|
|
|
|
|
$priceByModel := {
|
|
|
|
|
"seedance-1-5-pro": {
|
|
|
|
|
"480p":[0.12,0.12],
|
|
|
|
|
"720p":[0.26,0.26],
|
|
|
|
|
"1080p":[0.58,0.59]
|
|
|
|
|
},
|
|
|
|
|
"seedance-1-0-pro": {
|
|
|
|
|
"480p":[0.23,0.24],
|
|
|
|
|
"720p":[0.51,0.56],
|
|
|
|
|
@ -1071,7 +989,6 @@ PRICE_BADGE_VIDEO = IO.PriceBadge(
|
|
|
|
|
};
|
|
|
|
|
$model := widgets.model;
|
|
|
|
|
$modelKey :=
|
|
|
|
|
$contains($model, "seedance-1-5-pro") ? "seedance-1-5-pro" :
|
|
|
|
|
$contains($model, "seedance-1-0-pro-fast") ? "seedance-1-0-pro-fast" :
|
|
|
|
|
$contains($model, "seedance-1-0-pro") ? "seedance-1-0-pro" :
|
|
|
|
|
"seedance-1-0-lite";
|
|
|
|
|
@ -1085,12 +1002,11 @@ PRICE_BADGE_VIDEO = IO.PriceBadge(
|
|
|
|
|
$min10s := $baseRange[0];
|
|
|
|
|
$max10s := $baseRange[1];
|
|
|
|
|
$scale := widgets.duration / 10;
|
|
|
|
|
$audioMultiplier := ($modelKey = "seedance-1-5-pro" and widgets.generate_audio) ? 2 : 1;
|
|
|
|
|
$minCost := $min10s * $scale * $audioMultiplier;
|
|
|
|
|
$maxCost := $max10s * $scale * $audioMultiplier;
|
|
|
|
|
$minCost := $min10s * $scale;
|
|
|
|
|
$maxCost := $max10s * $scale;
|
|
|
|
|
($minCost = $maxCost)
|
|
|
|
|
? {"type":"usd","usd": $minCost, "format": { "approximate": true }}
|
|
|
|
|
: {"type":"range_usd","min_usd": $minCost, "max_usd": $maxCost, "format": { "approximate": true }}
|
|
|
|
|
? {"type":"usd","usd": $minCost}
|
|
|
|
|
: {"type":"range_usd","min_usd": $minCost, "max_usd": $maxCost}
|
|
|
|
|
)
|
|
|
|
|
""",
|
|
|
|
|
)
|
|
|
|
|
|