mirror of
https://github.com/invoke-ai/InvokeAI.git
synced 2026-01-16 13:08:01 -05:00
Compare commits
191 Commits
v6.6.0rc2
...
controlnet
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
21a05f4287 | ||
|
|
efcb1bea7f | ||
|
|
e0d7a401f3 | ||
|
|
aac979e9a4 | ||
|
|
3b0d7f076d | ||
|
|
e1acbcdbd5 | ||
|
|
7d9b81550b | ||
|
|
6a447dd1fe | ||
|
|
c2dc63ddbc | ||
|
|
1bc689d531 | ||
|
|
4829975827 | ||
|
|
49da4e00c3 | ||
|
|
89dfe5e729 | ||
|
|
6816d366df | ||
|
|
9d3d2a36c9 | ||
|
|
ed231044c8 | ||
|
|
b51a232794 | ||
|
|
4412143a6e | ||
|
|
de11cafdb3 | ||
|
|
4d9114aa7d | ||
|
|
67e2da1ebf | ||
|
|
33ecc591c3 | ||
|
|
b57459a226 | ||
|
|
01282b1c90 | ||
|
|
3f302906dc | ||
|
|
81d56596fb | ||
|
|
b536b0df0c | ||
|
|
692af1d93d | ||
|
|
bb7ef77b50 | ||
|
|
1862548573 | ||
|
|
242c1b6350 | ||
|
|
fc6e4bb04e | ||
|
|
20841abca6 | ||
|
|
e8b69d99a4 | ||
|
|
d6eaff8237 | ||
|
|
068b095956 | ||
|
|
f795a47340 | ||
|
|
df47345eb0 | ||
|
|
def04095a4 | ||
|
|
28be8f0911 | ||
|
|
b50c44bac0 | ||
|
|
b4ce0e02fc | ||
|
|
d6442d9a34 | ||
|
|
4528bcafaf | ||
|
|
8b82b81ee2 | ||
|
|
757acdd49e | ||
|
|
94b7cc583a | ||
|
|
b663a6bac4 | ||
|
|
65d40153fb | ||
|
|
c8b741a514 | ||
|
|
6d3aeffed9 | ||
|
|
203be96910 | ||
|
|
b0aa48ddb8 | ||
|
|
867dbe51e5 | ||
|
|
ff8948b6f1 | ||
|
|
fa3a6425a6 | ||
|
|
c5992ece89 | ||
|
|
12a6239929 | ||
|
|
e9238c59f4 | ||
|
|
c1cbbe51d6 | ||
|
|
4219b4a288 | ||
|
|
48c8a9c09d | ||
|
|
a67efdf4ad | ||
|
|
d6ff9c2e49 | ||
|
|
e768a3bc7b | ||
|
|
7273700f61 | ||
|
|
f909e81d91 | ||
|
|
8c85f168f6 | ||
|
|
263d86d46f | ||
|
|
0921805160 | ||
|
|
517f4811e7 | ||
|
|
0dc73c8803 | ||
|
|
26702b54c0 | ||
|
|
2d65e4543f | ||
|
|
309113956b | ||
|
|
0ac4099bc6 | ||
|
|
899dc739fa | ||
|
|
4e2439fc8e | ||
|
|
00864c24e0 | ||
|
|
b73aaa7d6f | ||
|
|
85057ae704 | ||
|
|
c3fb3a43a2 | ||
|
|
51d0a15a1b | ||
|
|
5991067fd9 | ||
|
|
32c2d3f740 | ||
|
|
c661f86b34 | ||
|
|
cc72d8eab4 | ||
|
|
e8550f9355 | ||
|
|
a1d0386ca4 | ||
|
|
495d089f85 | ||
|
|
913b91e9dd | ||
|
|
3e907f4e14 | ||
|
|
756df6ebe4 | ||
|
|
2a6be99152 | ||
|
|
3099e2bf9d | ||
|
|
6921f0412a | ||
|
|
022d5a8863 | ||
|
|
af99beedc5 | ||
|
|
f3d83dc6b7 | ||
|
|
ebc3f18a1a | ||
|
|
aeb512f8d9 | ||
|
|
a1810acb93 | ||
|
|
aa35a5083b | ||
|
|
4f17de0b32 | ||
|
|
370c3cd59b | ||
|
|
67214e16c0 | ||
|
|
4880a1d946 | ||
|
|
0f0988610f | ||
|
|
6805d28b7a | ||
|
|
9b45a24136 | ||
|
|
4e9d66a64b | ||
|
|
8fec530b0f | ||
|
|
50c66f8671 | ||
|
|
f0aa39ea81 | ||
|
|
faac814a3d | ||
|
|
fb9545bb90 | ||
|
|
8ad2ee83b6 | ||
|
|
f8ad62b5eb | ||
|
|
03ae78bc7c | ||
|
|
ec1a058dbe | ||
|
|
9e4d441e2e | ||
|
|
3770fd22f8 | ||
|
|
a0232b0e63 | ||
|
|
e1e964bf0e | ||
|
|
1b1759cffc | ||
|
|
d828502bc8 | ||
|
|
7a073b6de7 | ||
|
|
338ff8d588 | ||
|
|
a3625efd3a | ||
|
|
5efb37fe63 | ||
|
|
aef0b81d5b | ||
|
|
544edff507 | ||
|
|
42b1adab22 | ||
|
|
a2b9d12e88 | ||
|
|
7a94fb6c04 | ||
|
|
efcd159704 | ||
|
|
997e619a9d | ||
|
|
4bc184ff16 | ||
|
|
0b605a745b | ||
|
|
22b038ce3b | ||
|
|
0bb5d647b5 | ||
|
|
4a3599929b | ||
|
|
f959ce8323 | ||
|
|
74e1047870 | ||
|
|
732881c51b | ||
|
|
107be8e166 | ||
|
|
3c2f654da8 | ||
|
|
474fd44e50 | ||
|
|
0dc5f8fd65 | ||
|
|
d4215fb460 | ||
|
|
0cd05ee9fd | ||
|
|
9fcb3af1d8 | ||
|
|
c9da7e2172 | ||
|
|
9788735d6b | ||
|
|
d6139748e2 | ||
|
|
602dfb1e5d | ||
|
|
5bb3a78f56 | ||
|
|
d58df1e17b | ||
|
|
5d0e37eb2f | ||
|
|
486b333cef | ||
|
|
6fa437af03 | ||
|
|
787ef6fa27 | ||
|
|
7f0571c229 | ||
|
|
f5a58c0ceb | ||
|
|
d16eef4e66 | ||
|
|
681ff2b2b3 | ||
|
|
0d81b4ce98 | ||
|
|
99f1667ced | ||
|
|
aa5597ab4d | ||
|
|
9bbb8e8a5e | ||
|
|
f284d282c1 | ||
|
|
4231488da6 | ||
|
|
a014867e68 | ||
|
|
22654fbc9c | ||
|
|
daa4fd751c | ||
|
|
3fd265c333 | ||
|
|
26a3a9130c | ||
|
|
3dfeaab4b2 | ||
|
|
a33707cc76 | ||
|
|
21e13daf6e | ||
|
|
fa2614ee02 | ||
|
|
4be6ddb23d | ||
|
|
bba0e01926 | ||
|
|
20d57d5ccf | ||
|
|
d9121271a2 | ||
|
|
30b487c71c | ||
|
|
8a81c05caf | ||
|
|
a0dceecab9 | ||
|
|
98945a4560 | ||
|
|
9610f34dd4 | ||
|
|
8a00d855b4 |
1
.github/pull_request_template.md
vendored
1
.github/pull_request_template.md
vendored
@@ -18,5 +18,6 @@
|
||||
|
||||
- [ ] _The PR has a short but descriptive title, suitable for a changelog_
|
||||
- [ ] _Tests added / updated (if applicable)_
|
||||
- [ ] _❗Changes to a redux slice have a corresponding migration_
|
||||
- [ ] _Documentation added / updated (if applicable)_
|
||||
- [ ] _Updated `What's New` copy (if doing a release after this PR)_
|
||||
|
||||
@@ -7,7 +7,6 @@ from pydantic import BaseModel, Field
|
||||
from invokeai.app.api.dependencies import ApiDependencies
|
||||
from invokeai.app.services.session_processor.session_processor_common import SessionProcessorStatus
|
||||
from invokeai.app.services.session_queue.session_queue_common import (
|
||||
QUEUE_ITEM_STATUS,
|
||||
Batch,
|
||||
BatchStatus,
|
||||
CancelAllExceptCurrentResult,
|
||||
@@ -18,6 +17,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
||||
DeleteByDestinationResult,
|
||||
EnqueueBatchResult,
|
||||
FieldIdentifier,
|
||||
ItemIdsResult,
|
||||
PruneResult,
|
||||
RetryItemsResult,
|
||||
SessionQueueCountsByDestination,
|
||||
@@ -25,7 +25,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
||||
SessionQueueItemNotFoundError,
|
||||
SessionQueueStatus,
|
||||
)
|
||||
from invokeai.app.services.shared.pagination import CursorPaginatedResults
|
||||
from invokeai.app.services.shared.sqlite.sqlite_common import SQLiteDirection
|
||||
|
||||
session_queue_router = APIRouter(prefix="/v1/queue", tags=["queue"])
|
||||
|
||||
@@ -68,36 +68,6 @@ async def enqueue_batch(
|
||||
raise HTTPException(status_code=500, detail=f"Unexpected error while enqueuing batch: {e}")
|
||||
|
||||
|
||||
@session_queue_router.get(
|
||||
"/{queue_id}/list",
|
||||
operation_id="list_queue_items",
|
||||
responses={
|
||||
200: {"model": CursorPaginatedResults[SessionQueueItem]},
|
||||
},
|
||||
)
|
||||
async def list_queue_items(
|
||||
queue_id: str = Path(description="The queue id to perform this operation on"),
|
||||
limit: int = Query(default=50, description="The number of items to fetch"),
|
||||
status: Optional[QUEUE_ITEM_STATUS] = Query(default=None, description="The status of items to fetch"),
|
||||
cursor: Optional[int] = Query(default=None, description="The pagination cursor"),
|
||||
priority: int = Query(default=0, description="The pagination cursor priority"),
|
||||
destination: Optional[str] = Query(default=None, description="The destination of queue items to fetch"),
|
||||
) -> CursorPaginatedResults[SessionQueueItem]:
|
||||
"""Gets cursor-paginated queue items"""
|
||||
|
||||
try:
|
||||
return ApiDependencies.invoker.services.session_queue.list_queue_items(
|
||||
queue_id=queue_id,
|
||||
limit=limit,
|
||||
status=status,
|
||||
cursor=cursor,
|
||||
priority=priority,
|
||||
destination=destination,
|
||||
)
|
||||
except Exception as e:
|
||||
raise HTTPException(status_code=500, detail=f"Unexpected error while listing all items: {e}")
|
||||
|
||||
|
||||
@session_queue_router.get(
|
||||
"/{queue_id}/list_all",
|
||||
operation_id="list_all_queue_items",
|
||||
@@ -119,6 +89,56 @@ async def list_all_queue_items(
|
||||
raise HTTPException(status_code=500, detail=f"Unexpected error while listing all queue items: {e}")
|
||||
|
||||
|
||||
@session_queue_router.get(
|
||||
"/{queue_id}/item_ids",
|
||||
operation_id="get_queue_item_ids",
|
||||
responses={
|
||||
200: {"model": ItemIdsResult},
|
||||
},
|
||||
)
|
||||
async def get_queue_item_ids(
|
||||
queue_id: str = Path(description="The queue id to perform this operation on"),
|
||||
order_dir: SQLiteDirection = Query(default=SQLiteDirection.Descending, description="The order of sort"),
|
||||
) -> ItemIdsResult:
|
||||
"""Gets all queue item ids that match the given parameters"""
|
||||
try:
|
||||
return ApiDependencies.invoker.services.session_queue.get_queue_item_ids(queue_id=queue_id, order_dir=order_dir)
|
||||
except Exception as e:
|
||||
raise HTTPException(status_code=500, detail=f"Unexpected error while listing all queue item ids: {e}")
|
||||
|
||||
|
||||
@session_queue_router.post(
|
||||
"/{queue_id}/items_by_ids",
|
||||
operation_id="get_queue_items_by_item_ids",
|
||||
responses={200: {"model": list[SessionQueueItem]}},
|
||||
)
|
||||
async def get_queue_items_by_item_ids(
|
||||
queue_id: str = Path(description="The queue id to perform this operation on"),
|
||||
item_ids: list[int] = Body(
|
||||
embed=True, description="Object containing list of queue item ids to fetch queue items for"
|
||||
),
|
||||
) -> list[SessionQueueItem]:
|
||||
"""Gets queue items for the specified queue item ids. Maintains order of item ids."""
|
||||
try:
|
||||
session_queue_service = ApiDependencies.invoker.services.session_queue
|
||||
|
||||
# Fetch queue items preserving the order of requested item ids
|
||||
queue_items: list[SessionQueueItem] = []
|
||||
for item_id in item_ids:
|
||||
try:
|
||||
queue_item = session_queue_service.get_queue_item(item_id=item_id)
|
||||
if queue_item.queue_id != queue_id: # Auth protection for items from other queues
|
||||
continue
|
||||
queue_items.append(queue_item)
|
||||
except Exception:
|
||||
# Skip missing queue items - they may have been deleted between item id fetch and queue item fetch
|
||||
continue
|
||||
|
||||
return queue_items
|
||||
except Exception:
|
||||
raise HTTPException(status_code=500, detail="Failed to get queue items")
|
||||
|
||||
|
||||
@session_queue_router.put(
|
||||
"/{queue_id}/processor/resume",
|
||||
operation_id="resume",
|
||||
@@ -354,7 +374,10 @@ async def get_queue_item(
|
||||
) -> SessionQueueItem:
|
||||
"""Gets a queue item"""
|
||||
try:
|
||||
return ApiDependencies.invoker.services.session_queue.get_queue_item(item_id)
|
||||
queue_item = ApiDependencies.invoker.services.session_queue.get_queue_item(item_id=item_id)
|
||||
if queue_item.queue_id != queue_id:
|
||||
raise HTTPException(status_code=404, detail=f"Queue item with id {item_id} not found in queue {queue_id}")
|
||||
return queue_item
|
||||
except SessionQueueItemNotFoundError:
|
||||
raise HTTPException(status_code=404, detail=f"Queue item with id {item_id} not found in queue {queue_id}")
|
||||
except Exception as e:
|
||||
|
||||
@@ -1,11 +1,12 @@
|
||||
from enum import Enum
|
||||
from typing import Any, Callable, Optional, Tuple
|
||||
|
||||
from pydantic import BaseModel, ConfigDict, Field, RootModel, TypeAdapter, model_validator
|
||||
from pydantic import BaseModel, ConfigDict, Field, RootModel, TypeAdapter
|
||||
from pydantic.fields import _Unset
|
||||
from pydantic_core import PydanticUndefined
|
||||
|
||||
from invokeai.app.util.metaenum import MetaEnum
|
||||
from invokeai.backend.image_util.segment_anything.shared import BoundingBox
|
||||
from invokeai.backend.util.logging import InvokeAILogger
|
||||
|
||||
logger = InvokeAILogger.get_logger()
|
||||
@@ -331,14 +332,9 @@ class ConditioningField(BaseModel):
|
||||
)
|
||||
|
||||
|
||||
class BoundingBoxField(BaseModel):
|
||||
class BoundingBoxField(BoundingBox):
|
||||
"""A bounding box primitive value."""
|
||||
|
||||
x_min: int = Field(ge=0, description="The minimum x-coordinate of the bounding box (inclusive).")
|
||||
x_max: int = Field(ge=0, description="The maximum x-coordinate of the bounding box (exclusive).")
|
||||
y_min: int = Field(ge=0, description="The minimum y-coordinate of the bounding box (inclusive).")
|
||||
y_max: int = Field(ge=0, description="The maximum y-coordinate of the bounding box (exclusive).")
|
||||
|
||||
score: Optional[float] = Field(
|
||||
default=None,
|
||||
ge=0.0,
|
||||
@@ -347,21 +343,6 @@ class BoundingBoxField(BaseModel):
|
||||
"when the bounding box was produced by a detector and has an associated confidence score.",
|
||||
)
|
||||
|
||||
@model_validator(mode="after")
|
||||
def check_coords(self):
|
||||
if self.x_min > self.x_max:
|
||||
raise ValueError(f"x_min ({self.x_min}) is greater than x_max ({self.x_max}).")
|
||||
if self.y_min > self.y_max:
|
||||
raise ValueError(f"y_min ({self.y_min}) is greater than y_max ({self.y_max}).")
|
||||
return self
|
||||
|
||||
def tuple(self) -> Tuple[int, int, int, int]:
|
||||
"""
|
||||
Returns the bounding box as a tuple suitable for use with PIL's `Image.crop()` method.
|
||||
This method returns a tuple of the form (left, upper, right, lower) == (x_min, y_min, x_max, y_max).
|
||||
"""
|
||||
return (self.x_min, self.y_min, self.x_max, self.y_max)
|
||||
|
||||
|
||||
class MetadataField(RootModel[dict[str, Any]]):
|
||||
"""
|
||||
|
||||
@@ -1,72 +1,75 @@
|
||||
from enum import Enum
|
||||
from itertools import zip_longest
|
||||
from pathlib import Path
|
||||
from typing import Literal
|
||||
|
||||
import numpy as np
|
||||
import torch
|
||||
from PIL import Image
|
||||
from pydantic import BaseModel, Field
|
||||
from transformers import AutoProcessor
|
||||
from pydantic import BaseModel, Field, model_validator
|
||||
from transformers.models.sam import SamModel
|
||||
from transformers.models.sam.processing_sam import SamProcessor
|
||||
from transformers.models.sam2 import Sam2Model
|
||||
from transformers.models.sam2.processing_sam2 import Sam2Processor
|
||||
|
||||
from invokeai.app.invocations.baseinvocation import BaseInvocation, invocation
|
||||
from invokeai.app.invocations.fields import BoundingBoxField, ImageField, InputField, TensorField
|
||||
from invokeai.app.invocations.primitives import MaskOutput
|
||||
from invokeai.app.services.shared.invocation_context import InvocationContext
|
||||
from invokeai.backend.image_util.segment_anything.mask_refinement import mask_to_polygon, polygon_to_mask
|
||||
from invokeai.backend.image_util.segment_anything.segment_anything_2_pipeline import SegmentAnything2Pipeline
|
||||
from invokeai.backend.image_util.segment_anything.segment_anything_pipeline import SegmentAnythingPipeline
|
||||
from invokeai.backend.image_util.segment_anything.shared import SAMInput, SAMPoint
|
||||
|
||||
SegmentAnythingModelKey = Literal["segment-anything-base", "segment-anything-large", "segment-anything-huge"]
|
||||
SegmentAnythingModelKey = Literal[
|
||||
"segment-anything-base",
|
||||
"segment-anything-large",
|
||||
"segment-anything-huge",
|
||||
"segment-anything-2-tiny",
|
||||
"segment-anything-2-small",
|
||||
"segment-anything-2-base",
|
||||
"segment-anything-2-large",
|
||||
]
|
||||
SEGMENT_ANYTHING_MODEL_IDS: dict[SegmentAnythingModelKey, str] = {
|
||||
"segment-anything-base": "facebook/sam-vit-base",
|
||||
"segment-anything-large": "facebook/sam-vit-large",
|
||||
"segment-anything-huge": "facebook/sam-vit-huge",
|
||||
"segment-anything-2-tiny": "facebook/sam2.1-hiera-tiny",
|
||||
"segment-anything-2-small": "facebook/sam2.1-hiera-small",
|
||||
"segment-anything-2-base": "facebook/sam2.1-hiera-base-plus",
|
||||
"segment-anything-2-large": "facebook/sam2.1-hiera-large",
|
||||
}
|
||||
|
||||
|
||||
class SAMPointLabel(Enum):
|
||||
negative = -1
|
||||
neutral = 0
|
||||
positive = 1
|
||||
|
||||
|
||||
class SAMPoint(BaseModel):
|
||||
x: int = Field(..., description="The x-coordinate of the point")
|
||||
y: int = Field(..., description="The y-coordinate of the point")
|
||||
label: SAMPointLabel = Field(..., description="The label of the point")
|
||||
|
||||
|
||||
class SAMPointsField(BaseModel):
|
||||
points: list[SAMPoint] = Field(..., description="The points of the object")
|
||||
points: list[SAMPoint] = Field(..., description="The points of the object", min_length=1)
|
||||
|
||||
def to_list(self) -> list[list[int]]:
|
||||
def to_list(self) -> list[list[float]]:
|
||||
return [[point.x, point.y, point.label.value] for point in self.points]
|
||||
|
||||
|
||||
@invocation(
|
||||
"segment_anything",
|
||||
title="Segment Anything",
|
||||
tags=["prompt", "segmentation"],
|
||||
tags=["prompt", "segmentation", "sam", "sam2"],
|
||||
category="segmentation",
|
||||
version="1.2.0",
|
||||
version="1.3.0",
|
||||
)
|
||||
class SegmentAnythingInvocation(BaseInvocation):
|
||||
"""Runs a Segment Anything Model."""
|
||||
"""Runs a Segment Anything Model (SAM or SAM2)."""
|
||||
|
||||
# Reference:
|
||||
# - https://arxiv.org/pdf/2304.02643
|
||||
# - https://huggingface.co/docs/transformers/v4.43.3/en/model_doc/grounding-dino#grounded-sam
|
||||
# - https://github.com/NielsRogge/Transformers-Tutorials/blob/a39f33ac1557b02ebfb191ea7753e332b5ca933f/Grounding%20DINO/GroundingDINO_with_Segment_Anything.ipynb
|
||||
|
||||
model: SegmentAnythingModelKey = InputField(description="The Segment Anything model to use.")
|
||||
model: SegmentAnythingModelKey = InputField(description="The Segment Anything model to use (SAM or SAM2).")
|
||||
image: ImageField = InputField(description="The image to segment.")
|
||||
bounding_boxes: list[BoundingBoxField] | None = InputField(
|
||||
default=None, description="The bounding boxes to prompt the SAM model with."
|
||||
default=None, description="The bounding boxes to prompt the model with."
|
||||
)
|
||||
point_lists: list[SAMPointsField] | None = InputField(
|
||||
default=None,
|
||||
description="The list of point lists to prompt the SAM model with. Each list of points represents a single object.",
|
||||
description="The list of point lists to prompt the model with. Each list of points represents a single object.",
|
||||
)
|
||||
apply_polygon_refinement: bool = InputField(
|
||||
description="Whether to apply polygon refinement to the masks. This will smooth the edges of the masks slightly and ensure that each mask consists of a single closed polygon (before merging).",
|
||||
@@ -77,14 +80,18 @@ class SegmentAnythingInvocation(BaseInvocation):
|
||||
default="all",
|
||||
)
|
||||
|
||||
@model_validator(mode="after")
|
||||
def validate_points_and_boxes_len(self):
|
||||
if self.point_lists is not None and self.bounding_boxes is not None:
|
||||
if len(self.point_lists) != len(self.bounding_boxes):
|
||||
raise ValueError("If both point_lists and bounding_boxes are provided, they must have the same length.")
|
||||
return self
|
||||
|
||||
@torch.no_grad()
|
||||
def invoke(self, context: InvocationContext) -> MaskOutput:
|
||||
# The models expect a 3-channel RGB image.
|
||||
image_pil = context.images.get_pil(self.image.image_name, mode="RGB")
|
||||
|
||||
if self.point_lists is not None and self.bounding_boxes is not None:
|
||||
raise ValueError("Only one of point_lists or bounding_box can be provided.")
|
||||
|
||||
if (not self.bounding_boxes or len(self.bounding_boxes) == 0) and (
|
||||
not self.point_lists or len(self.point_lists) == 0
|
||||
):
|
||||
@@ -111,26 +118,38 @@ class SegmentAnythingInvocation(BaseInvocation):
|
||||
# model, and figure out how to make it work in the pipeline.
|
||||
# torch_dtype=TorchDevice.choose_torch_dtype(),
|
||||
)
|
||||
|
||||
sam_processor = AutoProcessor.from_pretrained(model_path, local_files_only=True)
|
||||
assert isinstance(sam_processor, SamProcessor)
|
||||
sam_processor = SamProcessor.from_pretrained(model_path, local_files_only=True)
|
||||
return SegmentAnythingPipeline(sam_model=sam_model, sam_processor=sam_processor)
|
||||
|
||||
def _segment(self, context: InvocationContext, image: Image.Image) -> list[torch.Tensor]:
|
||||
"""Use Segment Anything (SAM) to generate masks given an image + a set of bounding boxes."""
|
||||
# Convert the bounding boxes to the SAM input format.
|
||||
sam_bounding_boxes = (
|
||||
[[bb.x_min, bb.y_min, bb.x_max, bb.y_max] for bb in self.bounding_boxes] if self.bounding_boxes else None
|
||||
)
|
||||
sam_points = [p.to_list() for p in self.point_lists] if self.point_lists else None
|
||||
@staticmethod
|
||||
def _load_sam_2_model(model_path: Path):
|
||||
sam2_model = Sam2Model.from_pretrained(model_path, local_files_only=True)
|
||||
sam2_processor = Sam2Processor.from_pretrained(model_path, local_files_only=True)
|
||||
return SegmentAnything2Pipeline(sam2_model=sam2_model, sam2_processor=sam2_processor)
|
||||
|
||||
with (
|
||||
context.models.load_remote_model(
|
||||
source=SEGMENT_ANYTHING_MODEL_IDS[self.model], loader=SegmentAnythingInvocation._load_sam_model
|
||||
) as sam_pipeline,
|
||||
):
|
||||
assert isinstance(sam_pipeline, SegmentAnythingPipeline)
|
||||
masks = sam_pipeline.segment(image=image, bounding_boxes=sam_bounding_boxes, point_lists=sam_points)
|
||||
def _segment(self, context: InvocationContext, image: Image.Image) -> list[torch.Tensor]:
|
||||
"""Use Segment Anything (SAM or SAM2) to generate masks given an image + a set of bounding boxes."""
|
||||
|
||||
source = SEGMENT_ANYTHING_MODEL_IDS[self.model]
|
||||
inputs: list[SAMInput] = []
|
||||
for bbox_field, point_field in zip_longest(self.bounding_boxes or [], self.point_lists or [], fillvalue=None):
|
||||
inputs.append(
|
||||
SAMInput(
|
||||
bounding_box=bbox_field,
|
||||
points=point_field.points if point_field else None,
|
||||
)
|
||||
)
|
||||
|
||||
if "sam2" in source:
|
||||
loader = SegmentAnythingInvocation._load_sam_2_model
|
||||
with context.models.load_remote_model(source=source, loader=loader) as pipeline:
|
||||
assert isinstance(pipeline, SegmentAnything2Pipeline)
|
||||
masks = pipeline.segment(image=image, inputs=inputs)
|
||||
else:
|
||||
loader = SegmentAnythingInvocation._load_sam_model
|
||||
with context.models.load_remote_model(source=source, loader=loader) as pipeline:
|
||||
assert isinstance(pipeline, SegmentAnythingPipeline)
|
||||
masks = pipeline.segment(image=image, inputs=inputs)
|
||||
|
||||
masks = self._process_masks(masks)
|
||||
if self.apply_polygon_refinement:
|
||||
|
||||
@@ -150,4 +150,15 @@ class BulkDownloadService(BulkDownloadBase):
|
||||
def _is_valid_path(self, path: Union[str, Path]) -> bool:
|
||||
"""Validates the path given for a bulk download."""
|
||||
path = path if isinstance(path, Path) else Path(path)
|
||||
return path.exists()
|
||||
|
||||
# Resolve the path to handle any path traversal attempts (e.g., ../)
|
||||
resolved_path = path.resolve()
|
||||
|
||||
# The path may not traverse out of the bulk downloads folder or its subfolders
|
||||
does_not_traverse = resolved_path.parent == self._bulk_downloads_folder.resolve()
|
||||
|
||||
# The path must exist and be a .zip file
|
||||
does_exist = resolved_path.exists()
|
||||
is_zip_file = resolved_path.suffix == ".zip"
|
||||
|
||||
return does_exist and is_zip_file and does_not_traverse
|
||||
|
||||
@@ -234,8 +234,8 @@ class QueueItemStatusChangedEvent(QueueItemEventBase):
|
||||
error_type: Optional[str] = Field(default=None, description="The error type, if any")
|
||||
error_message: Optional[str] = Field(default=None, description="The error message, if any")
|
||||
error_traceback: Optional[str] = Field(default=None, description="The error traceback, if any")
|
||||
created_at: Optional[str] = Field(default=None, description="The timestamp when the queue item was created")
|
||||
updated_at: Optional[str] = Field(default=None, description="The timestamp when the queue item was last updated")
|
||||
created_at: str = Field(description="The timestamp when the queue item was created")
|
||||
updated_at: str = Field(description="The timestamp when the queue item was last updated")
|
||||
started_at: Optional[str] = Field(default=None, description="The timestamp when the queue item was started")
|
||||
completed_at: Optional[str] = Field(default=None, description="The timestamp when the queue item was completed")
|
||||
batch_status: BatchStatus = Field(description="The status of the batch")
|
||||
@@ -258,8 +258,8 @@ class QueueItemStatusChangedEvent(QueueItemEventBase):
|
||||
error_type=queue_item.error_type,
|
||||
error_message=queue_item.error_message,
|
||||
error_traceback=queue_item.error_traceback,
|
||||
created_at=str(queue_item.created_at) if queue_item.created_at else None,
|
||||
updated_at=str(queue_item.updated_at) if queue_item.updated_at else None,
|
||||
created_at=str(queue_item.created_at),
|
||||
updated_at=str(queue_item.updated_at),
|
||||
started_at=str(queue_item.started_at) if queue_item.started_at else None,
|
||||
completed_at=str(queue_item.completed_at) if queue_item.completed_at else None,
|
||||
batch_status=batch_status,
|
||||
|
||||
@@ -2,7 +2,6 @@ from abc import ABC, abstractmethod
|
||||
from typing import Any, Coroutine, Optional
|
||||
|
||||
from invokeai.app.services.session_queue.session_queue_common import (
|
||||
QUEUE_ITEM_STATUS,
|
||||
Batch,
|
||||
BatchStatus,
|
||||
CancelAllExceptCurrentResult,
|
||||
@@ -15,6 +14,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
||||
EnqueueBatchResult,
|
||||
IsEmptyResult,
|
||||
IsFullResult,
|
||||
ItemIdsResult,
|
||||
PruneResult,
|
||||
RetryItemsResult,
|
||||
SessionQueueCountsByDestination,
|
||||
@@ -22,7 +22,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
||||
SessionQueueStatus,
|
||||
)
|
||||
from invokeai.app.services.shared.graph import GraphExecutionState
|
||||
from invokeai.app.services.shared.pagination import CursorPaginatedResults
|
||||
from invokeai.app.services.shared.sqlite.sqlite_common import SQLiteDirection
|
||||
|
||||
|
||||
class SessionQueueBase(ABC):
|
||||
@@ -135,19 +135,6 @@ class SessionQueueBase(ABC):
|
||||
"""Deletes all queue items except in-progress items"""
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def list_queue_items(
|
||||
self,
|
||||
queue_id: str,
|
||||
limit: int,
|
||||
priority: int,
|
||||
cursor: Optional[int] = None,
|
||||
status: Optional[QUEUE_ITEM_STATUS] = None,
|
||||
destination: Optional[str] = None,
|
||||
) -> CursorPaginatedResults[SessionQueueItem]:
|
||||
"""Gets a page of session queue items"""
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def list_all_queue_items(
|
||||
self,
|
||||
@@ -157,9 +144,18 @@ class SessionQueueBase(ABC):
|
||||
"""Gets all queue items that match the given parameters"""
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def get_queue_item_ids(
|
||||
self,
|
||||
queue_id: str,
|
||||
order_dir: SQLiteDirection = SQLiteDirection.Descending,
|
||||
) -> ItemIdsResult:
|
||||
"""Gets all queue item ids that match the given parameters"""
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def get_queue_item(self, item_id: int) -> SessionQueueItem:
|
||||
"""Gets a session queue item by ID"""
|
||||
"""Gets a session queue item by ID for a given queue"""
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
|
||||
@@ -176,6 +176,14 @@ DEFAULT_QUEUE_ID = "default"
|
||||
|
||||
QUEUE_ITEM_STATUS = Literal["pending", "in_progress", "completed", "failed", "canceled"]
|
||||
|
||||
|
||||
class ItemIdsResult(BaseModel):
|
||||
"""Response containing ordered item ids with metadata for optimistic updates."""
|
||||
|
||||
item_ids: list[int] = Field(description="Ordered list of item ids")
|
||||
total_count: int = Field(description="Total number of queue items matching the query")
|
||||
|
||||
|
||||
NodeFieldValueValidator = TypeAdapter(list[NodeFieldValue])
|
||||
|
||||
|
||||
|
||||
@@ -22,6 +22,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
||||
EnqueueBatchResult,
|
||||
IsEmptyResult,
|
||||
IsFullResult,
|
||||
ItemIdsResult,
|
||||
PruneResult,
|
||||
RetryItemsResult,
|
||||
SessionQueueCountsByDestination,
|
||||
@@ -33,7 +34,7 @@ from invokeai.app.services.session_queue.session_queue_common import (
|
||||
prepare_values_to_insert,
|
||||
)
|
||||
from invokeai.app.services.shared.graph import GraphExecutionState
|
||||
from invokeai.app.services.shared.pagination import CursorPaginatedResults
|
||||
from invokeai.app.services.shared.sqlite.sqlite_common import SQLiteDirection
|
||||
from invokeai.app.services.shared.sqlite.sqlite_database import SqliteDatabase
|
||||
|
||||
|
||||
@@ -587,59 +588,6 @@ class SqliteSessionQueue(SessionQueueBase):
|
||||
)
|
||||
return self.get_queue_item(item_id)
|
||||
|
||||
def list_queue_items(
|
||||
self,
|
||||
queue_id: str,
|
||||
limit: int,
|
||||
priority: int,
|
||||
cursor: Optional[int] = None,
|
||||
status: Optional[QUEUE_ITEM_STATUS] = None,
|
||||
destination: Optional[str] = None,
|
||||
) -> CursorPaginatedResults[SessionQueueItem]:
|
||||
with self._db.transaction() as cursor_:
|
||||
item_id = cursor
|
||||
query = """--sql
|
||||
SELECT *
|
||||
FROM session_queue
|
||||
WHERE queue_id = ?
|
||||
"""
|
||||
params: list[Union[str, int]] = [queue_id]
|
||||
|
||||
if status is not None:
|
||||
query += """--sql
|
||||
AND status = ?
|
||||
"""
|
||||
params.append(status)
|
||||
|
||||
if destination is not None:
|
||||
query += """---sql
|
||||
AND destination = ?
|
||||
"""
|
||||
params.append(destination)
|
||||
|
||||
if item_id is not None:
|
||||
query += """--sql
|
||||
AND (priority < ?) OR (priority = ? AND item_id > ?)
|
||||
"""
|
||||
params.extend([priority, priority, item_id])
|
||||
|
||||
query += """--sql
|
||||
ORDER BY
|
||||
priority DESC,
|
||||
item_id ASC
|
||||
LIMIT ?
|
||||
"""
|
||||
params.append(limit + 1)
|
||||
cursor_.execute(query, params)
|
||||
results = cast(list[sqlite3.Row], cursor_.fetchall())
|
||||
items = [SessionQueueItem.queue_item_from_dict(dict(result)) for result in results]
|
||||
has_more = False
|
||||
if len(items) > limit:
|
||||
# remove the extra item
|
||||
items.pop()
|
||||
has_more = True
|
||||
return CursorPaginatedResults(items=items, limit=limit, has_more=has_more)
|
||||
|
||||
def list_all_queue_items(
|
||||
self,
|
||||
queue_id: str,
|
||||
@@ -671,6 +619,26 @@ class SqliteSessionQueue(SessionQueueBase):
|
||||
items = [SessionQueueItem.queue_item_from_dict(dict(result)) for result in results]
|
||||
return items
|
||||
|
||||
def get_queue_item_ids(
|
||||
self,
|
||||
queue_id: str,
|
||||
order_dir: SQLiteDirection = SQLiteDirection.Descending,
|
||||
) -> ItemIdsResult:
|
||||
with self._db.transaction() as cursor_:
|
||||
query = f"""--sql
|
||||
SELECT item_id
|
||||
FROM session_queue
|
||||
WHERE queue_id = ?
|
||||
ORDER BY created_at {order_dir.value}
|
||||
"""
|
||||
query_params = [queue_id]
|
||||
|
||||
cursor_.execute(query, query_params)
|
||||
result = cast(list[sqlite3.Row], cursor_.fetchall())
|
||||
item_ids = [row[0] for row in result]
|
||||
|
||||
return ItemIdsResult(item_ids=item_ids, total_count=len(item_ids))
|
||||
|
||||
def get_queue_status(self, queue_id: str) -> SessionQueueStatus:
|
||||
with self._db.transaction() as cursor:
|
||||
cursor.execute(
|
||||
|
||||
@@ -0,0 +1,109 @@
|
||||
from typing import Optional
|
||||
|
||||
import torch
|
||||
from PIL import Image
|
||||
|
||||
# Import SAM2 components - these should be available in transformers 4.56.0+
|
||||
from transformers.models.sam2 import Sam2Model
|
||||
from transformers.models.sam2.processing_sam2 import Sam2Processor
|
||||
|
||||
from invokeai.backend.image_util.segment_anything.shared import SAMInput
|
||||
from invokeai.backend.raw_model import RawModel
|
||||
|
||||
|
||||
class SegmentAnything2Pipeline(RawModel):
|
||||
"""A wrapper class for the transformers SAM2 model and processor that makes it compatible with the model manager."""
|
||||
|
||||
def __init__(self, sam2_model: Sam2Model, sam2_processor: Sam2Processor):
|
||||
"""Initialize the SAM2 pipeline.
|
||||
|
||||
Args:
|
||||
sam2_model: The SAM2 model
|
||||
sam2_processor: The SAM2 processor (can be Sam2Processor or Sam2VideoProcessor)
|
||||
"""
|
||||
self._sam2_model = sam2_model
|
||||
self._sam2_processor = sam2_processor
|
||||
|
||||
def to(self, device: Optional[torch.device] = None, dtype: Optional[torch.dtype] = None):
|
||||
# HACK: The SAM2 pipeline may not work on MPS devices. We only allow it to be moved to CPU or CUDA.
|
||||
if device is not None and device.type not in {"cpu", "cuda"}:
|
||||
device = None
|
||||
self._sam2_model.to(device=device, dtype=dtype)
|
||||
|
||||
def calc_size(self) -> int:
|
||||
# HACK: Fix the circular import issue.
|
||||
from invokeai.backend.model_manager.load.model_util import calc_module_size
|
||||
|
||||
return calc_module_size(self._sam2_model)
|
||||
|
||||
def segment(
|
||||
self,
|
||||
image: Image.Image,
|
||||
inputs: list[SAMInput],
|
||||
) -> torch.Tensor:
|
||||
"""Segment the image using the provided inputs.
|
||||
|
||||
Args:
|
||||
image: The image to segment.
|
||||
inputs: A list of SAMInput objects containing bounding boxes and/or point lists.
|
||||
|
||||
Returns:
|
||||
torch.Tensor: The segmentation masks. dtype: torch.bool. shape: [num_masks, channels, height, width].
|
||||
"""
|
||||
|
||||
input_boxes: list[list[float]] = []
|
||||
input_points: list[list[list[float]]] = []
|
||||
input_labels: list[list[int]] = []
|
||||
|
||||
for i in inputs:
|
||||
box: list[float] | None = None
|
||||
points: list[list[float]] | None = None
|
||||
labels: list[int] | None = None
|
||||
|
||||
if i.bounding_box is not None:
|
||||
box: list[float] | None = [
|
||||
i.bounding_box.x_min,
|
||||
i.bounding_box.y_min,
|
||||
i.bounding_box.x_max,
|
||||
i.bounding_box.y_max,
|
||||
]
|
||||
|
||||
if i.points is not None:
|
||||
points = []
|
||||
labels = []
|
||||
for point in i.points:
|
||||
points.append([point.x, point.y])
|
||||
labels.append(point.label.value)
|
||||
|
||||
if box is not None:
|
||||
input_boxes.append(box)
|
||||
if points is not None:
|
||||
input_points.append(points)
|
||||
if labels is not None:
|
||||
input_labels.append(labels)
|
||||
|
||||
batched_input_boxes = [input_boxes] if input_boxes else None
|
||||
batched_input_points = [input_points] if input_points else None
|
||||
batched_input_labels = [input_labels] if input_labels else None
|
||||
|
||||
processed_inputs = self._sam2_processor(
|
||||
images=image,
|
||||
input_boxes=batched_input_boxes,
|
||||
input_points=batched_input_points,
|
||||
input_labels=batched_input_labels,
|
||||
return_tensors="pt",
|
||||
).to(self._sam2_model.device)
|
||||
|
||||
# Generate masks using the SAM2 model
|
||||
outputs = self._sam2_model(**processed_inputs)
|
||||
|
||||
# Post-process the masks to get the final segmentation
|
||||
masks = self._sam2_processor.post_process_masks(
|
||||
masks=outputs.pred_masks,
|
||||
original_sizes=processed_inputs.original_sizes,
|
||||
reshaped_input_sizes=processed_inputs.reshaped_input_sizes,
|
||||
)
|
||||
|
||||
# There should be only one batch.
|
||||
assert len(masks) == 1
|
||||
return masks[0]
|
||||
@@ -1,20 +1,13 @@
|
||||
from typing import Optional, TypeAlias
|
||||
from typing import Optional
|
||||
|
||||
import torch
|
||||
from PIL import Image
|
||||
from transformers.models.sam import SamModel
|
||||
from transformers.models.sam.processing_sam import SamProcessor
|
||||
|
||||
from invokeai.backend.image_util.segment_anything.shared import SAMInput
|
||||
from invokeai.backend.raw_model import RawModel
|
||||
|
||||
# Type aliases for the inputs to the SAM model.
|
||||
ListOfBoundingBoxes: TypeAlias = list[list[int]]
|
||||
"""A list of bounding boxes. Each bounding box is in the format [xmin, ymin, xmax, ymax]."""
|
||||
ListOfPoints: TypeAlias = list[list[int]]
|
||||
"""A list of points. Each point is in the format [x, y]."""
|
||||
ListOfPointLabels: TypeAlias = list[int]
|
||||
"""A list of SAM point labels. Each label is an integer where -1 is background, 0 is neutral, and 1 is foreground."""
|
||||
|
||||
|
||||
class SegmentAnythingPipeline(RawModel):
|
||||
"""A wrapper class for the transformers SAM model and processor that makes it compatible with the model manager."""
|
||||
@@ -38,55 +31,65 @@ class SegmentAnythingPipeline(RawModel):
|
||||
def segment(
|
||||
self,
|
||||
image: Image.Image,
|
||||
bounding_boxes: list[list[int]] | None = None,
|
||||
point_lists: list[list[list[int]]] | None = None,
|
||||
inputs: list[SAMInput],
|
||||
) -> torch.Tensor:
|
||||
"""Run the SAM model.
|
||||
|
||||
Either bounding_boxes or point_lists must be provided. If both are provided, bounding_boxes will be used and
|
||||
point_lists will be ignored.
|
||||
"""Segment the image using the provided inputs.
|
||||
|
||||
Args:
|
||||
image (Image.Image): The image to segment.
|
||||
bounding_boxes (list[list[int]]): The bounding box prompts. Each bounding box is in the format
|
||||
[xmin, ymin, xmax, ymax].
|
||||
point_lists (list[list[list[int]]]): The points prompts. Each point is in the format [x, y, label].
|
||||
`label` is an integer where -1 is background, 0 is neutral, and 1 is foreground.
|
||||
image: The image to segment.
|
||||
inputs: A list of SAMInput objects containing bounding boxes and/or point lists.
|
||||
|
||||
Returns:
|
||||
torch.Tensor: The segmentation masks. dtype: torch.bool. shape: [num_masks, channels, height, width].
|
||||
"""
|
||||
|
||||
# Prep the inputs:
|
||||
# - Create a list of bounding boxes or points and labels.
|
||||
# - Add a batch dimension of 1 to the inputs.
|
||||
if bounding_boxes:
|
||||
input_boxes: list[ListOfBoundingBoxes] | None = [bounding_boxes]
|
||||
input_points: list[ListOfPoints] | None = None
|
||||
input_labels: list[ListOfPointLabels] | None = None
|
||||
elif point_lists:
|
||||
input_boxes: list[ListOfBoundingBoxes] | None = None
|
||||
input_points: list[ListOfPoints] | None = []
|
||||
input_labels: list[ListOfPointLabels] | None = []
|
||||
for point_list in point_lists:
|
||||
input_points.append([[p[0], p[1]] for p in point_list])
|
||||
input_labels.append([p[2] for p in point_list])
|
||||
input_boxes: list[list[float]] = []
|
||||
input_points: list[list[list[float]]] = []
|
||||
input_labels: list[list[int]] = []
|
||||
|
||||
else:
|
||||
raise ValueError("Either bounding_boxes or points and labels must be provided.")
|
||||
for i in inputs:
|
||||
box: list[float] | None = None
|
||||
points: list[list[float]] | None = None
|
||||
labels: list[int] | None = None
|
||||
|
||||
inputs = self._sam_processor(
|
||||
if i.bounding_box is not None:
|
||||
box: list[float] | None = [
|
||||
i.bounding_box.x_min,
|
||||
i.bounding_box.y_min,
|
||||
i.bounding_box.x_max,
|
||||
i.bounding_box.y_max,
|
||||
]
|
||||
|
||||
if i.points is not None:
|
||||
points = []
|
||||
labels = []
|
||||
for point in i.points:
|
||||
points.append([point.x, point.y])
|
||||
labels.append(point.label.value)
|
||||
|
||||
if box is not None:
|
||||
input_boxes.append(box)
|
||||
if points is not None:
|
||||
input_points.append(points)
|
||||
if labels is not None:
|
||||
input_labels.append(labels)
|
||||
|
||||
batched_input_boxes = [input_boxes] if input_boxes else None
|
||||
batched_input_points = input_points if input_points else None
|
||||
batched_input_labels = input_labels if input_labels else None
|
||||
|
||||
processed_inputs = self._sam_processor(
|
||||
images=image,
|
||||
input_boxes=input_boxes,
|
||||
input_points=input_points,
|
||||
input_labels=input_labels,
|
||||
input_boxes=batched_input_boxes,
|
||||
input_points=batched_input_points,
|
||||
input_labels=batched_input_labels,
|
||||
return_tensors="pt",
|
||||
).to(self._sam_model.device)
|
||||
outputs = self._sam_model(**inputs)
|
||||
outputs = self._sam_model(**processed_inputs)
|
||||
masks = self._sam_processor.post_process_masks(
|
||||
masks=outputs.pred_masks,
|
||||
original_sizes=inputs.original_sizes,
|
||||
reshaped_input_sizes=inputs.reshaped_input_sizes,
|
||||
original_sizes=processed_inputs.original_sizes,
|
||||
reshaped_input_sizes=processed_inputs.reshaped_input_sizes,
|
||||
)
|
||||
|
||||
# There should be only one batch.
|
||||
|
||||
49
invokeai/backend/image_util/segment_anything/shared.py
Normal file
49
invokeai/backend/image_util/segment_anything/shared.py
Normal file
@@ -0,0 +1,49 @@
|
||||
from enum import Enum
|
||||
|
||||
from pydantic import BaseModel, model_validator
|
||||
from pydantic.fields import Field
|
||||
|
||||
|
||||
class BoundingBox(BaseModel):
|
||||
x_min: int = Field(..., description="The minimum x-coordinate of the bounding box (inclusive).")
|
||||
x_max: int = Field(..., description="The maximum x-coordinate of the bounding box (exclusive).")
|
||||
y_min: int = Field(..., description="The minimum y-coordinate of the bounding box (inclusive).")
|
||||
y_max: int = Field(..., description="The maximum y-coordinate of the bounding box (exclusive).")
|
||||
|
||||
@model_validator(mode="after")
|
||||
def check_coords(self):
|
||||
if self.x_min > self.x_max:
|
||||
raise ValueError(f"x_min ({self.x_min}) is greater than x_max ({self.x_max}).")
|
||||
if self.y_min > self.y_max:
|
||||
raise ValueError(f"y_min ({self.y_min}) is greater than y_max ({self.y_max}).")
|
||||
return self
|
||||
|
||||
def tuple(self) -> tuple[int, int, int, int]:
|
||||
"""
|
||||
Returns the bounding box as a tuple suitable for use with PIL's `Image.crop()` method.
|
||||
This method returns a tuple of the form (left, upper, right, lower) == (x_min, y_min, x_max, y_max).
|
||||
"""
|
||||
return (self.x_min, self.y_min, self.x_max, self.y_max)
|
||||
|
||||
|
||||
class SAMPointLabel(Enum):
|
||||
negative = -1
|
||||
neutral = 0
|
||||
positive = 1
|
||||
|
||||
|
||||
class SAMPoint(BaseModel):
|
||||
x: int = Field(..., description="The x-coordinate of the point")
|
||||
y: int = Field(..., description="The y-coordinate of the point")
|
||||
label: SAMPointLabel = Field(..., description="The label of the point")
|
||||
|
||||
|
||||
class SAMInput(BaseModel):
|
||||
bounding_box: BoundingBox | None = Field(None, description="The bounding box to use for segmentation")
|
||||
points: list[SAMPoint] | None = Field(None, description="The points to use for segmentation")
|
||||
|
||||
@model_validator(mode="after")
|
||||
def check_input(self):
|
||||
if not self.bounding_box and not self.points:
|
||||
raise ValueError("Either bounding_box or points must be provided")
|
||||
return self
|
||||
@@ -5,7 +5,11 @@ import torch
|
||||
from diffusers.configuration_utils import ConfigMixin, register_to_config
|
||||
from diffusers.loaders.single_file_model import FromOriginalModelMixin
|
||||
from diffusers.models.attention_processor import AttentionProcessor, AttnProcessor
|
||||
from diffusers.models.controlnet import ControlNetConditioningEmbedding, ControlNetOutput, zero_module
|
||||
from diffusers.models.controlnets.controlnet import (
|
||||
ControlNetConditioningEmbedding,
|
||||
ControlNetOutput,
|
||||
zero_module,
|
||||
)
|
||||
from diffusers.models.embeddings import (
|
||||
TextImageProjection,
|
||||
TextImageTimeEmbedding,
|
||||
@@ -775,7 +779,15 @@ class ControlNetModel(ModelMixin, ConfigMixin, FromOriginalModelMixin):
|
||||
|
||||
|
||||
diffusers.ControlNetModel = ControlNetModel
|
||||
diffusers.models.controlnet.ControlNetModel = ControlNetModel
|
||||
# Patch both the new and legacy module paths for compatibility
|
||||
try:
|
||||
diffusers.models.controlnets.controlnet.ControlNetModel = ControlNetModel
|
||||
except Exception:
|
||||
# Fallback for environments still exposing the legacy path
|
||||
try:
|
||||
diffusers.models.controlnet.ControlNetModel = ControlNetModel
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
|
||||
# patch LoRACompatibleConv to use original Conv2D forward function
|
||||
|
||||
39
invokeai/frontend/web/CLAUDE.md
Normal file
39
invokeai/frontend/web/CLAUDE.md
Normal file
@@ -0,0 +1,39 @@
|
||||
# Bash commands
|
||||
|
||||
All commands should be run from `<REPO_ROOT>/invokeai/frontend/web/`.
|
||||
|
||||
- `pnpm lint:prettier`: check formatting
|
||||
- `pnpm lint:eslint`: check for linting issues
|
||||
- `pnpm lint:knip`: check for unused dependencies
|
||||
- `pnpm lint:dpdm`: check for dependency cycles
|
||||
- `pnpm lint:tsc`: check for TypeScript issues
|
||||
- `pnpm lint`: run all checks
|
||||
- `pnpm fix`: automatically fix issues where possible
|
||||
- `pnpm test:no-watch`: run the test suite
|
||||
|
||||
# Writing Tests
|
||||
|
||||
This repo uses `vitest` for unit tests.
|
||||
|
||||
Tests should be colocated with the code they test, and should use the `.test.ts` suffix.
|
||||
|
||||
Tests do not need to be written for code that is trivial or has no logic (e.g. simple type definitions, re-exports, etc.). We currently do not do UI tests.
|
||||
|
||||
# Agents
|
||||
|
||||
- Use @agent-javascript-pro and @agent-typescript-pro for JavaScript and TypeScript code generation and assistance.
|
||||
- Use @frontend-developer for general frontend development tasks.
|
||||
|
||||
## Workflow
|
||||
|
||||
Split up tasks into smaller subtasks and handle them one at a time using an agent. Ensure each subtask is completed before moving on to the next.
|
||||
|
||||
Each agent should maintain a work log in a markdown file.
|
||||
|
||||
When an agent completes a task, it should:
|
||||
|
||||
1. Summarize the changes made.
|
||||
2. List any files that were added, modified, or deleted.
|
||||
3. Commit the changes with a descriptive commit message.
|
||||
|
||||
DO NOT PUSH ANY CHANGES TO THE REMOTE REPOSITORY.
|
||||
@@ -45,7 +45,7 @@
|
||||
"@dagrejs/dagre": "^1.1.5",
|
||||
"@dagrejs/graphlib": "^2.2.4",
|
||||
"@fontsource-variable/inter": "^5.2.6",
|
||||
"@invoke-ai/ui-library": "^0.0.46",
|
||||
"@invoke-ai/ui-library": "^0.0.47",
|
||||
"@nanostores/react": "^1.0.0",
|
||||
"@observ33r/object-equals": "^1.1.5",
|
||||
"@reduxjs/toolkit": "2.8.2",
|
||||
|
||||
116
invokeai/frontend/web/pnpm-lock.yaml
generated
116
invokeai/frontend/web/pnpm-lock.yaml
generated
@@ -27,8 +27,8 @@ importers:
|
||||
specifier: ^5.2.6
|
||||
version: 5.2.6
|
||||
'@invoke-ai/ui-library':
|
||||
specifier: ^0.0.46
|
||||
version: 0.0.46(@chakra-ui/system@2.6.2(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1))(react@18.3.1))(@fontsource-variable/inter@5.2.6)(@types/react@18.3.23)(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3)
|
||||
specifier: ^0.0.47
|
||||
version: 0.0.47(@chakra-ui/system@2.6.2(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1))(react@18.3.1))(@fontsource-variable/inter@5.2.6)(@types/react@18.3.23)(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3)
|
||||
'@nanostores/react':
|
||||
specifier: ^1.0.0
|
||||
version: 1.0.0(nanostores@1.0.1)(react@18.3.1)
|
||||
@@ -887,8 +887,8 @@ packages:
|
||||
resolution: {integrity: sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==}
|
||||
engines: {node: '>=18.18'}
|
||||
|
||||
'@invoke-ai/ui-library@0.0.46':
|
||||
resolution: {integrity: sha512-3YBuWWhRbTUHi0RZKeyvDEvweoyZmeBdUGJIhemjdAgGx6l98rAMeCs8IQH+SYjSAIhiGRGf45fQ33PDK8Jkmw==}
|
||||
'@invoke-ai/ui-library@0.0.47':
|
||||
resolution: {integrity: sha512-zmO2bAkkqT2yhkHjrsDnYio3YNKYyBSJXDZFmTSxWdK58UM2+Zq3h7cpVbDgS7Dzo4RXdF7p+DdlYPm2iIey5A==}
|
||||
peerDependencies:
|
||||
'@fontsource-variable/inter': ^5.0.16
|
||||
react: ^18.2.0
|
||||
@@ -968,13 +968,6 @@ packages:
|
||||
'@mux/playback-core@0.30.1':
|
||||
resolution: {integrity: sha512-rnO1NE9xHDyzbAkmE6ygJYcD7cyyMt7xXqWTykxlceaoSXLjUqgp42HDio7Lcidto4x/O4FIa7ztjV2aCBCXgQ==}
|
||||
|
||||
'@nanostores/react@0.7.3':
|
||||
resolution: {integrity: sha512-/XuLAMENRu/Q71biW4AZ4qmU070vkZgiQ28gaTSNRPm2SZF5zGAR81zPE1MaMB4SeOp6ZTst92NBaG75XSspNg==}
|
||||
engines: {node: ^18.0.0 || >=20.0.0}
|
||||
peerDependencies:
|
||||
nanostores: ^0.9.0 || ^0.10.0 || ^0.11.0
|
||||
react: '>=18.0.0'
|
||||
|
||||
'@nanostores/react@1.0.0':
|
||||
resolution: {integrity: sha512-eDduyNy+lbQJMg6XxZ/YssQqF6b4OXMFEZMYKPJCCmBevp1lg0g+4ZRi94qGHirMtsNfAWKNwsjOhC+q1gvC+A==}
|
||||
engines: {node: ^20.0.0 || >=22.0.0}
|
||||
@@ -2423,6 +2416,9 @@ packages:
|
||||
resolution: {integrity: sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g==}
|
||||
engines: {node: '>= 0.4'}
|
||||
|
||||
es-toolkit@1.39.10:
|
||||
resolution: {integrity: sha512-E0iGnTtbDhkeczB0T+mxmoVlT4YNweEKBLq7oaU4p11mecdsZpNWOglI4895Vh4usbQ+LsJiuLuI2L0Vdmfm2w==}
|
||||
|
||||
es-toolkit@1.39.7:
|
||||
resolution: {integrity: sha512-ek/wWryKouBrZIjkwW2BFf91CWOIMvoy2AE5YYgUrfWsJQM2Su1LoLtrw8uusEpN9RfqLlV/0FVNjT0WMv8Bxw==}
|
||||
|
||||
@@ -3198,9 +3194,6 @@ packages:
|
||||
resolution: {integrity: sha512-gvVijfZvn7R+2qyPX8mAuKcFGDf6Nc61GdvGafQsHL0sBIxfKzA+usWn4GFC/bk+QdwPUD4kWFJLhElipq+0VA==}
|
||||
engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0}
|
||||
|
||||
lodash-es@4.17.21:
|
||||
resolution: {integrity: sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw==}
|
||||
|
||||
lodash.merge@4.6.2:
|
||||
resolution: {integrity: sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==}
|
||||
|
||||
@@ -3249,6 +3242,9 @@ packages:
|
||||
resolution: {integrity: sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==}
|
||||
engines: {node: '>=10'}
|
||||
|
||||
math-expression-evaluator@2.0.7:
|
||||
resolution: {integrity: sha512-uwliJZ6BPHRq4eiqNWxZBDzKUiS5RIynFFcgchqhBOloVLVBpZpNG8jRYkedLcBvhph8TnRyWEuxPqiQcwIdog==}
|
||||
|
||||
math-intrinsics@1.1.0:
|
||||
resolution: {integrity: sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==}
|
||||
engines: {node: '>= 0.4'}
|
||||
@@ -3352,10 +3348,6 @@ packages:
|
||||
engines: {node: ^18 || >=20}
|
||||
hasBin: true
|
||||
|
||||
nanostores@0.11.4:
|
||||
resolution: {integrity: sha512-k1oiVNN4hDK8NcNERSZLQiMfRzEGtfnvZvdBvey3SQbgn8Dcrk0h1I6vpxApjb10PFUflZrgJ2WEZyJQ+5v7YQ==}
|
||||
engines: {node: ^18.0.0 || >=20.0.0}
|
||||
|
||||
nanostores@1.0.1:
|
||||
resolution: {integrity: sha512-kNZ9xnoJYKg/AfxjrVL4SS0fKX++4awQReGqWnwTRHxeHGZ1FJFVgTqr/eMrNQdp0Tz7M7tG/TDaX8QfHDwVCw==}
|
||||
engines: {node: ^20.0.0 || >=22.0.0}
|
||||
@@ -3449,12 +3441,12 @@ packages:
|
||||
overlayscrollbars: ^2.0.0
|
||||
react: '>=16.8.0'
|
||||
|
||||
overlayscrollbars@2.10.0:
|
||||
resolution: {integrity: sha512-diNMeEafWTE0A4GJfwRpdBp2rE/BEvrhptBdBcDu8/UeytWcdCy9Td8tZWnztJeJ26f8/uHCWfPnPUC/dtgJdw==}
|
||||
|
||||
overlayscrollbars@2.11.4:
|
||||
resolution: {integrity: sha512-GKYQo3OZ1QWnppNjQVv5hfpn+glYUxc6+ufW+ivdXUyLWFNc01XoH2Z36KGM4I8e5pXYeA3ElNItcXiLvmUhnQ==}
|
||||
|
||||
overlayscrollbars@2.12.0:
|
||||
resolution: {integrity: sha512-mWJ5MOkcZ/ljHwfLw8+bN0V9ziGCoNoqULcp994j5DTGNQvnkWKWkA7rnO29Kyew5AoHxUnJ4Ndqfcl0HSQjXg==}
|
||||
|
||||
own-keys@1.0.1:
|
||||
resolution: {integrity: sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg==}
|
||||
engines: {node: '>= 0.4'}
|
||||
@@ -3687,6 +3679,22 @@ packages:
|
||||
typescript:
|
||||
optional: true
|
||||
|
||||
react-i18next@15.7.3:
|
||||
resolution: {integrity: sha512-AANws4tOE+QSq/IeMF/ncoHlMNZaVLxpa5uUGW1wjike68elVYr0018L9xYoqBr1OFO7G7boDPrbn0HpMCJxTw==}
|
||||
peerDependencies:
|
||||
i18next: '>= 25.4.1'
|
||||
react: '>= 16.8.0'
|
||||
react-dom: '*'
|
||||
react-native: '*'
|
||||
typescript: ^5
|
||||
peerDependenciesMeta:
|
||||
react-dom:
|
||||
optional: true
|
||||
react-native:
|
||||
optional: true
|
||||
typescript:
|
||||
optional: true
|
||||
|
||||
react-icons@5.5.0:
|
||||
resolution: {integrity: sha512-MEFcXdkP3dLo8uumGI5xN3lDFNsRtrjbOEKDLD7yv76v4wpnEq2Lt2qeHaQOr34I/wPN3s3+N08WkQ+CW37Xiw==}
|
||||
peerDependencies:
|
||||
@@ -3743,8 +3751,8 @@ packages:
|
||||
react: ^16.14.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc
|
||||
react-dom: ^16.14.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc
|
||||
|
||||
react-select@5.10.1:
|
||||
resolution: {integrity: sha512-roPEZUL4aRZDx6DcsD+ZNreVl+fM8VsKn0Wtex1v4IazH60ILp5xhdlp464IsEAlJdXeD+BhDAFsBVMfvLQueA==}
|
||||
react-select@5.10.2:
|
||||
resolution: {integrity: sha512-Z33nHdEFWq9tfnfVXaiM12rbJmk+QjFEztWLtmXqQhz6Al4UZZ9xc0wiatmGtUOCCnHN0WizL3tCMYRENX4rVQ==}
|
||||
peerDependencies:
|
||||
react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0
|
||||
react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0
|
||||
@@ -5119,7 +5127,7 @@ snapshots:
|
||||
|
||||
'@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1)':
|
||||
dependencies:
|
||||
'@babel/runtime': 7.27.6
|
||||
'@babel/runtime': 7.28.3
|
||||
'@emotion/babel-plugin': 11.13.5
|
||||
'@emotion/is-prop-valid': 1.3.1
|
||||
'@emotion/react': 11.14.0(@types/react@18.3.23)(react@18.3.1)
|
||||
@@ -5290,7 +5298,7 @@ snapshots:
|
||||
|
||||
'@humanwhocodes/retry@0.4.3': {}
|
||||
|
||||
'@invoke-ai/ui-library@0.0.46(@chakra-ui/system@2.6.2(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1))(react@18.3.1))(@fontsource-variable/inter@5.2.6)(@types/react@18.3.23)(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3)':
|
||||
'@invoke-ai/ui-library@0.0.47(@chakra-ui/system@2.6.2(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1))(react@18.3.1))(@fontsource-variable/inter@5.2.6)(@types/react@18.3.23)(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3)':
|
||||
dependencies:
|
||||
'@chakra-ui/anatomy': 2.3.4
|
||||
'@chakra-ui/icons': 2.2.4(@chakra-ui/react@2.10.9(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(framer-motion@10.18.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react@18.3.1)
|
||||
@@ -5302,18 +5310,19 @@ snapshots:
|
||||
'@emotion/react': 11.14.0(@types/react@18.3.23)(react@18.3.1)
|
||||
'@emotion/styled': 11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1)
|
||||
'@fontsource-variable/inter': 5.2.6
|
||||
'@nanostores/react': 0.7.3(nanostores@0.11.4)(react@18.3.1)
|
||||
'@nanostores/react': 1.0.0(nanostores@1.0.1)(react@18.3.1)
|
||||
chakra-react-select: 4.10.1(@chakra-ui/react@2.10.9(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@emotion/styled@11.14.1(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(framer-motion@10.18.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(react-dom@18.3.1(react@18.3.1))(react@18.3.1))(@emotion/react@11.14.0(@types/react@18.3.23)(react@18.3.1))(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
||||
es-toolkit: 1.39.10
|
||||
framer-motion: 10.18.0(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
||||
lodash-es: 4.17.21
|
||||
nanostores: 0.11.4
|
||||
overlayscrollbars: 2.10.0
|
||||
overlayscrollbars-react: 0.5.6(overlayscrollbars@2.10.0)(react@18.3.1)
|
||||
math-expression-evaluator: 2.0.7
|
||||
nanostores: 1.0.1
|
||||
overlayscrollbars: 2.12.0
|
||||
overlayscrollbars-react: 0.5.6(overlayscrollbars@2.12.0)(react@18.3.1)
|
||||
react: 18.3.1
|
||||
react-dom: 18.3.1(react@18.3.1)
|
||||
react-i18next: 15.6.0(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3)
|
||||
react-i18next: 15.7.3(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3)
|
||||
react-icons: 5.5.0(react@18.3.1)
|
||||
react-select: 5.10.1(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
||||
react-select: 5.10.2(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
|
||||
transitivePeerDependencies:
|
||||
- '@chakra-ui/system'
|
||||
- '@types/react'
|
||||
@@ -5434,11 +5443,6 @@ snapshots:
|
||||
hls.js: 1.6.9
|
||||
mux-embed: 5.11.0
|
||||
|
||||
'@nanostores/react@0.7.3(nanostores@0.11.4)(react@18.3.1)':
|
||||
dependencies:
|
||||
nanostores: 0.11.4
|
||||
react: 18.3.1
|
||||
|
||||
'@nanostores/react@1.0.0(nanostores@1.0.1)(react@18.3.1)':
|
||||
dependencies:
|
||||
nanostores: 1.0.1
|
||||
@@ -7038,6 +7042,8 @@ snapshots:
|
||||
is-date-object: 1.1.0
|
||||
is-symbol: 1.1.1
|
||||
|
||||
es-toolkit@1.39.10: {}
|
||||
|
||||
es-toolkit@1.39.7: {}
|
||||
|
||||
esbuild-register@3.6.0(esbuild@0.25.6):
|
||||
@@ -7869,8 +7875,6 @@ snapshots:
|
||||
dependencies:
|
||||
p-locate: 6.0.0
|
||||
|
||||
lodash-es@4.17.21: {}
|
||||
|
||||
lodash.merge@4.6.2: {}
|
||||
|
||||
lodash.mergewith@4.6.2: {}
|
||||
@@ -7916,6 +7920,8 @@ snapshots:
|
||||
dependencies:
|
||||
semver: 7.7.2
|
||||
|
||||
math-expression-evaluator@2.0.7: {}
|
||||
|
||||
math-intrinsics@1.1.0: {}
|
||||
|
||||
mdn-data@2.0.14: {}
|
||||
@@ -8012,8 +8018,6 @@ snapshots:
|
||||
|
||||
nanoid@5.1.5: {}
|
||||
|
||||
nanostores@0.11.4: {}
|
||||
|
||||
nanostores@1.0.1: {}
|
||||
|
||||
native-promise-only@0.8.1: {}
|
||||
@@ -8120,20 +8124,20 @@ snapshots:
|
||||
strip-ansi: 6.0.1
|
||||
wcwidth: 1.0.1
|
||||
|
||||
overlayscrollbars-react@0.5.6(overlayscrollbars@2.10.0)(react@18.3.1):
|
||||
dependencies:
|
||||
overlayscrollbars: 2.10.0
|
||||
react: 18.3.1
|
||||
|
||||
overlayscrollbars-react@0.5.6(overlayscrollbars@2.11.4)(react@18.3.1):
|
||||
dependencies:
|
||||
overlayscrollbars: 2.11.4
|
||||
react: 18.3.1
|
||||
|
||||
overlayscrollbars@2.10.0: {}
|
||||
overlayscrollbars-react@0.5.6(overlayscrollbars@2.12.0)(react@18.3.1):
|
||||
dependencies:
|
||||
overlayscrollbars: 2.12.0
|
||||
react: 18.3.1
|
||||
|
||||
overlayscrollbars@2.11.4: {}
|
||||
|
||||
overlayscrollbars@2.12.0: {}
|
||||
|
||||
own-keys@1.0.1:
|
||||
dependencies:
|
||||
get-intrinsic: 1.3.0
|
||||
@@ -8293,7 +8297,7 @@ snapshots:
|
||||
|
||||
react-clientside-effect@1.2.8(react@18.3.1):
|
||||
dependencies:
|
||||
'@babel/runtime': 7.27.6
|
||||
'@babel/runtime': 7.28.3
|
||||
react: 18.3.1
|
||||
|
||||
react-colorful@5.6.1(react-dom@18.3.1(react@18.3.1))(react@18.3.1):
|
||||
@@ -8342,7 +8346,7 @@ snapshots:
|
||||
|
||||
react-focus-lock@2.13.6(@types/react@18.3.23)(react@18.3.1):
|
||||
dependencies:
|
||||
'@babel/runtime': 7.27.6
|
||||
'@babel/runtime': 7.28.3
|
||||
focus-lock: 1.3.6
|
||||
prop-types: 15.8.1
|
||||
react: 18.3.1
|
||||
@@ -8371,6 +8375,16 @@ snapshots:
|
||||
react-dom: 18.3.1(react@18.3.1)
|
||||
typescript: 5.8.3
|
||||
|
||||
react-i18next@15.7.3(i18next@25.3.2(typescript@5.8.3))(react-dom@18.3.1(react@18.3.1))(react@18.3.1)(typescript@5.8.3):
|
||||
dependencies:
|
||||
'@babel/runtime': 7.28.3
|
||||
html-parse-stringify: 3.0.1
|
||||
i18next: 25.3.2(typescript@5.8.3)
|
||||
react: 18.3.1
|
||||
optionalDependencies:
|
||||
react-dom: 18.3.1(react@18.3.1)
|
||||
typescript: 5.8.3
|
||||
|
||||
react-icons@5.5.0(react@18.3.1):
|
||||
dependencies:
|
||||
react: 18.3.1
|
||||
@@ -8430,9 +8444,9 @@ snapshots:
|
||||
react: 18.3.1
|
||||
react-dom: 18.3.1(react@18.3.1)
|
||||
|
||||
react-select@5.10.1(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1):
|
||||
react-select@5.10.2(@types/react@18.3.23)(react-dom@18.3.1(react@18.3.1))(react@18.3.1):
|
||||
dependencies:
|
||||
'@babel/runtime': 7.27.6
|
||||
'@babel/runtime': 7.28.3
|
||||
'@emotion/cache': 11.14.0
|
||||
'@emotion/react': 11.14.0(@types/react@18.3.23)(react@18.3.1)
|
||||
'@floating-ui/dom': 1.7.2
|
||||
|
||||
@@ -14,8 +14,7 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "حجم الصورة",
|
||||
"gallerySettings": "إعدادات المعرض",
|
||||
"autoSwitchNewImages": "التبديل التلقائي إلى الصور الجديدة",
|
||||
"noImagesInGallery": "لا توجد صور في المعرض"
|
||||
"autoSwitchNewImages": "التبديل التلقائي إلى الصور الجديدة"
|
||||
},
|
||||
"modelManager": {
|
||||
"modelManager": "مدير النموذج",
|
||||
@@ -62,12 +61,10 @@
|
||||
"infillMethod": "طريقة التعبئة",
|
||||
"tileSize": "حجم البلاطة",
|
||||
"copyImage": "نسخ الصورة",
|
||||
"downloadImage": "تحميل الصورة",
|
||||
"usePrompt": "استخدم المحث",
|
||||
"useSeed": "استخدام البذور",
|
||||
"useAll": "استخدام الكل",
|
||||
"info": "معلومات",
|
||||
"showOptionsPanel": "إظهار لوحة الخيارات"
|
||||
"info": "معلومات"
|
||||
},
|
||||
"settings": {
|
||||
"models": "موديلات",
|
||||
|
||||
@@ -24,7 +24,6 @@
|
||||
"ipAdapter": "IP Adapter",
|
||||
"auto": "Auto",
|
||||
"controlNet": "ControlNet",
|
||||
"imageFailedToLoad": "Kann Bild nicht laden",
|
||||
"modelManager": "Model Manager",
|
||||
"learnMore": "Mehr erfahren",
|
||||
"loading": "Lade",
|
||||
@@ -52,7 +51,6 @@
|
||||
"somethingWentWrong": "Etwas ist schief gelaufen",
|
||||
"copyError": "$t(gallery.copy) Fehler",
|
||||
"input": "Eingabe",
|
||||
"notInstalled": "Nicht $t(common.installed)",
|
||||
"alpha": "Alpha",
|
||||
"red": "Rot",
|
||||
"green": "Grün",
|
||||
@@ -62,11 +60,8 @@
|
||||
"direction": "Richtung",
|
||||
"save": "Speichern",
|
||||
"created": "Erstellt",
|
||||
"prevPage": "Vorherige Seite",
|
||||
"nextPage": "Nächste Seite",
|
||||
"unknownError": "Unbekannter Fehler",
|
||||
"aboutDesc": "Verwenden Sie Invoke für die Arbeit? Siehe hier:",
|
||||
"localSystem": "Lokales System",
|
||||
"orderBy": "Ordnen nach",
|
||||
"saveAs": "Speichern als",
|
||||
"updated": "Aktualisiert",
|
||||
@@ -77,7 +72,6 @@
|
||||
"selected": "Ausgewählt",
|
||||
"beta": "Beta",
|
||||
"editor": "Editor",
|
||||
"goTo": "Gehe zu",
|
||||
"positivePrompt": "Positiv-Prompt",
|
||||
"negativePrompt": "Negativ-Prompt",
|
||||
"tab": "Tabulator",
|
||||
@@ -106,7 +100,6 @@
|
||||
"values": "Werte",
|
||||
"min": "Min",
|
||||
"max": "Max",
|
||||
"resetToDefaults": "Auf Standard zurücksetzen",
|
||||
"seed": "Seed",
|
||||
"row": "Reihe",
|
||||
"column": "Spalte",
|
||||
@@ -135,14 +128,12 @@
|
||||
"galleryImageSize": "Bildgröße",
|
||||
"gallerySettings": "Galerie-Einstellungen",
|
||||
"autoSwitchNewImages": "Auto-Wechsel zu neuen Bildern",
|
||||
"noImagesInGallery": "Keine Bilder in der Galerie",
|
||||
"loading": "Lade",
|
||||
"deleteImage_one": "Lösche Bild",
|
||||
"deleteImage_other": "Lösche {{count}} Bilder",
|
||||
"copy": "Kopieren",
|
||||
"download": "Runterladen",
|
||||
"featuresWillReset": "Wenn Sie dieses Bild löschen, werden diese Funktionen sofort zurückgesetzt.",
|
||||
"unableToLoad": "Galerie kann nicht geladen werden",
|
||||
"downloadSelection": "Auswahl herunterladen",
|
||||
"currentlyInUse": "Dieses Bild wird derzeit in den folgenden Funktionen verwendet:",
|
||||
"deleteImagePermanent": "Gelöschte Bilder können nicht wiederhergestellt werden.",
|
||||
@@ -182,16 +173,12 @@
|
||||
"gallery": "Galerie",
|
||||
"sortDirection": "Sortierreihenfolge",
|
||||
"sideBySide": "Nebeneinander",
|
||||
"openViewer": "Viewer öffnen",
|
||||
"viewerImage": "Viewer-Bild",
|
||||
"exitCompare": "Vergleichen beenden",
|
||||
"closeViewer": "Viewer schließen",
|
||||
"selectAnImageToCompare": "Wählen Sie ein Bild zum Vergleichen",
|
||||
"stretchToFit": "Strecken bis es passt",
|
||||
"displayBoardSearch": "Board durchsuchen",
|
||||
"displaySearch": "Bild suchen",
|
||||
"go": "Los",
|
||||
"jump": "Springen",
|
||||
"assetsTab": "Dateien, die Sie zur Verwendung in Ihren Projekten hochgeladen haben.",
|
||||
"imagesTab": "Bilder, die Sie in Invoke erstellt und gespeichert haben.",
|
||||
"boardsSettings": "Ordnereinstellungen",
|
||||
@@ -574,7 +561,6 @@
|
||||
"urlOrLocalPath": "URL oder lokaler Pfad",
|
||||
"install": "Installieren",
|
||||
"textualInversions": "Textuelle Inversionen",
|
||||
"ipAdapters": "IP-Adapter",
|
||||
"modelImageUpdated": "Modellbild aktualisiert",
|
||||
"path": "Pfad",
|
||||
"pathToConfig": "Pfad zur Konfiguration",
|
||||
@@ -597,7 +583,6 @@
|
||||
"repoVariant": "Repo Variante",
|
||||
"learnMoreAboutSupportedModels": "Erfahren Sie mehr über die Modelle, die wir unterstützen",
|
||||
"clipEmbed": "CLIP einbetten",
|
||||
"starterModelsInModelManager": "Modelle für Ihren Start finden Sie im Modell-Manager",
|
||||
"noModelsInstalledDesc1": "Installiere Modelle mit dem",
|
||||
"modelImageUpdateFailed": "Modellbild-Update fehlgeschlagen",
|
||||
"prune": "Bereinigen",
|
||||
@@ -657,11 +642,9 @@
|
||||
"scaledHeight": "Skaliert H",
|
||||
"infillMethod": "Infill-Methode",
|
||||
"tileSize": "Kachelgröße",
|
||||
"downloadImage": "Bild herunterladen",
|
||||
"usePrompt": "Prompt verwenden",
|
||||
"useSeed": "Seed verwenden",
|
||||
"useAll": "Alle verwenden",
|
||||
"showOptionsPanel": "Optionsleiste zeigen",
|
||||
"copyImage": "Bild kopieren",
|
||||
"denoisingStrength": "Stärke der Entrauschung",
|
||||
"symmetry": "Symmetrie",
|
||||
@@ -677,10 +660,6 @@
|
||||
"remixImage": "Remix des Bilds erstellen",
|
||||
"imageActions": "Weitere Bildaktionen",
|
||||
"invoke": {
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), Skalierte Bbox-Breite ist {{width}}",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), Skalierte Bbox-Höhe ist {{height}}",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), Bbox-Breite ist {{width}}",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), Bbox-Höhe ist {{height}}",
|
||||
"noNodesInGraph": "Keine Knoten im Graphen",
|
||||
"canvasIsTransforming": "Leinwand ist beschäftigt (wird transformiert)",
|
||||
"canvasIsRasterizing": "Leinwand ist beschäftigt (wird gerastert)",
|
||||
@@ -746,7 +725,6 @@
|
||||
"parametersNotSet": "Parameter nicht zurückgerufen",
|
||||
"addedToBoard": "Dem Board hinzugefügt",
|
||||
"loadedWithWarnings": "Workflow mit Warnungen geladen",
|
||||
"imageSaved": "Bild gespeichert",
|
||||
"linkCopied": "Link kopiert",
|
||||
"problemCopyingLayer": "Ebene kann nicht kopiert werden",
|
||||
"problemSavingLayer": "Ebene kann nicht gespeichert werden",
|
||||
@@ -757,8 +735,6 @@
|
||||
"prunedQueue": "Warteschlange bereinigt",
|
||||
"modelAddedSimple": "Modell zur Warteschlange hinzugefügt",
|
||||
"parametersSet": "Parameter zurückgerufen",
|
||||
"imageNotLoadedDesc": "Bild konnte nicht gefunden werden",
|
||||
"setControlImage": "Als Kontrollbild festlegen",
|
||||
"sentToUpscale": "An Vergrößerung gesendet",
|
||||
"parameterNotSetDescWithMessage": "{{parameter}} kann nicht zurückgerufen werden: {{message}}",
|
||||
"unableToLoadImageMetadata": "Bildmetadaten können nicht geladen werden",
|
||||
@@ -771,7 +747,6 @@
|
||||
"parameterSet": "Parameter zurückgerufen",
|
||||
"importFailed": "Import fehlgeschlagen",
|
||||
"importSuccessful": "Import erfolgreich",
|
||||
"setNodeField": "Als Knotenfeld festlegen",
|
||||
"somethingWentWrong": "Etwas ist schief gelaufen",
|
||||
"workflowLoaded": "Arbeitsablauf geladen",
|
||||
"workflowDeleted": "Arbeitsablauf gelöscht",
|
||||
@@ -779,16 +754,12 @@
|
||||
"layerCopiedToClipboard": "Ebene in die Zwischenablage kopiert",
|
||||
"sentToCanvas": "An Leinwand gesendet",
|
||||
"problemDeletingWorkflow": "Problem beim Löschen des Arbeitsablaufs",
|
||||
"uploadFailedInvalidUploadDesc_withCount_one": "Darf maximal 1 PNG-, JPEG- oder WEBP-Bild sein.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "Dürfen maximal {{count}} PNG-, JPEG- oder WEBP-Bild sein.",
|
||||
"problemRetrievingWorkflow": "Problem beim Abrufen des Arbeitsablaufs",
|
||||
"uploadFailedInvalidUploadDesc": "Müssen PNG-, JPEG- oder WEBP-Bilder sein.",
|
||||
"pasteSuccess": "Eingefügt in {{destination}}",
|
||||
"pasteFailed": "Einfügen fehlgeschlagen",
|
||||
"unableToCopy": "Kopieren nicht möglich",
|
||||
"unableToCopyDesc_theseSteps": "diese Schritte",
|
||||
"noRasterLayers": "Keine Rasterebenen gefunden",
|
||||
"noActiveRasterLayers": "Keine aktiven Rasterebenen",
|
||||
"noVisibleRasterLayers": "Keine sichtbaren Rasterebenen"
|
||||
},
|
||||
"accessibility": {
|
||||
@@ -841,16 +812,13 @@
|
||||
"archiveBoard": "Ordner archivieren",
|
||||
"archived": "Archiviert",
|
||||
"noBoards": "Kein {{boardType}} Ordner",
|
||||
"hideBoards": "Ordner verstecken",
|
||||
"viewBoards": "Ordner ansehen",
|
||||
"deletedPrivateBoardsCannotbeRestored": "Gelöschte Boards können nicht wiederhergestellt werden. Wenn Sie „Nur Board löschen“ wählen, werden die Bilder in einen privaten, nicht kategorisierten Status für den Ersteller des Bildes versetzt.",
|
||||
"assetsWithCount_one": "{{count}} in der Sammlung",
|
||||
"assetsWithCount_other": "{{count}} in der Sammlung",
|
||||
"deletedBoardsCannotbeRestored": "Gelöschte Ordner können nicht wiederhergestellt werden. Die Auswahl von \"Nur Ordner löschen\" verschiebt Bilder in einen unkategorisierten Zustand.",
|
||||
"updateBoardError": "Fehler beim Aktualisieren des Ordners",
|
||||
"uncategorizedImages": "Nicht kategorisierte Bilder",
|
||||
"deleteAllUncategorizedImages": "Alle nicht kategorisierten Bilder löschen",
|
||||
"deletedImagesCannotBeRestored": "Gelöschte Bilder können nicht wiederhergestellt werden."
|
||||
"deleteAllUncategorizedImages": "Alle nicht kategorisierten Bilder löschen"
|
||||
},
|
||||
"queue": {
|
||||
"status": "Status",
|
||||
@@ -905,7 +873,6 @@
|
||||
"batchQueuedDesc_other": "{{count}} Einträge an {{direction}} der Wartschlange hinzugefügt",
|
||||
"openQueue": "Warteschlange öffnen",
|
||||
"batchFailedToQueue": "Fehler beim Einreihen in die Stapelverarbeitung",
|
||||
"batchFieldValues": "Stapelverarbeitungswerte",
|
||||
"batchQueued": "Stapelverarbeitung eingereiht",
|
||||
"graphQueued": "Graph eingereiht",
|
||||
"graphFailedToQueue": "Fehler beim Einreihen des Graphen",
|
||||
@@ -952,8 +919,6 @@
|
||||
"allPrompts": "Alle Prompts",
|
||||
"imageDimensions": "Bilder Auslösungen",
|
||||
"parameterSet": "Parameter {{parameter}} setzen",
|
||||
"recallParameter": "{{label}} Abrufen",
|
||||
"parsingFailed": "Parsing Fehlgeschlagen",
|
||||
"canvasV2Metadata": "Leinwand",
|
||||
"guidance": "Führung",
|
||||
"seamlessXAxis": "Nahtlose X Achse",
|
||||
@@ -1236,9 +1201,7 @@
|
||||
"collectionFieldType": "{{name}} (Sammlung)",
|
||||
"connectionWouldCreateCycle": "Verbindung würde einen Kreislauf/cycle schaffen",
|
||||
"inputMayOnlyHaveOneConnection": "Eingang darf nur eine Verbindung haben",
|
||||
"hideLegendNodes": "Feldtyp-Legende ausblenden",
|
||||
"integer": "Ganze Zahl",
|
||||
"addLinearView": "Zur linearen Ansicht hinzufügen",
|
||||
"currentImageDescription": "Zeigt das aktuelle Bild im Node-Editor an",
|
||||
"ipAdapter": "IP-Adapter",
|
||||
"hideMinimapnodes": "Miniatur-Kartenansicht ausblenden",
|
||||
@@ -1247,7 +1210,6 @@
|
||||
"reloadNodeTemplates": "Knoten-Vorlagen neu laden",
|
||||
"newWorkflow": "Neuer Arbeitsablauf / Workflow",
|
||||
"newWorkflowDesc": "Einen neuen Arbeitsablauf erstellen?",
|
||||
"noFieldsLinearview": "Keine Felder zur linearen Ansicht hinzugefügt",
|
||||
"clearWorkflow": "Workflow löschen",
|
||||
"clearWorkflowDesc": "Diesen Arbeitsablauf löschen und neu starten?",
|
||||
"noConnectionInProgress": "Es besteht keine Verbindung",
|
||||
@@ -1255,7 +1217,6 @@
|
||||
"nodeVersion": "Knoten Version",
|
||||
"node": "Knoten",
|
||||
"nodeSearch": "Knoten suchen",
|
||||
"removeLinearView": "Entfernen aus Linear View",
|
||||
"nodeOutputs": "Knoten-Ausgänge",
|
||||
"nodeTemplate": "Knoten-Vorlage",
|
||||
"nodeType": "Knotentyp",
|
||||
@@ -1266,7 +1227,6 @@
|
||||
"clearWorkflowDesc2": "Ihr aktueller Arbeitsablauf hat ungespeicherte Änderungen.",
|
||||
"scheduler": "Planer",
|
||||
"showMinimapnodes": "MiniMap anzeigen",
|
||||
"showLegendNodes": "Feldtyp-Legende anzeigen",
|
||||
"executionStateCompleted": "Erledigt",
|
||||
"downloadWorkflow": "Workflow JSON herunterladen",
|
||||
"executionStateInProgress": "In Bearbeitung",
|
||||
@@ -1276,7 +1236,6 @@
|
||||
"fieldTypesMustMatch": "Feldtypen müssen übereinstimmen",
|
||||
"fitViewportNodes": "An Ansichtsgröße anpassen",
|
||||
"loadingNodes": "Lade Nodes...",
|
||||
"mismatchedVersion": "Ungültiger Knoten: Knoten {{node}} vom Typ {{type}} hat keine passende Version (Update versuchen?)",
|
||||
"fullyContainNodesHelp": "Nodes müssen vollständig innerhalb der Auswahlbox sein, um ausgewählt werden zu können",
|
||||
"noWorkflow": "Kein Workflow",
|
||||
"executionStateError": "Fehler",
|
||||
@@ -1288,9 +1247,7 @@
|
||||
"sourceNodeDoesNotExist": "Ungültiger Rand: Quell- / Ausgabe-Knoten {{node}} existiert nicht",
|
||||
"updateAllNodes": "Update Knoten",
|
||||
"allNodesUpdated": "Alle Knoten aktualisiert",
|
||||
"unknownTemplate": "Unbekannte Vorlage",
|
||||
"updateApp": "Update App",
|
||||
"unknownInput": "Unbekannte Eingabe: {{name}}",
|
||||
"unknownNodeType": "Unbekannter Knotentyp",
|
||||
"float": "Kommazahlen",
|
||||
"enum": "Aufzählung",
|
||||
@@ -1306,7 +1263,6 @@
|
||||
"workflowAuthor": "Autor",
|
||||
"graph": "Graph",
|
||||
"workflowDescription": "Kurze Beschreibung",
|
||||
"versionUnknown": " Version unbekannt",
|
||||
"workflow": "Arbeitsablauf",
|
||||
"noGraph": "Kein Graph",
|
||||
"version": "Version",
|
||||
@@ -1324,7 +1280,6 @@
|
||||
"unknownErrorValidatingWorkflow": "Unbekannter Fehler beim Validieren des Arbeitsablaufes",
|
||||
"inputFieldTypeParseError": "Typ des Eingabefelds {{node}}.{{field}} kann nicht analysiert werden ({{message}})",
|
||||
"workflowSettings": "Arbeitsablauf Editor Einstellungen",
|
||||
"unableToLoadWorkflow": "Arbeitsablauf kann nicht geladen werden",
|
||||
"viewMode": "In linearen Ansicht verwenden",
|
||||
"unableToValidateWorkflow": "Arbeitsablauf kann nicht validiert werden",
|
||||
"outputFieldTypeParseError": "Typ des Ausgabefelds {{node}}.{{field}} kann nicht analysiert werden ({{message}})",
|
||||
@@ -1340,7 +1295,6 @@
|
||||
"arithmeticSequence": "Arithmetische Folge",
|
||||
"noBatchGroup": "keine Gruppe",
|
||||
"generatorNoValues": "leer",
|
||||
"generatorLoading": "wird geladen",
|
||||
"generatorLoadFromFile": "Aus Datei laden",
|
||||
"showEdgeLabels": "Kantenbeschriftungen anzeigen",
|
||||
"downloadWorkflowError": "Fehler beim Herunterladen des Arbeitsablaufs",
|
||||
@@ -1348,14 +1302,11 @@
|
||||
"description": "Beschreibung",
|
||||
"loadWorkflowDesc": "Arbeitsablauf laden?",
|
||||
"loadWorkflowDesc2": "Ihr aktueller Arbeitsablauf enthält nicht gespeicherte Änderungen.",
|
||||
"loadingTemplates": "Lade {{name}}",
|
||||
"missingSourceOrTargetHandle": "Fehlender Quell- oder Zielgriff",
|
||||
"missingSourceOrTargetNode": "Fehlender Quell- oder Zielknoten",
|
||||
"showEdgeLabelsHelp": "Beschriftungen an Kanten anzeigen, um die verknüpften Knoten zu kennzeichnen"
|
||||
},
|
||||
"hrf": {
|
||||
"enableHrf": "Korrektur für hohe Auflösungen",
|
||||
"upscaleMethod": "Vergrößerungsmethode",
|
||||
"metadata": {
|
||||
"strength": "Auflösungs-Fix Stärke",
|
||||
"enabled": "Auflösungs-Fix aktiviert",
|
||||
@@ -1366,11 +1317,9 @@
|
||||
"models": {
|
||||
"noMatchingModels": "Keine passenden Modelle",
|
||||
"loading": "lade",
|
||||
"noMatchingLoRAs": "Keine passenden LoRAs",
|
||||
"noModelsAvailable": "Keine Modelle verfügbar",
|
||||
"selectModel": "Wählen ein Modell aus",
|
||||
"noRefinerModelsInstalled": "Keine SDXL Refiner-Modelle installiert",
|
||||
"noLoRAsInstalled": "Keine LoRAs installiert",
|
||||
"addLora": "LoRA hinzufügen",
|
||||
"defaultVAE": "Standard VAE",
|
||||
"lora": "LoRA",
|
||||
@@ -1400,31 +1349,23 @@
|
||||
"workflows": "Arbeitsabläufe",
|
||||
"workflowName": "Arbeitsablauf-Name",
|
||||
"saveWorkflowAs": "Arbeitsablauf speichern als",
|
||||
"searchWorkflows": "Suche Arbeitsabläufe",
|
||||
"newWorkflowCreated": "Neuer Arbeitsablauf erstellt",
|
||||
"problemSavingWorkflow": "Problem beim Speichern des Arbeitsablaufs",
|
||||
"problemLoading": "Problem beim Laden von Arbeitsabläufen",
|
||||
"downloadWorkflow": "Speichern als",
|
||||
"savingWorkflow": "Speichere Arbeitsablauf...",
|
||||
"saveWorkflow": "Arbeitsablauf speichern",
|
||||
"noWorkflows": "Keine Arbeitsabläufe",
|
||||
"workflowLibrary": "Bibliothek",
|
||||
"unnamedWorkflow": "Unbenannter Arbeitsablauf",
|
||||
"noDescription": "Keine Beschreibung",
|
||||
"clearWorkflowSearchFilter": "Suchfilter zurücksetzen",
|
||||
"workflowEditorMenu": "Arbeitsablauf-Editor Menü",
|
||||
"deleteWorkflow": "Arbeitsablauf löschen",
|
||||
"workflowSaved": "Arbeitsablauf gespeichert",
|
||||
"uploadWorkflow": "Aus Datei laden",
|
||||
"openWorkflow": "Arbeitsablauf öffnen",
|
||||
"saveWorkflowToProject": "Arbeitsablauf in Projekt speichern",
|
||||
"workflowCleared": "Arbeitsablauf gelöscht",
|
||||
"loading": "Lade Arbeitsabläufe",
|
||||
"name": "Name",
|
||||
"ascending": "Aufsteigend",
|
||||
"defaultWorkflows": "Standard Arbeitsabläufe",
|
||||
"userWorkflows": "Benutzer Arbeitsabläufe",
|
||||
"projectWorkflows": "Projekt Arbeitsabläufe",
|
||||
"opened": "Geöffnet",
|
||||
"loadWorkflow": "Arbeitsablauf $t(common.load)",
|
||||
"updated": "Aktualisiert",
|
||||
@@ -1438,12 +1379,10 @@
|
||||
"copyShareLink": "Teilen-Link kopieren",
|
||||
"download": "Herunterladen",
|
||||
"convertGraph": "Graph konvertieren",
|
||||
"filterByTags": "Nach Tags filtern",
|
||||
"yourWorkflows": "Ihre Arbeitsabläufe",
|
||||
"recentlyOpened": "Kürzlich geöffnet"
|
||||
},
|
||||
"sdxl": {
|
||||
"concatPromptStyle": "Verknüpfen von Prompt & Stil",
|
||||
"scheduler": "Planer",
|
||||
"steps": "Schritte"
|
||||
},
|
||||
@@ -1455,13 +1394,11 @@
|
||||
"addPromptTrigger": "Prompt-Trigger hinzufügen",
|
||||
"compatibleEmbeddings": "Kompatible Einbettungen",
|
||||
"replace": "Ersetzen",
|
||||
"insert": "Einfügen",
|
||||
"discard": "Verwerfen",
|
||||
"generateFromImage": "Prompt aus Bild generieren",
|
||||
"expandCurrentPrompt": "Aktuelle Prompt erweitern",
|
||||
"uploadImageForPromptGeneration": "Bild zur Prompt-Generierung hochladen",
|
||||
"expandingPrompt": "Prompt wird erweitert...",
|
||||
"resultTitle": "Prompt-Erweiterung abgeschlossen"
|
||||
"expandingPrompt": "Prompt wird erweitert..."
|
||||
},
|
||||
"ui": {
|
||||
"tabs": {
|
||||
@@ -1600,8 +1537,6 @@
|
||||
"opacity": "Opazität",
|
||||
"removeBookmark": "Lesezeichen entfernen",
|
||||
"rasterLayer": "Rasterebene",
|
||||
"rasterLayers_withCount_visible": "Rasterebenen ({{count}})",
|
||||
"controlLayers_withCount_visible": "Kontroll-Ebenen ({{count}})",
|
||||
"deleteSelected": "Ausgewählte löschen",
|
||||
"newRegionalReferenceImageError": "Problem beim Erstellen eines regionalen Referenzbilds",
|
||||
"newControlLayerOk": "Kontroll-Ebene erstellt",
|
||||
@@ -1609,10 +1544,8 @@
|
||||
"newRasterLayerOk": "Rasterebene erstellt",
|
||||
"moveToFront": "Nach vorne bringen",
|
||||
"copyToClipboard": "In die Zwischenablage kopieren",
|
||||
"controlLayers_withCount_hidden": "Kontroll-Ebenen ({{count}} ausgeblendet)",
|
||||
"clearCaches": "Cache leeren",
|
||||
"controlLayer": "Kontroll-Ebene",
|
||||
"rasterLayers_withCount_hidden": "Rasterebenen ({{count}} ausgeblendet)",
|
||||
"transparency": "Transparenz",
|
||||
"canvas": "Leinwand",
|
||||
"global": "Global",
|
||||
@@ -1635,9 +1568,7 @@
|
||||
"weight": "Gewichtung",
|
||||
"addReferenceImage": "$t(controlLayers.referenceImage) hinzufügen",
|
||||
"addInpaintMask": "$t(controlLayers.inpaintMask) hinzufügen",
|
||||
"addGlobalReferenceImage": "$t(controlLayers.globalReferenceImage) hinzufügen",
|
||||
"regionalGuidance": "Regionale Führung",
|
||||
"globalReferenceImages_withCount_visible": "Globale Referenzbilder ({{count}})",
|
||||
"addPositivePrompt": "$t(controlLayers.prompt) hinzufügen",
|
||||
"locked": "Gesperrt",
|
||||
"showHUD": "HUD anzeigen",
|
||||
@@ -1645,16 +1576,12 @@
|
||||
"addRasterLayer": "$t(controlLayers.rasterLayer) hinzufügen",
|
||||
"addRegionalGuidance": "$t(controlLayers.regionalGuidance) hinzufügen",
|
||||
"addControlLayer": "$t(controlLayers.controlLayer) hinzufügen",
|
||||
"newCanvasSession": "Neue Leinwand-Sitzung",
|
||||
"replaceLayer": "Ebene ersetzen",
|
||||
"newGallerySession": "Neue Galerie-Sitzung",
|
||||
"unlocked": "Entsperrt",
|
||||
"showProgressOnCanvas": "Fortschritt auf Leinwand anzeigen",
|
||||
"controlMode": {
|
||||
"balanced": "Ausgewogen"
|
||||
},
|
||||
"globalReferenceImages_withCount_hidden": "Globale Referenzbilder ({{count}} ausgeblendet)",
|
||||
"sendToGallery": "An Galerie senden",
|
||||
"stagingArea": {
|
||||
"accept": "Annehmen",
|
||||
"next": "Nächste",
|
||||
@@ -1662,8 +1589,6 @@
|
||||
"discard": "Verwerfen",
|
||||
"previous": "Vorherige"
|
||||
},
|
||||
"regionalGuidance_withCount_visible": "Regionale Führung ({{count}})",
|
||||
"regionalGuidance_withCount_hidden": "Regionale Führung ({{count}} ausgeblendet)",
|
||||
"settings": {
|
||||
"snapToGrid": {
|
||||
"on": "Ein",
|
||||
@@ -1673,8 +1598,6 @@
|
||||
},
|
||||
"layer_one": "Ebene",
|
||||
"layer_other": "Ebenen",
|
||||
"layer_withCount_one": "Ebene ({{count}})",
|
||||
"layer_withCount_other": "Ebenen ({{count}})",
|
||||
"fill": {
|
||||
"fillStyle": "Füllstil",
|
||||
"diagonal": "Diagonal",
|
||||
|
||||
@@ -104,6 +104,7 @@
|
||||
"copy": "Copy",
|
||||
"copyError": "$t(gallery.copy) Error",
|
||||
"clipboard": "Clipboard",
|
||||
"crop": "Crop",
|
||||
"on": "On",
|
||||
"off": "Off",
|
||||
"or": "or",
|
||||
@@ -242,7 +243,10 @@
|
||||
"resultSubtitle": "Choose how to handle the expanded prompt:",
|
||||
"replace": "Replace",
|
||||
"insert": "Insert",
|
||||
"discard": "Discard"
|
||||
"discard": "Discard",
|
||||
"noPromptHistory": "No prompt history recorded.",
|
||||
"noMatchingPrompts": "No matching prompts in history.",
|
||||
"toSwitchBetweenPrompts": "to switch between prompts."
|
||||
},
|
||||
"queue": {
|
||||
"queue": "Queue",
|
||||
@@ -298,7 +302,7 @@
|
||||
"completedIn": "Completed in",
|
||||
"batch": "Batch",
|
||||
"origin": "Origin",
|
||||
"destination": "Destination",
|
||||
"destination": "Dest",
|
||||
"upscaling": "Upscaling",
|
||||
"canvas": "Canvas",
|
||||
"generation": "Generation",
|
||||
@@ -324,7 +328,13 @@
|
||||
"iterations_other": "Iterations",
|
||||
"generations_one": "Generation",
|
||||
"generations_other": "Generations",
|
||||
"batchSize": "Batch Size"
|
||||
"batchSize": "Batch Size",
|
||||
"createdAt": "Created At",
|
||||
"completedAt": "Completed At",
|
||||
"sortColumn": "Sort Column",
|
||||
"sortBy": "Sort by {{column}}",
|
||||
"sortOrderAscending": "Ascending",
|
||||
"sortOrderDescending": "Descending"
|
||||
},
|
||||
"invocationCache": {
|
||||
"invocationCache": "Invocation Cache",
|
||||
@@ -474,6 +484,14 @@
|
||||
"title": "Focus Prompt",
|
||||
"desc": "Move cursor focus to the positive prompt."
|
||||
},
|
||||
"promptHistoryPrev": {
|
||||
"title": "Previous Prompt in History",
|
||||
"desc": "When the prompt is focused, move to the previous (older) prompt in your history."
|
||||
},
|
||||
"promptHistoryNext": {
|
||||
"title": "Next Prompt in History",
|
||||
"desc": "When the prompt is focused, move to the next (newer) prompt in your history."
|
||||
},
|
||||
"toggleLeftPanel": {
|
||||
"title": "Toggle Left Panel",
|
||||
"desc": "Show or hide the left panel."
|
||||
@@ -1252,6 +1270,7 @@
|
||||
"infillColorValue": "Fill Color",
|
||||
"info": "Info",
|
||||
"startingFrameImage": "Start Frame",
|
||||
"startingFrameImageAspectRatioWarning": "Image aspect ratio does not match the video aspect ratio ({{videoAspectRatio}}). This could lead to unexpected cropping during video generation.",
|
||||
"invoke": {
|
||||
"addingImagesTo": "Adding images to",
|
||||
"modelDisabledForTrial": "Generating with {{modelName}} is not available on trial accounts. Visit your account settings to upgrade.",
|
||||
@@ -2077,6 +2096,24 @@
|
||||
"pullBboxIntoLayerError": "Problem Pulling BBox Into Layer",
|
||||
"pullBboxIntoReferenceImageOk": "Bbox Pulled Into ReferenceImage",
|
||||
"pullBboxIntoReferenceImageError": "Problem Pulling BBox Into ReferenceImage",
|
||||
"addAdjustments": "Add Adjustments",
|
||||
"removeAdjustments": "Remove Adjustments",
|
||||
"adjustments": {
|
||||
"simple": "Simple",
|
||||
"curves": "Curves",
|
||||
"heading": "Adjustments",
|
||||
"expand": "Expand adjustments",
|
||||
"collapse": "Collapse adjustments",
|
||||
"brightness": "Brightness",
|
||||
"contrast": "Contrast",
|
||||
"saturation": "Saturation",
|
||||
"temperature": "Temperature",
|
||||
"tint": "Tint",
|
||||
"sharpness": "Sharpness",
|
||||
"finish": "Finish",
|
||||
"reset": "Reset",
|
||||
"master": "Master"
|
||||
},
|
||||
"regionIsEmpty": "Selected region is empty",
|
||||
"mergeVisible": "Merge Visible",
|
||||
"mergeDown": "Merge Down",
|
||||
@@ -2448,12 +2485,21 @@
|
||||
"saveAs": "Save As",
|
||||
"cancel": "Cancel",
|
||||
"process": "Process",
|
||||
"help1": "Select a single target object. Add <Bold>Include</Bold> and <Bold>Exclude</Bold> points to indicate which parts of the layer are part of the target object.",
|
||||
"help2": "Start with one <Bold>Include</Bold> point within the target object. Add more points to refine the selection. Fewer points typically produce better results.",
|
||||
"help3": "Invert the selection to select everything except the target object.",
|
||||
"desc": "Select a single target object. After selection is complete, click <Bold>Apply</Bold> to discard everything outside the selected area, or save the selection as a new layer.",
|
||||
"visualModeDesc": "Visual mode uses box and point inputs to select an object.",
|
||||
"visualMode1": "Click and drag to draw a box around the object you want to select. You may get better results by drawing the box a bit larger or smaller than the object.",
|
||||
"visualMode2": "Click to add a green <Bold>include</Bold> point, or shift-click to add a red <Bold>exclude</Bold> point to tell the model what to include or exclude.",
|
||||
"visualMode3": "Points can be used to refine a box selection or used independently.",
|
||||
"promptModeDesc": "Prompt mode uses text input to select an object.",
|
||||
"promptMode1": "Type a brief description of the object you want to select.",
|
||||
"promptMode2": "Use simple language, avoiding complex descriptions or multiple objects.",
|
||||
"clickToAdd": "Click on the layer to add a point",
|
||||
"dragToMove": "Drag a point to move it",
|
||||
"clickToRemove": "Click on a point to remove it"
|
||||
"clickToRemove": "Click on a point to remove it",
|
||||
"model": "Model",
|
||||
"segmentAnything1": "Segment Anything 1",
|
||||
"segmentAnything2": "Segment Anything 2",
|
||||
"prompt": "Selection Prompt"
|
||||
},
|
||||
"settings": {
|
||||
"snapToGrid": {
|
||||
@@ -2748,8 +2794,9 @@
|
||||
"whatsNew": {
|
||||
"whatsNewInInvoke": "What's New in Invoke",
|
||||
"items": [
|
||||
"Canvas: Color Picker does not sample alpha, bbox respects aspect ratio lock when resizing shuffle button for number fields in Workflow Builder, hide pixel dimension sliders when using a model that doesn't support them",
|
||||
"Workflows: Add a Shuffle button to number input fields"
|
||||
"Select Object v2: Improved object selection with point and box inputs or text prompts.",
|
||||
"Raster Layer Adjustments: Easily adjust layer brightness, contrast, saturation, curves and more.",
|
||||
"Prompt History: Review and quickly recall your last 100 prompts."
|
||||
],
|
||||
"readReleaseNotes": "Read Release Notes",
|
||||
"watchRecentReleaseVideos": "Watch Recent Release Videos",
|
||||
|
||||
@@ -47,11 +47,8 @@
|
||||
"editor": "Editor",
|
||||
"orderBy": "Ordenar por",
|
||||
"file": "Archivo",
|
||||
"goTo": "Ir a",
|
||||
"imageFailedToLoad": "No se puede cargar la imagen",
|
||||
"saveAs": "Guardar Como",
|
||||
"somethingWentWrong": "Algo salió mal",
|
||||
"nextPage": "Página Siguiente",
|
||||
"selected": "Seleccionado",
|
||||
"tab": "Tabulador",
|
||||
"positivePrompt": "Prompt Positivo",
|
||||
@@ -61,7 +58,6 @@
|
||||
"unknown": "Desconocido",
|
||||
"input": "Entrada",
|
||||
"template": "Plantilla",
|
||||
"prevPage": "Página Anterior",
|
||||
"red": "Rojo",
|
||||
"alpha": "Transparencia",
|
||||
"outputs": "Resultados",
|
||||
@@ -94,8 +90,6 @@
|
||||
"edit": "Editar",
|
||||
"safetensors": "Safetensors",
|
||||
"toResolve": "Para resolver",
|
||||
"localSystem": "Sistema local",
|
||||
"notInstalled": "No $t(common.installed)",
|
||||
"outpaint": "outpaint",
|
||||
"simple": "Sencillo",
|
||||
"close": "Cerrar"
|
||||
@@ -104,7 +98,6 @@
|
||||
"galleryImageSize": "Tamaño de la imagen",
|
||||
"gallerySettings": "Ajustes de la galería",
|
||||
"autoSwitchNewImages": "Auto seleccionar Imágenes nuevas",
|
||||
"noImagesInGallery": "No hay imágenes para mostrar",
|
||||
"deleteImage_one": "Eliminar Imagen",
|
||||
"deleteImage_many": "Eliminar {{count}} Imágenes",
|
||||
"deleteImage_other": "Eliminar {{count}} Imágenes",
|
||||
@@ -118,9 +111,7 @@
|
||||
"selectForCompare": "Seleccionar para comparar",
|
||||
"alwaysShowImageSizeBadge": "Mostrar siempre las dimensiones de la imagen",
|
||||
"currentlyInUse": "Esta imagen se utiliza actualmente con las siguientes funciones:",
|
||||
"unableToLoad": "No se puede cargar la galería",
|
||||
"selectAllOnPage": "Seleccionar todo en la página",
|
||||
"selectAnImageToCompare": "Seleccione una imagen para comparar",
|
||||
"bulkDownloadFailed": "Error en la descarga",
|
||||
"compareHelp2": "Presione <Kbd> M </Kbd> para recorrer los modos de comparación.",
|
||||
"move": "Mover",
|
||||
@@ -145,7 +136,6 @@
|
||||
"exitBoardSearch": "Finalizar búsqueda",
|
||||
"exitSearch": "Salir de la búsqueda de imágenes",
|
||||
"featuresWillReset": "Si elimina esta imagen, dichas funciones se restablecerán inmediatamente.",
|
||||
"jump": "Omitir",
|
||||
"loading": "Cargando",
|
||||
"newestFirst": "La más nueva primero",
|
||||
"unstarImage": "Dejar de ser favorita",
|
||||
@@ -163,9 +153,7 @@
|
||||
"boardsSettings": "Ajustes de los tableros",
|
||||
"imagesSettings": "Configuración de imágenes de la galería",
|
||||
"compareHelp3": "Presione <Kbd> C </Kbd> para intercambiar las imágenes comparadas.",
|
||||
"showArchivedBoards": "Mostrar paneles archivados",
|
||||
"closeViewer": "Cerrar visor",
|
||||
"openViewer": "Abrir visor"
|
||||
"showArchivedBoards": "Mostrar paneles archivados"
|
||||
},
|
||||
"modelManager": {
|
||||
"modelManager": "Gestor de Modelos",
|
||||
@@ -239,12 +227,10 @@
|
||||
"scaledHeight": "Alto escalado",
|
||||
"infillMethod": "Método de relleno",
|
||||
"tileSize": "Tamaño del mosaico",
|
||||
"downloadImage": "Descargar imagen",
|
||||
"usePrompt": "Usar Entrada",
|
||||
"useSeed": "Usar Semilla",
|
||||
"useAll": "Usar Todo",
|
||||
"info": "Información",
|
||||
"showOptionsPanel": "Mostrar panel lateral (O o T)",
|
||||
"symmetry": "Simetría",
|
||||
"copyImage": "Copiar la imagen",
|
||||
"general": "General",
|
||||
@@ -323,8 +309,6 @@
|
||||
"hideMinimapnodes": "Ocultar el minimapa",
|
||||
"fitViewportNodes": "Ajustar la vista",
|
||||
"zoomOutNodes": "Alejar",
|
||||
"hideLegendNodes": "Ocultar la leyenda del tipo de campo",
|
||||
"showLegendNodes": "Mostrar la leyenda del tipo de campo",
|
||||
"showMinimapnodes": "Mostrar el minimapa",
|
||||
"reloadNodeTemplates": "Recargar las plantillas de nodos",
|
||||
"loadWorkflow": "Cargar el flujo de trabajo",
|
||||
@@ -361,7 +345,6 @@
|
||||
"assetsWithCount_one": "{{count}} activo",
|
||||
"assetsWithCount_many": "{{count}} activos",
|
||||
"assetsWithCount_other": "{{count}} activos",
|
||||
"hideBoards": "Ocultar paneles",
|
||||
"addPrivateBoard": "Agregar un panel privado",
|
||||
"addSharedBoard": "Añadir panel compartido",
|
||||
"boards": "Paneles",
|
||||
@@ -372,7 +355,6 @@
|
||||
"noBoards": "No hay paneles {{boardType}}",
|
||||
"shared": "Paneles compartidos",
|
||||
"deletedPrivateBoardsCannotbeRestored": "Los paneles eliminados no se pueden restaurar. Al elegir \"Eliminar solo el panel\", las imágenes se colocan en un estado privado y sin categoría para el creador de la imagen.",
|
||||
"viewBoards": "Ver paneles",
|
||||
"private": "Paneles privados",
|
||||
"updateBoardError": "No se pudo actualizar el panel"
|
||||
},
|
||||
@@ -461,7 +443,6 @@
|
||||
"other": "Otro",
|
||||
"queueFront": "Añadir al principio de la cola",
|
||||
"gallery": "Galería",
|
||||
"batchFieldValues": "Valores de procesamiento por lotes",
|
||||
"session": "Sesión",
|
||||
"notReady": "La cola aún no está lista",
|
||||
"graphQueued": "Gráfico en cola",
|
||||
@@ -494,15 +475,11 @@
|
||||
"layer_one": "Capa",
|
||||
"layer_many": "Capas",
|
||||
"layer_other": "Capas",
|
||||
"layer_withCount_one": "({{count}}) capa",
|
||||
"layer_withCount_many": "({{count}}) capas",
|
||||
"layer_withCount_other": "({{count}}) capas",
|
||||
"copyToClipboard": "Copiar al portapapeles"
|
||||
},
|
||||
"whatsNew": {
|
||||
"readReleaseNotes": "Leer las notas de la versión",
|
||||
"watchRecentReleaseVideos": "Ver videos de versiones recientes",
|
||||
"watchUiUpdatesOverview": "Descripción general de las actualizaciones de la interfaz de usuario de Watch",
|
||||
"whatsNewInInvoke": "Novedades en Invoke",
|
||||
"items": [
|
||||
"<StrongComponent>SD 3.5</StrongComponent>: compatibilidad con SD 3.5 Medium y Large."
|
||||
@@ -527,13 +504,11 @@
|
||||
},
|
||||
"hrf": {
|
||||
"hrf": "Solución de alta resolución",
|
||||
"enableHrf": "Activar corrección de alta resolución",
|
||||
"metadata": {
|
||||
"enabled": "Corrección de alta resolución activada",
|
||||
"strength": "Forzar la corrección de alta resolución",
|
||||
"method": "Método de corrección de alta resolución"
|
||||
},
|
||||
"upscaleMethod": "Método de expansión"
|
||||
}
|
||||
},
|
||||
"prompt": {
|
||||
"addPromptTrigger": "Añadir activador de los avisos",
|
||||
@@ -864,10 +839,8 @@
|
||||
"seed": "Semilla",
|
||||
"strength": "Forzar imagen a imagen",
|
||||
"recallParameters": "Parámetros de recuperación",
|
||||
"recallParameter": "Recuperar {{label}}",
|
||||
"steps": "Pasos",
|
||||
"noRecallParameters": "Sin parámetros para recuperar",
|
||||
"parsingFailed": "Error al analizar"
|
||||
"noRecallParameters": "Sin parámetros para recuperar"
|
||||
},
|
||||
"system": {
|
||||
"logLevel": {
|
||||
|
||||
@@ -28,7 +28,6 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "Kuvan koko",
|
||||
"gallerySettings": "Gallerian asetukset",
|
||||
"autoSwitchNewImages": "Vaihda uusiin kuviin automaattisesti",
|
||||
"noImagesInGallery": "Ei kuvia galleriassa"
|
||||
"autoSwitchNewImages": "Vaihda uusiin kuviin automaattisesti"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -27,21 +27,15 @@
|
||||
"error": "Erreur",
|
||||
"installed": "Installé",
|
||||
"format": "format",
|
||||
"goTo": "Aller à",
|
||||
"input": "Entrée",
|
||||
"linear": "Linéaire",
|
||||
"localSystem": "Système local",
|
||||
"learnMore": "En savoir plus",
|
||||
"modelManager": "Gestionnaire de modèle",
|
||||
"notInstalled": "Non $t(common.installed)",
|
||||
"openInNewTab": "Ouvrir dans un nouvel onglet",
|
||||
"somethingWentWrong": "Une erreur s'est produite",
|
||||
"created": "Créé",
|
||||
"tab": "Onglet",
|
||||
"folder": "Dossier",
|
||||
"imageFailedToLoad": "Impossible de charger l'Image",
|
||||
"prevPage": "Page précédente",
|
||||
"nextPage": "Page suivante",
|
||||
"selected": "Sélectionné",
|
||||
"save": "Enregistrer",
|
||||
"updated": "Mis à jour",
|
||||
@@ -111,7 +105,6 @@
|
||||
"min": "Min",
|
||||
"max": "Max",
|
||||
"values": "Valeurs",
|
||||
"resetToDefaults": "Réinitialiser par défaut",
|
||||
"seed": "Graine",
|
||||
"combinatorial": "Combinatoire"
|
||||
},
|
||||
@@ -119,11 +112,9 @@
|
||||
"galleryImageSize": "Taille de l'image",
|
||||
"gallerySettings": "Paramètres de la galerie",
|
||||
"autoSwitchNewImages": "Basculer automatiquement vers de nouvelles images",
|
||||
"noImagesInGallery": "Aucune image à afficher",
|
||||
"bulkDownloadRequestedDesc": "Votre demande de téléchargement est en cours de traitement. Cela peut prendre quelques instants.",
|
||||
"deleteSelection": "Supprimer la sélection",
|
||||
"selectAllOnPage": "Séléctionner toute la page",
|
||||
"unableToLoad": "Impossible de charger la Galerie",
|
||||
"featuresWillReset": "Si vous supprimez cette image, ces fonctionnalités vont être réinitialisés.",
|
||||
"loading": "Chargement",
|
||||
"sortDirection": "Direction de tri",
|
||||
@@ -149,7 +140,6 @@
|
||||
"openInViewer": "Ouvrir dans le Visualiseur",
|
||||
"showArchivedBoards": "Montrer les Planches archivées",
|
||||
"selectForCompare": "Séléctionner pour comparaison",
|
||||
"selectAnImageToCompare": "Séléctionner une Image à comparer",
|
||||
"exitCompare": "Sortir de la comparaison",
|
||||
"compareHelp2": "Appuyez sur <Kbd>M</Kbd> pour faire défiler les modes de comparaison.",
|
||||
"swapImages": "Échanger les Images",
|
||||
@@ -157,10 +147,7 @@
|
||||
"compareHelp1": "Maintenir <Kbd>Alt</Kbd> lors du clic d'une image dans la galerie ou en utilisant les flèches du clavier pour changer l'Image à comparer.",
|
||||
"compareHelp3": "Appuyer sur <Kbd>C</Kbd> pour échanger les images à comparer.",
|
||||
"image": "image",
|
||||
"openViewer": "Ouvrir le Visualisateur",
|
||||
"closeViewer": "Fermer le Visualisateur",
|
||||
"currentlyInUse": "Cette image est actuellement utilisée dans ces fonctionalités :",
|
||||
"jump": "Sauter",
|
||||
"starImage": "Marquer l'Image",
|
||||
"download": "Téléchargement",
|
||||
"deleteImage_one": "Supprimer l'Image",
|
||||
@@ -247,7 +234,6 @@
|
||||
"metadata": "Métadonnées",
|
||||
"scanFolder": "Scanner le dossier",
|
||||
"inplaceInstallDesc": "Installez les modèles sans copier les fichiers. Lors de l'utilisation du modèle, il sera chargé depuis cet emplacement. Si cette option est désactivée, le(s) fichier(s) du modèle seront copiés dans le répertoire des modèles géré par Invoke lors de l'installation.",
|
||||
"ipAdapters": "Adaptateurs IP",
|
||||
"installQueue": "File d'attente d'installation",
|
||||
"modelImageDeleteFailed": "Échec de la suppression de l'image du modèle",
|
||||
"modelName": "Nom du modèle",
|
||||
@@ -288,7 +274,6 @@
|
||||
"scanFolderHelper": "Le dossier sera analysé de manière récursive à la recherche de modèles. Cela peut prendre quelques instants pour des dossiers très volumineux.",
|
||||
"clipEmbed": "Intégration CLIP",
|
||||
"spandrelImageToImage": "Image vers Image (Spandrel)",
|
||||
"starterModelsInModelManager": "Les modèles de démarrage peuvent être trouvés dans le gestionnaire de modèles",
|
||||
"t5Encoder": "Encodeur T5",
|
||||
"learnMoreAboutSupportedModels": "En savoir plus sur les modèles que nous prenons en charge",
|
||||
"includesNModels": "Contient {{n}} modèles et leurs dépendances",
|
||||
@@ -346,12 +331,10 @@
|
||||
"infillMethod": "Méthode de Remplissage",
|
||||
"tileSize": "Taille des Tuiles",
|
||||
"copyImage": "Copier Image",
|
||||
"downloadImage": "Télécharger Image",
|
||||
"usePrompt": "Utiliser la suggestion",
|
||||
"useSeed": "Utiliser la graine",
|
||||
"useAll": "Tout utiliser",
|
||||
"info": "Info",
|
||||
"showOptionsPanel": "Afficher le panneau latéral (O ou T)",
|
||||
"invoke": {
|
||||
"noPrompts": "Aucun prompts généré",
|
||||
"missingInputForField": "entrée manquante",
|
||||
@@ -362,21 +345,16 @@
|
||||
"noModelSelected": "Aucun modèle sélectionné",
|
||||
"noNodesInGraph": "Aucun nœud dans le graphique",
|
||||
"systemDisconnected": "Système déconnecté",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), la hauteur de la bounding box est {{height}}",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), la hauteur de la bounding box est {{height}}",
|
||||
"noFLUXVAEModelSelected": "Aucun modèle VAE sélectionné pour la génération FLUX",
|
||||
"canvasIsTransforming": "La Toile est occupée (en transformation)",
|
||||
"canvasIsRasterizing": "La Toile est occupée (en rastérisation)",
|
||||
"noCLIPEmbedModelSelected": "Aucun modèle CLIP Embed sélectionné pour la génération FLUX",
|
||||
"canvasIsFiltering": "La Toile est occupée (en filtration)",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), la largeur de la bounding box est {{width}}",
|
||||
"noT5EncoderModelSelected": "Aucun modèle T5 Encoder sélectionné pour la génération FLUX",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), la largeur de la bounding box mise à l'échelle est {{width}}",
|
||||
"canvasIsCompositing": "La Toile est occupée (en composition)",
|
||||
"collectionTooFewItems": "trop peu d'éléments, minimum {{minItems}}",
|
||||
"collectionTooManyItems": "trop d'éléments, maximum {{maxItems}}",
|
||||
"canvasIsSelectingObject": "La toile est occupée (sélection d'objet)",
|
||||
"emptyBatches": "lots vides",
|
||||
"batchNodeNotConnected": "Noeud de lots non connecté : {{label}}",
|
||||
"fluxModelMultipleControlLoRAs": "Vous ne pouvez utiliser qu'un seul Control LoRA à la fois",
|
||||
"collectionNumberLTMin": "{{value}} < {{minimum}} (incl. min)",
|
||||
@@ -468,9 +446,7 @@
|
||||
"informationalPopoversDisabled": "Pop-ups d'information désactivés",
|
||||
"informationalPopoversDisabledDesc": "Les pop-ups d'information ont été désactivés. Activez-les dans les paramètres.",
|
||||
"confirmOnNewSession": "Confirmer lors d'une nouvelle session",
|
||||
"modelDescriptionsDisabledDesc": "Les descriptions des modèles dans les menus déroulants ont été désactivées. Activez-les dans les paramètres.",
|
||||
"enableModelDescriptions": "Activer les descriptions de modèle dans les menus déroulants",
|
||||
"modelDescriptionsDisabled": "Descriptions de modèle dans les menus déroulants désactivés",
|
||||
"showDetailedInvocationProgress": "Afficher les détails de progression"
|
||||
},
|
||||
"toast": {
|
||||
@@ -486,17 +462,14 @@
|
||||
"addedToBoard": "Ajouté aux ressources de la planche {{name}}",
|
||||
"workflowLoaded": "Workflow chargé",
|
||||
"connected": "Connecté au serveur",
|
||||
"setNodeField": "Définir comme champ de nœud",
|
||||
"imageUploadFailed": "Échec de l'importation de l'image",
|
||||
"loadedWithWarnings": "Workflow chargé avec des avertissements",
|
||||
"imageUploaded": "Image importée",
|
||||
"modelAddedSimple": "Modèle ajouté à la file d'attente",
|
||||
"setControlImage": "Définir comme image de contrôle",
|
||||
"workflowDeleted": "Workflow supprimé",
|
||||
"baseModelChangedCleared_one": "Effacé ou désactivé {{count}} sous-modèle incompatible",
|
||||
"baseModelChangedCleared_many": "Effacé ou désactivé {{count}} sous-modèles incompatibles",
|
||||
"baseModelChangedCleared_other": "Effacé ou désactivé {{count}} sous-modèles incompatibles",
|
||||
"invalidUpload": "Importation invalide",
|
||||
"problemDownloadingImage": "Impossible de télécharger l'image",
|
||||
"problemRetrievingWorkflow": "Problème de récupération du Workflow",
|
||||
"problemDeletingWorkflow": "Problème de suppression du Workflow",
|
||||
@@ -510,12 +483,10 @@
|
||||
"errorCopied": "Erreur copiée",
|
||||
"parametersSet": "Paramètres rappelés",
|
||||
"somethingWentWrong": "Quelque chose a échoué",
|
||||
"imageSaved": "Image enregistrée",
|
||||
"unableToLoadStylePreset": "Impossible de charger le préréglage de style",
|
||||
"stylePresetLoaded": "Préréglage de style chargé",
|
||||
"parameterNotSetDescWithMessage": "Impossible de rappeler {{parameter}} : {{message}}",
|
||||
"importFailed": "Importation échouée",
|
||||
"imageSavingFailed": "Échec de l'enregistrement de l'image",
|
||||
"importSuccessful": "Importation réussie",
|
||||
"outOfMemoryError": "Erreur de mémoire insuffisante",
|
||||
"sessionRef": "Session : {{sessionId}}",
|
||||
@@ -523,16 +494,11 @@
|
||||
"parameterSetDesc": "Rappelé {{parameter}}",
|
||||
"parameterNotSetDesc": "Impossible de rappeler {{parameter}}",
|
||||
"layerCopiedToClipboard": "Calque copié dans le presse-papiers",
|
||||
"layerSavedToAssets": "Calque enregistré dans les ressources",
|
||||
"problemCopyingLayer": "Impossible de copier la couche",
|
||||
"baseModelChanged": "Modèle de base changé",
|
||||
"problemSavingLayer": "Impossible d'enregistrer la couche",
|
||||
"imageNotLoadedDesc": "Image introuvable",
|
||||
"linkCopied": "Lien copié",
|
||||
"imagesWillBeAddedTo": "Les images Importées seront ajoutées au ressources de la Planche {{boardName}}.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_one": "Doit être au maximum une image PNG ou JPEG.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_many": "Doit être au maximum {{count}} images PNG ou JPEG.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "Doit être au maximum {{count}} images PNG ou JPEG.",
|
||||
"addedToUncategorized": "Ajouté aux ressources de la planche $t(boards.uncategorized)",
|
||||
"pasteSuccess": "Collé à {{destination}}",
|
||||
"pasteFailed": "Échec du collage",
|
||||
@@ -580,8 +546,6 @@
|
||||
"movingImagesToBoard_one": "Déplacer {{count}} image à cette planche :",
|
||||
"movingImagesToBoard_many": "Déplacer {{count}} images à cette planche :",
|
||||
"movingImagesToBoard_other": "Déplacer {{count}} image à cette planche :",
|
||||
"viewBoards": "Voir les Planches",
|
||||
"hideBoards": "Cacher les Planches",
|
||||
"noBoards": "Pas de Planches {{boardType}}",
|
||||
"shared": "Planches Partagées",
|
||||
"searchBoard": "Chercher les Planches...",
|
||||
@@ -681,7 +645,6 @@
|
||||
"batchQueued": "Lot ajouté à la file d'attente",
|
||||
"gallery": "Galerie",
|
||||
"notReady": "Impossible d'ajouter à la file d'attente",
|
||||
"batchFieldValues": "Valeurs Champ Lot",
|
||||
"front": "début",
|
||||
"graphQueued": "Graph ajouté à la file d'attente",
|
||||
"other": "Autre",
|
||||
@@ -712,13 +675,11 @@
|
||||
"compatibleEmbeddings": "Embeddings Compatibles"
|
||||
},
|
||||
"hrf": {
|
||||
"upscaleMethod": "Méthode d'Agrandissement",
|
||||
"metadata": {
|
||||
"enabled": "Correction Haute Résolution Activée",
|
||||
"strength": "Force de la Correction Haute Résolution",
|
||||
"method": "Méthode de la Correction Haute Résolution"
|
||||
},
|
||||
"enableHrf": "Activer la Correction Haute Résolution",
|
||||
"hrf": "Correction Haute Résolution"
|
||||
},
|
||||
"invocationCache": {
|
||||
@@ -1486,8 +1447,7 @@
|
||||
"showDynamicPrompts": "Afficher les Prompts dynamiques",
|
||||
"dynamicPrompts": "Prompts Dynamiques",
|
||||
"promptsPreview": "Prévisualisation des Prompts",
|
||||
"loading": "Génération des Pompts Dynamiques...",
|
||||
"promptsToGenerate": "Prompts à générer"
|
||||
"loading": "Génération des Pompts Dynamiques..."
|
||||
},
|
||||
"metadata": {
|
||||
"positivePrompt": "Prompt Positif",
|
||||
@@ -1515,18 +1475,12 @@
|
||||
"recallParameters": "Rappeler les paramètres",
|
||||
"imageDimensions": "Dimensions de l'image",
|
||||
"parameterSet": "Paramètre {{parameter}} défini",
|
||||
"parsingFailed": "L'analyse a échoué",
|
||||
"recallParameter": "Rappeler {{label}}",
|
||||
"canvasV2Metadata": "Toile",
|
||||
"guidance": "Guide",
|
||||
"seamlessXAxis": "Axe X sans bords",
|
||||
"seamlessYAxis": "Axe Y sans bords"
|
||||
},
|
||||
"sdxl": {
|
||||
"freePromptStyle": "Écriture de Prompt manuelle",
|
||||
"concatPromptStyle": "Lier Prompt & Style",
|
||||
"negStylePrompt": "Style Prompt Négatif",
|
||||
"posStylePrompt": "Style Prompt Positif",
|
||||
"refinerStart": "Démarrer le Refiner",
|
||||
"denoisingStrength": "Force de débruitage",
|
||||
"steps": "Étapes",
|
||||
@@ -1543,8 +1497,6 @@
|
||||
"nodes": {
|
||||
"showMinimapnodes": "Afficher la MiniCarte",
|
||||
"fitViewportNodes": "Ajuster la Vue",
|
||||
"hideLegendNodes": "Masquer la légende du type de champ",
|
||||
"showLegendNodes": "Afficher la légende du type de champ",
|
||||
"hideMinimapnodes": "Masquer MiniCarte",
|
||||
"zoomOutNodes": "Dézoomer",
|
||||
"zoomInNodes": "Zoomer",
|
||||
@@ -1568,9 +1520,7 @@
|
||||
"colorCodeEdges": "Code de couleur des connexions",
|
||||
"colorCodeEdgesHelp": "Code couleur des connexions en fonction de leurs champs connectés",
|
||||
"currentImage": "Image actuelle",
|
||||
"noFieldsLinearview": "Aucun champ ajouté à la vue linéaire",
|
||||
"float": "Flottant",
|
||||
"mismatchedVersion": "Nœud invalide : le nœud {{node}} de type {{type}} a une version incompatible (essayez de mettre à jour ?)",
|
||||
"missingTemplate": "Nœud invalide : le nœud {{node}} de type {{type}} modèle manquant (non installé ?)",
|
||||
"noWorkflow": "Pas de Workflow",
|
||||
"validateConnectionsHelp": "Prévenir la création de connexions invalides et l'invocation de graphes invalides",
|
||||
@@ -1581,12 +1531,10 @@
|
||||
"scheduler": "Planificateur",
|
||||
"notes": "Notes",
|
||||
"notesDescription": "Ajouter des notes sur votre workflow",
|
||||
"unableToLoadWorkflow": "Impossible de charger le Workflow",
|
||||
"addNode": "Ajouter un nœud",
|
||||
"problemSettingTitle": "Problème lors de définition du Titre",
|
||||
"connectionWouldCreateCycle": "La connexion créerait un cycle",
|
||||
"currentImageDescription": "Affiche l'image actuelle dans l'éditeur de nœuds",
|
||||
"versionUnknown": " Version inconnue",
|
||||
"cannotConnectInputToInput": "Impossible de connecter l'entrée à l'entrée",
|
||||
"addNodeToolTip": "Ajouter un nœud (Shift+A, Espace)",
|
||||
"fullyContainNodesHelp": "Les nœuds doivent être entièrement à l'intérieur de la zone de sélection pour être sélectionnés",
|
||||
@@ -1602,7 +1550,6 @@
|
||||
"nodeSearch": "Rechercher des nœuds",
|
||||
"collection": "Collection",
|
||||
"noOutputRecorded": "Aucun résultat enregistré",
|
||||
"removeLinearView": "Retirer de la vue linéaire",
|
||||
"snapToGrid": "Aligner sur la grille",
|
||||
"workflow": "Workflow",
|
||||
"updateApp": "Mettre à jour l'application",
|
||||
@@ -1611,7 +1558,6 @@
|
||||
"noConnectionInProgress": "Aucune connexion en cours",
|
||||
"nodeType": "Type de nœud",
|
||||
"workflowContact": "Contact",
|
||||
"unknownTemplate": "Modèle inconnu",
|
||||
"unknownNode": "Nœud inconnu",
|
||||
"workflowVersion": "Version",
|
||||
"string": "Chaîne de caractères",
|
||||
@@ -1625,7 +1571,6 @@
|
||||
"cannotDuplicateConnection": "Impossible de créer des connexions en double",
|
||||
"resetToDefaultValue": "Réinitialiser à la valeur par défaut",
|
||||
"unknownNodeType": "Type de nœud inconnu",
|
||||
"unknownInput": "Entrée inconnue : {{name}}",
|
||||
"prototypeDesc": "Cette invocation est un prototype. Elle peut subir des modifications majeures lors des mises à jour de l'application et peut être supprimée à tout moment.",
|
||||
"nodePack": "Paquet de nœuds",
|
||||
"sourceNodeDoesNotExist": "Connexion invalide : le nœud source/de sortie {{node}} n'existe pas",
|
||||
@@ -1640,7 +1585,6 @@
|
||||
"clearWorkflow": "Effacer le Workflow",
|
||||
"clearWorkflowDesc": "Effacer ce workflow et en commencer un nouveau ?",
|
||||
"unsupportedArrayItemType": "type d'élément de tableau non pris en charge \"{{type}}\"",
|
||||
"addLinearView": "Ajouter à la vue linéaire",
|
||||
"collectionOrScalarFieldType": "{{name}} (Unique ou Collection)",
|
||||
"unableToExtractEnumOptions": "impossible d'extraire les options d'énumération",
|
||||
"unsupportedAnyOfLength": "trop de membres dans l'union ({{count}})",
|
||||
@@ -1648,7 +1592,6 @@
|
||||
"viewMode": "Utiliser en vue linéaire",
|
||||
"collectionFieldType": "{{name}} (Collection)",
|
||||
"newWorkflow": "Nouveau Workflow",
|
||||
"reorderLinearView": "Réorganiser la vue linéaire",
|
||||
"outputFieldTypeParseError": "Impossible d'analyser le type du champ de sortie {{node}}.{{field}} ({{message}})",
|
||||
"unsupportedMismatchedUnion": "type CollectionOrScalar non concordant avec les types de base {{firstType}} et {{secondType}}",
|
||||
"unableToParseFieldType": "impossible d'analyser le type de champ",
|
||||
@@ -1682,13 +1625,9 @@
|
||||
"arithmeticSequence": "Séquence Arithmétique",
|
||||
"uniformRandomDistribution": "Distribution Aléatoire Uniforme",
|
||||
"noBatchGroup": "aucun groupe",
|
||||
"generatorLoading": "chargement",
|
||||
"generatorLoadFromFile": "Charger depuis un Fichier",
|
||||
"dynamicPromptsRandom": "Prompts Dynamiques (Aléatoire)",
|
||||
"integerRangeGenerator": "Générateur d'interval d'entiers",
|
||||
"generateValues": "Générer Valeurs",
|
||||
"linearDistribution": "Distribution Linéaire",
|
||||
"floatRangeGenerator": "Générateur d'interval de nombres décimaux",
|
||||
"generatorNRandomValues_one": "{{count}} valeur aléatoire",
|
||||
"generatorNRandomValues_many": "{{count}} valeurs aléatoires",
|
||||
"generatorNRandomValues_other": "{{count}} valeurs aléatoires",
|
||||
@@ -1708,7 +1647,6 @@
|
||||
"generatorImagesCategory": "Catégorie",
|
||||
"generatorImagesFromBoard": "Images de la Planche",
|
||||
"missingSourceOrTargetHandle": "Manque de gestionnaire source ou cible",
|
||||
"loadingTemplates": "Chargement de {{name}}",
|
||||
"loadWorkflowDesc2": "Votre workflow actuel contient des modifications non enregistrées.",
|
||||
"generatorImages_one": "{{count}} image",
|
||||
"generatorImages_many": "{{count}} images",
|
||||
@@ -1719,10 +1657,8 @@
|
||||
"noModelsAvailable": "Aucun modèle disponible",
|
||||
"loading": "chargement",
|
||||
"selectModel": "Sélectionner un modèle",
|
||||
"noMatchingLoRAs": "Aucun LoRA correspondant",
|
||||
"lora": "LoRA",
|
||||
"noRefinerModelsInstalled": "Aucun modèle SDXL Refiner installé",
|
||||
"noLoRAsInstalled": "Aucun LoRA installé",
|
||||
"addLora": "Ajouter LoRA",
|
||||
"defaultVAE": "VAE par défaut",
|
||||
"concepts": "Concepts"
|
||||
@@ -1730,11 +1666,8 @@
|
||||
"workflows": {
|
||||
"workflowLibrary": "Bibliothèque",
|
||||
"loading": "Chargement des Workflows",
|
||||
"searchWorkflows": "Chercher des Workflows",
|
||||
"workflowCleared": "Workflow effacé",
|
||||
"noDescription": "Aucune description",
|
||||
"deleteWorkflow": "Supprimer le Workflow",
|
||||
"openWorkflow": "Ouvrir le Workflow",
|
||||
"uploadWorkflow": "Charger à partir d'un fichier",
|
||||
"workflowName": "Nom du Workflow",
|
||||
"unnamedWorkflow": "Workflow sans nom",
|
||||
@@ -1747,8 +1680,6 @@
|
||||
"problemSavingWorkflow": "Problème de sauvegarde du Workflow",
|
||||
"workflowEditorMenu": "Menu de l'Éditeur de Workflow",
|
||||
"newWorkflowCreated": "Nouveau Workflow créé",
|
||||
"clearWorkflowSearchFilter": "Réinitialiser le filtre de recherche de Workflow",
|
||||
"problemLoading": "Problème de chargement des Workflows",
|
||||
"workflowSaved": "Workflow enregistré",
|
||||
"noWorkflows": "Pas de Workflows",
|
||||
"ascending": "Ascendant",
|
||||
@@ -1761,9 +1692,6 @@
|
||||
"opened": "Ouvert",
|
||||
"name": "Nom",
|
||||
"autoLayout": "Mise en page automatique",
|
||||
"defaultWorkflows": "Workflows par défaut",
|
||||
"userWorkflows": "Workflows de l'utilisateur",
|
||||
"projectWorkflows": "Workflows du projet",
|
||||
"copyShareLink": "Copier le lien de partage",
|
||||
"chooseWorkflowFromLibrary": "Choisir le Workflow dans la Bibliothèque",
|
||||
"edit": "Modifer",
|
||||
@@ -1780,7 +1708,6 @@
|
||||
"multiLine": "Multi Ligne",
|
||||
"headingPlaceholder": "En-tête vide",
|
||||
"emptyRootPlaceholderEditMode": "Faites glisser un élément de formulaire ou un champ de nœud ici pour commencer.",
|
||||
"emptyRootPlaceholderViewMode": "Cliquez sur Modifier pour commencer à créer un formulaire pour ce workflow.",
|
||||
"containerPlaceholder": "Conteneur Vide",
|
||||
"row": "Ligne",
|
||||
"column": "Colonne",
|
||||
@@ -1794,10 +1721,8 @@
|
||||
"builder": "Constructeur de Formulaire",
|
||||
"resetAllNodeFields": "Réinitialiser tous les champs de nœud",
|
||||
"deleteAllElements": "Supprimer tous les éléments de formulaire",
|
||||
"workflowBuilderAlphaWarning": "Le constructeur de workflow est actuellement en version alpha. Il peut y avoir des changements majeurs avant la version stable.",
|
||||
"showDescription": "Afficher la description"
|
||||
},
|
||||
"openLibrary": "Ouvrir la Bibliothèque"
|
||||
}
|
||||
},
|
||||
"whatsNew": {
|
||||
"whatsNewInInvoke": "Quoi de neuf dans Invoke",
|
||||
@@ -1806,8 +1731,7 @@
|
||||
"<StrongComponent>FLUX Guidage Régional (bêta)</StrongComponent> : Notre version bêta de FLUX Guidage Régional est en ligne pour le contrôle des prompt régionaux.",
|
||||
"Autres améliorations : mise en file d'attente par lots plus rapide, meilleur redimensionnement, sélecteur de couleurs amélioré et nœuds de métadonnées."
|
||||
],
|
||||
"readReleaseNotes": "Notes de version",
|
||||
"watchUiUpdatesOverview": "Aperçu des mises à jour de l'interface utilisateur"
|
||||
"readReleaseNotes": "Notes de version"
|
||||
},
|
||||
"ui": {
|
||||
"tabs": {
|
||||
@@ -1824,7 +1748,6 @@
|
||||
},
|
||||
"controlLayers": {
|
||||
"newLayerFromImage": "Nouvelle couche à partir de l'image",
|
||||
"sendToGalleryDesc": "Appuyer sur Invoker génère et enregistre une image unique dans votre galerie.",
|
||||
"sendToCanvas": "Envoyer vers la Toile",
|
||||
"globalReferenceImage": "Image de référence globale",
|
||||
"newCanvasFromImage": "Nouvelle Toile à partir de l'image",
|
||||
@@ -1980,7 +1903,6 @@
|
||||
},
|
||||
"bookmark": "Marque-page pour Changement Rapide",
|
||||
"saveLayerToAssets": "Enregistrer la couche dans les ressources",
|
||||
"stagingOnCanvas": "Mise en attente des images sur",
|
||||
"enableTransparencyEffect": "Activer l'effet de transparence",
|
||||
"hidingType": "Masquer {{type}}",
|
||||
"settings": {
|
||||
@@ -2011,11 +1933,6 @@
|
||||
"disableAutoNegative": "Désactiver l'Auto Négatif",
|
||||
"addNegativePrompt": "Ajouter $t(controlLayers.negativePrompt)",
|
||||
"addRegionalGuidance": "Ajouter $t(controlLayers.regionalGuidance)",
|
||||
"controlLayers_withCount_hidden": "Control Layers ({{count}} cachées)",
|
||||
"rasterLayers_withCount_hidden": "Couche de Rastérisation ({{count}} cachées)",
|
||||
"regionalGuidance_withCount_hidden": "Guidage Régional ({{count}} caché)",
|
||||
"rasterLayers_withCount_visible": "Couche de Rastérisation ({{count}})",
|
||||
"inpaintMasks_withCount_visible": "Masques de remplissage ({{count}})",
|
||||
"layer_one": "Couche",
|
||||
"layer_many": "Couches",
|
||||
"layer_other": "Couches",
|
||||
@@ -2065,8 +1982,6 @@
|
||||
"next": "Suivant",
|
||||
"saveToGallery": "Enregistrer dans la galerie"
|
||||
},
|
||||
"viewProgressOnCanvas": "Voir les progrès et les sorties de la scène sur la <Btn>Toile</Btn>.",
|
||||
"sendToCanvasDesc": "Appuyer sur Invoker met en attente votre travail en cours sur la toile.",
|
||||
"mergeVisibleError": "Erreur lors de la fusion des calques visibles",
|
||||
"mergeVisibleOk": "Couches visibles fusionnées",
|
||||
"clearHistory": "Effacer l'historique",
|
||||
@@ -2075,8 +1990,6 @@
|
||||
"duplicate": "Dupliquer",
|
||||
"enableAutoNegative": "Activer l'Auto Négatif",
|
||||
"showHUD": "Afficher HUD",
|
||||
"sendToGallery": "Envoyer à la galerie",
|
||||
"sendingToGallery": "Envoi des générations à la galerie",
|
||||
"disableTransparencyEffect": "Désactiver l'effet de transparence",
|
||||
"HUD": {
|
||||
"entityStatus": {
|
||||
@@ -2093,16 +2006,11 @@
|
||||
"opacity": "Opacité",
|
||||
"savedToGalleryError": "Erreur lors de l'enregistrement dans la galerie",
|
||||
"addInpaintMask": "Ajouter $t(controlLayers.inpaintMask)",
|
||||
"newCanvasSessionDesc": "Cela effacera la toile et tous les paramètres, sauf votre sélection de modèle. Les générations seront mises en attente sur la toile.",
|
||||
"canvas": "Toile",
|
||||
"savedToGalleryOk": "Enregistré dans la galerie",
|
||||
"addPositivePrompt": "Ajouter $t(controlLayers.prompt)",
|
||||
"showProgressOnCanvas": "Afficher la progression sur la Toile",
|
||||
"newGallerySession": "Nouvelle session de galerie",
|
||||
"newCanvasSession": "Nouvelle session de toile",
|
||||
"showingType": "Afficher {{type}}",
|
||||
"viewProgressInViewer": "Voir les progrès et les résultats dans le <Btn>Visionneur d'images</Btn>.",
|
||||
"deletePrompt": "Supprimer le prompt",
|
||||
"addControlLayer": "Ajouter $t(controlLayers.controlLayer)",
|
||||
"global": "Global",
|
||||
"newGlobalReferenceImageOk": "Image de référence globale créée",
|
||||
@@ -2116,16 +2024,6 @@
|
||||
"newRasterLayerError": "Problème de création de couche de rastérisation",
|
||||
"negativePrompt": "Prompt négatif",
|
||||
"weight": "Poids",
|
||||
"globalReferenceImages_withCount_hidden": "Images de référence globales ({{count}} cachées)",
|
||||
"inpaintMasks_withCount_hidden": "Masques de remplissage ({{count}} cachés)",
|
||||
"regionalGuidance_withCount_visible": "Guidage Régional ({{count}})",
|
||||
"globalReferenceImage_withCount_one": "$t(controlLayers.globalReferenceImage)",
|
||||
"globalReferenceImage_withCount_many": "Images de référence globales",
|
||||
"globalReferenceImage_withCount_other": "Images de référence globales",
|
||||
"layer_withCount_one": "Couche {{count}}",
|
||||
"layer_withCount_many": "Couches {{count}}",
|
||||
"layer_withCount_other": "Couches {{count}}",
|
||||
"globalReferenceImages_withCount_visible": "Images de référence globales ({{count}})",
|
||||
"controlMode": {
|
||||
"controlMode": "Mode de contrôle",
|
||||
"balanced": "Équilibré",
|
||||
@@ -2149,18 +2047,14 @@
|
||||
},
|
||||
"fitBboxToLayers": "Ajuster la bounding box aux calques",
|
||||
"regionIsEmpty": "La zone sélectionnée est vide",
|
||||
"controlLayers_withCount_visible": "Couches de contrôle ({{count}})",
|
||||
"cropLayerToBbox": "Rogner la couche selon la bounding box",
|
||||
"sendingToCanvas": "Mise en attente des Générations sur la Toile",
|
||||
"copyToClipboard": "Copier dans le presse-papiers",
|
||||
"regionalGuidance_withCount_one": "$t(controlLayers.regionalGuidance)",
|
||||
"regionalGuidance_withCount_many": "Guidage Régional",
|
||||
"regionalGuidance_withCount_other": "Guidage Régional",
|
||||
"newGallerySessionDesc": "Cela effacera la toile et tous les paramètres, sauf votre sélection de modèle. Les générations seront envoyées à la galerie.",
|
||||
"inpaintMask_withCount_one": "$t(controlLayers.inpaintMask)",
|
||||
"inpaintMask_withCount_many": "Remplir les masques",
|
||||
"inpaintMask_withCount_other": "Remplir les masques",
|
||||
"newImg2ImgCanvasFromImage": "Nouvelle Img2Img à partir de l'image",
|
||||
"bboxOverlay": "Afficher la superposition des Bounding Box",
|
||||
"moveToFront": "Déplacer vers le permier plan",
|
||||
"moveToBack": "Déplacer vers l'arrière plan",
|
||||
@@ -2175,7 +2069,6 @@
|
||||
"inpaintMask": "Masque de remplissage",
|
||||
"deleteReferenceImage": "Supprimer l'image de référence",
|
||||
"addReferenceImage": "Ajouter $t(controlLayers.referenceImage)",
|
||||
"addGlobalReferenceImage": "Ajouter $t(controlLayers.globalReferenceImage)",
|
||||
"removeBookmark": "Supprimer le marque-page",
|
||||
"regionalGuidance": "Guide régional",
|
||||
"regionalReferenceImage": "Image de référence régionale",
|
||||
@@ -2204,16 +2097,12 @@
|
||||
"pointType": "Type de point",
|
||||
"exclude": "Exclure",
|
||||
"process": "Traiter",
|
||||
"reset": "Réinitialiser",
|
||||
"help1": "Sélectionnez un seul objet cible. Ajoutez des points <Bold>Inclure</Bold> et <Bold>Exclure</Bold> pour indiquer quelles parties de la couche font partie de l'objet cible.",
|
||||
"help2": "Commencez par un point <Bold>Inclure</Bold> au sein de l'objet cible. Ajoutez d'autres points pour affiner la sélection. Moins de points produisent généralement de meilleurs résultats.",
|
||||
"help3": "Inversez la sélection pour sélectionner tout sauf l'objet cible."
|
||||
"reset": "Réinitialiser"
|
||||
},
|
||||
"convertRegionalGuidanceTo": "Convertir $t(controlLayers.regionalGuidance) vers",
|
||||
"copyRasterLayerTo": "Copier $t(controlLayers.rasterLayer) vers",
|
||||
"newControlLayer": "Nouveau $t(controlLayers.controlLayer)",
|
||||
"newRegionalGuidance": "Nouveau $t(controlLayers.regionalGuidance)",
|
||||
"replaceCurrent": "Remplacer Actuel",
|
||||
"convertControlLayerTo": "Convertir $t(controlLayers.controlLayer) vers",
|
||||
"convertInpaintMaskTo": "Convertir $t(controlLayers.inpaintMask) vers",
|
||||
"copyControlLayerTo": "Copier $t(controlLayers.controlLayer) vers",
|
||||
@@ -2259,9 +2148,7 @@
|
||||
"pasteToBboxDesc": "Nouvelle couche (dans Bbox)",
|
||||
"pasteToCanvasDesc": "Nouvelle couche (dans la Toile)",
|
||||
"useImage": "Utiliser l'image",
|
||||
"pastedTo": "Collé à {{destination}}",
|
||||
"referenceImageEmptyState": "<UploadButton>Séléctionner une image</UploadButton> ou faites glisser une image depuis la <GalleryButton>galerie</GalleryButton> sur cette couche pour commencer.",
|
||||
"referenceImageGlobal": "Image de référence (Globale)"
|
||||
"referenceImageEmptyState": "<UploadButton>Séléctionner une image</UploadButton> ou faites glisser une image depuis la <GalleryButton>galerie</GalleryButton> sur cette couche pour commencer."
|
||||
},
|
||||
"upscaling": {
|
||||
"exceedsMaxSizeDetails": "La limite maximale d'agrandissement est de {{maxUpscaleDimension}}x{{maxUpscaleDimension}} pixels. Veuillez essayer une image plus petite ou réduire votre sélection d'échelle.",
|
||||
|
||||
@@ -50,8 +50,7 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "גודל תמונה",
|
||||
"gallerySettings": "הגדרות גלריה",
|
||||
"autoSwitchNewImages": "החלף אוטומטית לתמונות חדשות",
|
||||
"noImagesInGallery": "אין תמונות בגלריה"
|
||||
"autoSwitchNewImages": "החלף אוטומטית לתמונות חדשות"
|
||||
},
|
||||
"parameters": {
|
||||
"images": "תמונות",
|
||||
@@ -70,12 +69,10 @@
|
||||
"tileSize": "גודל אריח",
|
||||
"symmetry": "סימטריה",
|
||||
"copyImage": "העתקת תמונה",
|
||||
"downloadImage": "הורדת תמונה",
|
||||
"usePrompt": "שימוש בבקשה",
|
||||
"useSeed": "שימוש בזרע",
|
||||
"useAll": "שימוש בהכל",
|
||||
"info": "פרטים",
|
||||
"showOptionsPanel": "הצג חלונית אפשרויות",
|
||||
"shuffle": "ערבוב",
|
||||
"noiseThreshold": "סף רעש",
|
||||
"perlinNoise": "רעש פרלין",
|
||||
|
||||
@@ -26,7 +26,6 @@
|
||||
"modelManager": "Gestione Modelli",
|
||||
"communityLabel": "Comunità",
|
||||
"advanced": "Avanzate",
|
||||
"imageFailedToLoad": "Impossibile caricare l'immagine",
|
||||
"learnMore": "Per saperne di più",
|
||||
"ipAdapter": "Adattatore IP",
|
||||
"t2iAdapter": "Adattatore T2I",
|
||||
@@ -45,22 +44,18 @@
|
||||
"somethingWentWrong": "Qualcosa è andato storto",
|
||||
"copyError": "Errore $t(gallery.copy)",
|
||||
"input": "Ingresso",
|
||||
"notInstalled": "Non $t(common.installed)",
|
||||
"unknownError": "Errore sconosciuto",
|
||||
"updated": "Aggiornato",
|
||||
"save": "Salva",
|
||||
"created": "Creato",
|
||||
"prevPage": "Pagina precedente",
|
||||
"delete": "Elimina",
|
||||
"orderBy": "Ordina per",
|
||||
"nextPage": "Pagina successiva",
|
||||
"saveAs": "Salva come",
|
||||
"direction": "Direzione",
|
||||
"or": "o",
|
||||
"red": "Rosso",
|
||||
"aboutHeading": "Possiedi il tuo potere creativo",
|
||||
"aboutDesc": "Utilizzi Invoke per lavoro? Guarda qui:",
|
||||
"localSystem": "Sistema locale",
|
||||
"green": "Verde",
|
||||
"blue": "Blu",
|
||||
"alpha": "Alfa",
|
||||
@@ -76,7 +71,6 @@
|
||||
"positivePrompt": "Prompt positivo",
|
||||
"negativePrompt": "Prompt negativo",
|
||||
"selected": "Selezionato",
|
||||
"goTo": "Vai a",
|
||||
"editor": "Editor",
|
||||
"tab": "Scheda",
|
||||
"enabled": "Abilitato",
|
||||
@@ -102,7 +96,6 @@
|
||||
"values": "Valori",
|
||||
"start": "Inizio",
|
||||
"end": "Fine",
|
||||
"resetToDefaults": "Ripristina le impostazioni predefinite",
|
||||
"seed": "Seme",
|
||||
"combinatorial": "Combinatorio",
|
||||
"count": "Quantità",
|
||||
@@ -131,13 +124,19 @@
|
||||
"fullView": "Vista completa",
|
||||
"removeNegativePrompt": "Rimuovi prompt negativo",
|
||||
"addNegativePrompt": "Aggiungi prompt negativo",
|
||||
"selectYourModel": "Seleziona il modello"
|
||||
"selectYourModel": "Seleziona il modello",
|
||||
"goTo": "Vai a",
|
||||
"imageFailedToLoad": "Impossibile caricare l'immagine",
|
||||
"localSystem": "Sistema locale",
|
||||
"notInstalled": "Non $t(common.installed)",
|
||||
"prevPage": "Pagina precedente",
|
||||
"nextPage": "Pagina successiva",
|
||||
"resetToDefaults": "Ripristina impostazioni predefinite"
|
||||
},
|
||||
"gallery": {
|
||||
"galleryImageSize": "Dimensione dell'immagine",
|
||||
"gallerySettings": "Impostazioni della galleria",
|
||||
"autoSwitchNewImages": "Passaggio automatico a nuove immagini",
|
||||
"noImagesInGallery": "Nessuna immagine da visualizzare",
|
||||
"deleteImage_one": "Elimina l'immagine",
|
||||
"deleteImage_many": "Elimina {{count}} immagini",
|
||||
"deleteImage_other": "Elimina {{count}} immagini",
|
||||
@@ -145,7 +144,6 @@
|
||||
"autoAssignBoardOnClick": "Assegna automaticamente la bacheca al clic",
|
||||
"featuresWillReset": "Se elimini questa immagine, quelle funzionalità verranno immediatamente ripristinate.",
|
||||
"loading": "Caricamento in corso",
|
||||
"unableToLoad": "Impossibile caricare la Galleria",
|
||||
"currentlyInUse": "Questa immagine è attualmente utilizzata nelle seguenti funzionalità:",
|
||||
"copy": "Copia",
|
||||
"download": "Scarica",
|
||||
@@ -165,7 +163,6 @@
|
||||
"alwaysShowImageSizeBadge": "Mostra sempre le dimensioni dell'immagine",
|
||||
"openInViewer": "Apri nel visualizzatore",
|
||||
"selectForCompare": "Seleziona per il confronto",
|
||||
"selectAnImageToCompare": "Seleziona un'immagine da confrontare",
|
||||
"slider": "Cursore",
|
||||
"sideBySide": "Fianco a Fianco",
|
||||
"compareImage": "Immagine di confronto",
|
||||
@@ -190,11 +187,8 @@
|
||||
"exitBoardSearch": "Esci da Ricerca bacheca",
|
||||
"exitSearch": "Esci dalla ricerca immagini",
|
||||
"go": "Vai",
|
||||
"jump": "Salta",
|
||||
"move": "Sposta",
|
||||
"gallery": "Galleria",
|
||||
"openViewer": "Apri visualizzatore",
|
||||
"closeViewer": "Chiudi visualizzatore",
|
||||
"imagesTab": "Immagini create e salvate in Invoke.",
|
||||
"assetsTab": "File che hai caricato per usarli nei tuoi progetti.",
|
||||
"boardsSettings": "Impostazioni Bacheche",
|
||||
@@ -206,9 +200,15 @@
|
||||
"deleteVideo_many": "Elimina {{count}} video",
|
||||
"deleteVideo_other": "Elimina {{count}} video",
|
||||
"deleteVideoPermanent": "I video eliminati non possono essere ripristinati.",
|
||||
"noVideoSelected": "Nessun video selezionato",
|
||||
"videos": "Video",
|
||||
"videosTab": "Video creati e salvati in Invoke."
|
||||
"videosTab": "Video creati e salvati in Invoke.",
|
||||
"jump": "Salta",
|
||||
"noVideoSelected": "Nessun video selezionato",
|
||||
"noImagesInGallery": "Nessuna immagine da visualizzare",
|
||||
"unableToLoad": "Impossibile caricare la Galleria",
|
||||
"selectAnImageToCompare": "Seleziona un'immagine da confrontare",
|
||||
"openViewer": "Apri Visualizzatore",
|
||||
"closeViewer": "Chiudi Visualizzatore"
|
||||
},
|
||||
"hotkeys": {
|
||||
"searchHotkeys": "Cerca tasti di scelta rapida",
|
||||
@@ -428,6 +428,14 @@
|
||||
"toggleBbox": {
|
||||
"title": "Attiva/disattiva la visibilità del riquadro di delimitazione",
|
||||
"desc": "Nascondi o mostra il riquadro di delimitazione della generazione"
|
||||
},
|
||||
"setFillColorsToDefault": {
|
||||
"title": "Imposta i colori come predefiniti",
|
||||
"desc": "Imposta i colori degli strumenti correnti sui valori predefiniti."
|
||||
},
|
||||
"toggleFillColor": {
|
||||
"title": "Attiva/disattiva colore di riempimento",
|
||||
"desc": "Attiva/disattiva il colore di riempimento dello strumento corrente."
|
||||
}
|
||||
},
|
||||
"workflows": {
|
||||
@@ -665,9 +673,7 @@
|
||||
"noModelsInstalled": "Nessun modello installato",
|
||||
"main": "Principali",
|
||||
"noModelsInstalledDesc1": "Installa i modelli con",
|
||||
"ipAdapters": "Adattatori IP",
|
||||
"noMatchingModels": "Nessun modello corrispondente",
|
||||
"starterModelsInModelManager": "I modelli iniziali possono essere trovati in Gestione Modelli",
|
||||
"spandrelImageToImage": "Immagine a immagine (Spandrel)",
|
||||
"learnMoreAboutSupportedModels": "Scopri di più sui modelli che supportiamo",
|
||||
"starterBundles": "Pacchetti per iniziare",
|
||||
@@ -706,13 +712,11 @@
|
||||
"urlForbiddenErrorMessage": "Potrebbe essere necessario richiedere l'autorizzazione al sito che distribuisce il modello.",
|
||||
"urlUnauthorizedErrorMessage": "Potrebbe essere necessario configurare un gettone API per accedere a questo modello.",
|
||||
"fileSize": "Dimensione del file",
|
||||
"filterModels": "Filtra i modelli",
|
||||
"modelPickerFallbackNoModelsInstalled": "Nessun modello installato.",
|
||||
"modelPickerFallbackNoModelsInstalled2": "Visita <LinkComponent>Gestione modelli</LinkComponent> per installare i modelli.",
|
||||
"manageModels": "Gestione modelli",
|
||||
"hfTokenReset": "Ripristino del gettone HF",
|
||||
"relatedModels": "Modelli correlati",
|
||||
"showOnlyRelatedModels": "Correlati",
|
||||
"installedModelsCount": "{{installed}} di {{total}} modelli installati.",
|
||||
"allNModelsInstalled": "Tutti i {{count}} modelli installati",
|
||||
"nToInstall": "{{count}} da installare",
|
||||
@@ -728,14 +732,18 @@
|
||||
"recommendedModels": "Modelli consigliati",
|
||||
"exploreStarter": "Oppure sfoglia tutti i modelli iniziali disponibili",
|
||||
"welcome": "Benvenuti in Gestione Modelli",
|
||||
"quickStart": "Pacchetti di avvio rapido",
|
||||
"bundleDescription": "Ogni pacchetto include modelli essenziali per ogni famiglia di modelli e modelli base selezionati per iniziare.",
|
||||
"browseAll": "Oppure scopri tutti i modelli disponibili:"
|
||||
"quickStart": "Pacchetti di avvio rapido",
|
||||
"browseAll": "Oppure sfoglia tutti i modelli disponibili:"
|
||||
},
|
||||
"launchpadTab": "Rampa di lancio",
|
||||
"installBundle": "Installa pacchetto",
|
||||
"installBundleMsg1": "Vuoi davvero installare il pacchetto {{bundleName}}?",
|
||||
"installBundleMsg2": "Questo pacchetto installerà i seguenti {{count}} modelli:"
|
||||
"installBundleMsg2": "Questo pacchetto installerà i seguenti {{count}} modelli:",
|
||||
"filterModels": "Filtra i modelli",
|
||||
"ipAdapters": "Adattatori IP",
|
||||
"showOnlyRelatedModels": "Correlati",
|
||||
"starterModelsInModelManager": "I modelli di avvio possono essere trovati in Gestione Modelli"
|
||||
},
|
||||
"parameters": {
|
||||
"images": "Immagini",
|
||||
@@ -757,12 +765,10 @@
|
||||
"scaledHeight": "Altezza scalata",
|
||||
"infillMethod": "Metodo di riempimento",
|
||||
"tileSize": "Dimensione piastrella",
|
||||
"downloadImage": "Scarica l'immagine",
|
||||
"usePrompt": "Usa Prompt",
|
||||
"useSeed": "Usa Seme",
|
||||
"useAll": "Usa Tutto",
|
||||
"info": "Informazioni",
|
||||
"showOptionsPanel": "Mostra il pannello laterale (O o T)",
|
||||
"general": "Generale",
|
||||
"denoisingStrength": "Forza di riduzione del rumore",
|
||||
"copyImage": "Copia immagine",
|
||||
@@ -789,10 +795,6 @@
|
||||
"missingNodeTemplate": "Modello di nodo mancante",
|
||||
"missingInputForField": "ingresso mancante",
|
||||
"missingFieldTemplate": "Modello di campo mancante",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), altezza riquadro è {{height}}",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), larghezza riquadro è {{width}}",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), larghezza del riquadro scalato è {{width}}",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), altezza del riquadro scalato è {{height}}",
|
||||
"noT5EncoderModelSelected": "Nessun modello di encoder T5 selezionato per la generazione con FLUX",
|
||||
"noCLIPEmbedModelSelected": "Nessun modello CLIP Embed selezionato per la generazione con FLUX",
|
||||
"noFLUXVAEModelSelected": "Nessun modello VAE selezionato per la generazione con FLUX",
|
||||
@@ -808,7 +810,6 @@
|
||||
"collectionStringTooLong": "troppo lungo, massimo {{maxLength}}",
|
||||
"batchNodeNotConnected": "Nodo Lotto non connesso: {{label}}",
|
||||
"batchNodeEmptyCollection": "Alcuni nodi lotto hanno raccolte vuote",
|
||||
"emptyBatches": "lotti vuoti",
|
||||
"batchNodeCollectionSizeMismatch": "Le dimensioni della raccolta nel Lotto {{batchGroupId}} non corrispondono",
|
||||
"collectionStringTooShort": "troppo corto, minimo {{minLength}}",
|
||||
"collectionNumberNotMultipleOf": "{{value}} non è multiplo di {{multipleOf}}",
|
||||
@@ -826,7 +827,12 @@
|
||||
"promptExpansionPending": "Espansione del prompt in corso",
|
||||
"noStartingFrameImage": "Nessuna immagine del fotogramma iniziale",
|
||||
"videoIsDisabled": "La generazione di video non è abilitata per gli account {{accountType}}.",
|
||||
"incompatibleLoRAs": "Aggiunti LoRA incompatibili"
|
||||
"incompatibleLoRAs": "Aggiunti LoRA incompatibili",
|
||||
"emptyBatches": "lotti vuoti",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), la larghezza del riquadro è {{width}}",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), l'altezza del riquadro è {{height}}",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), la larghezza ridimensionata del riquadro è {{width}}",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), l'altezza ridimensionata del riquadro è {{height}}"
|
||||
},
|
||||
"useCpuNoise": "Usa la CPU per generare rumore",
|
||||
"iterations": "Iterazioni",
|
||||
@@ -867,7 +873,9 @@
|
||||
"videoActions": "Azioni video",
|
||||
"sendToVideo": "Invia al Video",
|
||||
"video": "Video",
|
||||
"resolution": "Risoluzione"
|
||||
"resolution": "Risoluzione",
|
||||
"downloadImage": "Scarica l'immagine",
|
||||
"showOptionsPanel": "Mostra pannello laterale (O o T)"
|
||||
},
|
||||
"settings": {
|
||||
"models": "Modelli",
|
||||
@@ -904,10 +912,10 @@
|
||||
"informationalPopoversDisabledDesc": "I testi informativi a comparsa sono disabilitati. Attivali nelle impostazioni.",
|
||||
"confirmOnNewSession": "Conferma su nuova sessione",
|
||||
"enableModelDescriptions": "Abilita le descrizioni dei modelli nei menu a discesa",
|
||||
"modelDescriptionsDisabled": "Descrizioni dei modelli nei menu a discesa disabilitate",
|
||||
"modelDescriptionsDisabledDesc": "Le descrizioni dei modelli nei menu a discesa sono state disabilitate. Abilitale nelle Impostazioni.",
|
||||
"showDetailedInvocationProgress": "Mostra dettagli avanzamento",
|
||||
"enableHighlightFocusedRegions": "Evidenzia le regioni interessate"
|
||||
"enableHighlightFocusedRegions": "Evidenzia le regioni interessate",
|
||||
"modelDescriptionsDisabled": "Descrizioni dei modelli nei menu a discesa disabilitate",
|
||||
"modelDescriptionsDisabledDesc": "Le descrizioni dei modelli nei menu a discesa sono state disattivate. Abilitale nelle Impostazioni."
|
||||
},
|
||||
"toast": {
|
||||
"uploadFailed": "Caricamento fallito",
|
||||
@@ -928,10 +936,7 @@
|
||||
"addedToBoard": "Aggiunto alle risorse della bacheca {{name}}",
|
||||
"modelAddedSimple": "Modello aggiunto alla Coda",
|
||||
"imageUploadFailed": "Caricamento immagine non riuscito",
|
||||
"setControlImage": "Imposta come immagine di controllo",
|
||||
"setNodeField": "Imposta come campo nodo",
|
||||
"workflowLoaded": "Flusso di lavoro caricato",
|
||||
"invalidUpload": "Caricamento non valido",
|
||||
"problemDeletingWorkflow": "Problema durante l'eliminazione del flusso di lavoro",
|
||||
"workflowDeleted": "Flusso di lavoro eliminato",
|
||||
"problemRetrievingWorkflow": "Problema nel recupero del flusso di lavoro",
|
||||
@@ -951,7 +956,6 @@
|
||||
"outOfMemoryErrorDesc": "Le impostazioni della generazione attuale superano la capacità del sistema. Modifica le impostazioni e riprova.",
|
||||
"importFailed": "Importazione non riuscita",
|
||||
"importSuccessful": "Importazione riuscita",
|
||||
"layerSavedToAssets": "Livello salvato nelle risorse",
|
||||
"problemSavingLayer": "Impossibile salvare il livello",
|
||||
"unableToLoadImage": "Impossibile caricare l'immagine",
|
||||
"problemCopyingLayer": "Impossibile copiare il livello",
|
||||
@@ -960,16 +964,10 @@
|
||||
"unableToLoadStylePreset": "Impossibile caricare lo stile predefinito",
|
||||
"stylePresetLoaded": "Stile predefinito caricato",
|
||||
"unableToLoadImageMetadata": "Impossibile caricare i metadati dell'immagine",
|
||||
"imageSaved": "Immagine salvata",
|
||||
"imageSavingFailed": "Salvataggio dell'immagine non riuscito",
|
||||
"layerCopiedToClipboard": "Livello copiato negli appunti",
|
||||
"imageNotLoadedDesc": "Impossibile trovare l'immagine",
|
||||
"linkCopied": "Collegamento copiato",
|
||||
"addedToUncategorized": "Aggiunto alle risorse della bacheca $t(boards.uncategorized)",
|
||||
"imagesWillBeAddedTo": "Le immagini caricate verranno aggiunte alle risorse della bacheca {{boardName}}.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_one": "Devi caricare al massimo 1 immagine PNG, JPEG o WEBP.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_many": "Devi caricare al massimo {{count}} immagini PNG, JPEG o WEBP.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "Devi caricare al massimo {{count}} immagini PNG, JPEG o WEBP.",
|
||||
"outOfMemoryErrorDescLocal": "Segui la nostra <LinkComponent>guida per bassa VRAM</LinkComponent> per ridurre gli OOM.",
|
||||
"pasteFailed": "Incolla non riuscita",
|
||||
"pasteSuccess": "Incollato su {{destination}}",
|
||||
@@ -982,33 +980,43 @@
|
||||
"workflowUnpublished": "Flusso di lavoro non pubblicato",
|
||||
"chatGPT4oIncompatibleGenerationMode": "ChatGPT 4o supporta solo la conversione da testo a immagine e da immagine a immagine. Utilizza altri modelli per le attività di Inpainting e Outpainting.",
|
||||
"imagenIncompatibleGenerationMode": "Google {{model}} supporta solo la generazione da testo a immagine. Utilizza altri modelli per le attività di conversione da immagine a immagine, inpainting e outpainting.",
|
||||
"noRasterLayers": "Nessun livello raster trovato",
|
||||
"noRasterLayersDesc": "Crea almeno un livello raster da esportare in PSD",
|
||||
"noActiveRasterLayers": "Nessun livello raster attivo",
|
||||
"noActiveRasterLayersDesc": "Abilitare almeno un livello raster da esportare in PSD",
|
||||
"noVisibleRasterLayers": "Nessun livello raster visibile",
|
||||
"noVisibleRasterLayersDesc": "Abilitare almeno un livello raster da esportare in PSD",
|
||||
"invalidCanvasDimensions": "Dimensioni della tela non valide",
|
||||
"canvasTooLarge": "Tela troppo grande",
|
||||
"canvasTooLargeDesc": "Le dimensioni della tela superano le dimensioni massime consentite per l'esportazione in formato PSD. Riduci la larghezza e l'altezza totali della tela e riprova.",
|
||||
"failedToProcessLayers": "Impossibile elaborare i livelli",
|
||||
"psdExportSuccess": "Esportazione PSD completata",
|
||||
"psdExportSuccessDesc": "Esportazione riuscita di {{count}} livelli nel file PSD",
|
||||
"problemExportingPSD": "Problema durante l'esportazione PSD",
|
||||
"noValidLayerAdapters": "Nessun adattatore di livello valido trovato",
|
||||
"fluxKontextIncompatibleGenerationMode": "FLUX Kontext non supporta la generazione di immagini posizionate sulla tela. Riprova utilizzando la sezione Immagine di riferimento e disattiva tutti i livelli raster.",
|
||||
"canvasManagerNotAvailable": "Gestione tela non disponibile",
|
||||
"promptExpansionFailed": "Abbiamo riscontrato un problema. Riprova a eseguire l'espansione del prompt.",
|
||||
"uploadAndPromptGenerationFailed": "Impossibile caricare l'immagine e generare il prompt",
|
||||
"promptGenerationStarted": "Generazione del prompt avviata",
|
||||
"invalidBboxDesc": "Il riquadro di delimitazione non ha dimensioni valide",
|
||||
"invalidBbox": "Riquadro di delimitazione non valido",
|
||||
"noInpaintMaskSelectedDesc": "Seleziona una maschera di inpaint da invertire",
|
||||
"noInpaintMaskSelected": "Nessuna maschera di inpaint selezionata",
|
||||
"noVisibleMasksDesc": "Crea o abilita almeno una maschera inpaint da invertire",
|
||||
"noVisibleMasks": "Nessuna maschera visibile",
|
||||
"maskInvertFailed": "Impossibile invertire la maschera",
|
||||
"maskInverted": "Maschera invertita"
|
||||
"maskInverted": "Maschera invertita",
|
||||
"uploadFailedInvalidUploadDesc_withCount_one": "Deve essere presente al massimo 1 immagine PNG, JPEG o WEBP.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_many": "Devono essere presenti al massimo {{count}} immagini PNG, JPEG o WEBP.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "Devono essere presenti al massimo {{count}} immagini PNG, JPEG o WEBP.",
|
||||
"imageNotLoadedDesc": "Impossibile trovare l'immagine",
|
||||
"imageSaved": "Immagine salvata",
|
||||
"imageSavingFailed": "Salvataggio dell'immagine non riuscito",
|
||||
"invalidUpload": "Caricamento non valido",
|
||||
"layerSavedToAssets": "Livello salvato nelle risorse",
|
||||
"noRasterLayers": "Nessun livello raster trovato",
|
||||
"noRasterLayersDesc": "Crea almeno un livello raster da esportare in PSD",
|
||||
"noActiveRasterLayers": "Nessun livello raster attivo",
|
||||
"noActiveRasterLayersDesc": "Abilita almeno un livello raster da esportare in PSD",
|
||||
"failedToProcessLayers": "Impossibile elaborare i livelli",
|
||||
"noValidLayerAdapters": "Nessun adattatore di livello valido trovato",
|
||||
"setControlImage": "Imposta come immagine di controllo",
|
||||
"setNodeField": "Imposta come campo nodo",
|
||||
"noInpaintMaskSelected": "Nessuna maschera di inpaint selezionata",
|
||||
"noInpaintMaskSelectedDesc": "Seleziona una maschera di inpaint da invertire",
|
||||
"invalidBbox": "Riquadro di delimitazione non valido",
|
||||
"invalidBboxDesc": "Il riquadro di delimitazione non ha dimensioni valide"
|
||||
},
|
||||
"accessibility": {
|
||||
"invokeProgressBar": "Barra di avanzamento generazione",
|
||||
@@ -1028,8 +1036,6 @@
|
||||
},
|
||||
"nodes": {
|
||||
"zoomOutNodes": "Rimpicciolire",
|
||||
"hideLegendNodes": "Nascondi la legenda del tipo di campo",
|
||||
"showLegendNodes": "Mostra legenda del tipo di campo",
|
||||
"hideMinimapnodes": "Nascondi minimappa",
|
||||
"showMinimapnodes": "Mostra minimappa",
|
||||
"zoomInNodes": "Ingrandire",
|
||||
@@ -1055,27 +1061,22 @@
|
||||
"workflowSettings": "Impostazioni Editor del flusso di lavoro",
|
||||
"colorCodeEdges": "Bordi con codice colore",
|
||||
"noOutputRecorded": "Nessun output registrato",
|
||||
"noFieldsLinearview": "Nessun campo aggiunto alla vista lineare",
|
||||
"removeLinearView": "Rimuovi dalla vista lineare",
|
||||
"workflowDescription": "Breve descrizione",
|
||||
"workflowContact": "Contatto",
|
||||
"workflowVersion": "Versione",
|
||||
"workflow": "Flusso di lavoro",
|
||||
"noWorkflow": "Nessun flusso di lavoro",
|
||||
"workflowTags": "Tag",
|
||||
"workflowTags": "Etichette",
|
||||
"workflowValidation": "Errore di convalida del flusso di lavoro",
|
||||
"workflowAuthor": "Autore",
|
||||
"workflowName": "Nome",
|
||||
"workflowNotes": "Note",
|
||||
"versionUnknown": " Versione sconosciuta",
|
||||
"unableToValidateWorkflow": "Impossibile convalidare il flusso di lavoro",
|
||||
"updateApp": "Aggiorna Applicazione",
|
||||
"unableToLoadWorkflow": "Impossibile caricare il flusso di lavoro",
|
||||
"updateNode": "Aggiorna nodo",
|
||||
"version": "Versione",
|
||||
"notes": "Note",
|
||||
"problemSettingTitle": "Problema nell'impostazione del titolo",
|
||||
"unknownTemplate": "Modello sconosciuto",
|
||||
"nodeType": "Tipo di nodo",
|
||||
"notesDescription": "Aggiunge note sul tuo flusso di lavoro",
|
||||
"unknownField": "Campo sconosciuto",
|
||||
@@ -1094,10 +1095,9 @@
|
||||
"cannotConnectInputToInput": "Impossibile collegare ingresso a ingresso",
|
||||
"cannotConnectOutputToOutput": "Impossibile collegare uscita ad uscita",
|
||||
"cannotConnectToSelf": "Impossibile connettersi a se stesso",
|
||||
"mismatchedVersion": "Nodo non valido: il nodo {{node}} di tipo {{type}} ha una versione non corrispondente (provare ad aggiornare?)",
|
||||
"loadingNodes": "Caricamento nodi...",
|
||||
"enum": "Enumeratore",
|
||||
"float": "In virgola mobile",
|
||||
"float": "Decimale",
|
||||
"currentImageDescription": "Visualizza l'immagine corrente nell'editor dei nodi",
|
||||
"fieldTypesMustMatch": "I tipi di campo devono corrispondere",
|
||||
"edge": "Collegamento",
|
||||
@@ -1111,7 +1111,6 @@
|
||||
"unableToUpdateNodes_one": "Impossibile aggiornare {{count}} nodo",
|
||||
"unableToUpdateNodes_many": "Impossibile aggiornare {{count}} nodi",
|
||||
"unableToUpdateNodes_other": "Impossibile aggiornare {{count}} nodi",
|
||||
"addLinearView": "Aggiungi alla vista Lineare",
|
||||
"unknownErrorValidatingWorkflow": "Errore sconosciuto durante la convalida del flusso di lavoro",
|
||||
"collectionFieldType": "{{name}} (Raccolta)",
|
||||
"collectionOrScalarFieldType": "{{name}} (Singola o Raccolta)",
|
||||
@@ -1133,7 +1132,6 @@
|
||||
"targetNodeDoesNotExist": "Connessione non valida: il nodo di destinazione/input {{node}} non esiste",
|
||||
"unknownFieldType": "$t(nodes.unknownField) tipo: {{type}}",
|
||||
"deletedInvalidEdge": "Eliminata connessione non valida {{source}} -> {{target}}",
|
||||
"unknownInput": "Input sconosciuto: {{name}}",
|
||||
"prototypeDesc": "Questa invocazione è un prototipo. Potrebbe subire modifiche sostanziali durante gli aggiornamenti dell'app e potrebbe essere rimossa in qualsiasi momento.",
|
||||
"betaDesc": "Questa invocazione è in versione beta. Fino a quando non sarà stabile, potrebbe subire modifiche importanti durante gli aggiornamenti dell'app. Abbiamo intenzione di supportare questa invocazione a lungo termine.",
|
||||
"newWorkflow": "Nuovo flusso di lavoro",
|
||||
@@ -1144,7 +1142,6 @@
|
||||
"clearWorkflow": "Cancella il flusso di lavoro",
|
||||
"clearWorkflowDesc2": "Il tuo flusso di lavoro attuale presenta modifiche non salvate.",
|
||||
"viewMode": "Usa la vista lineare",
|
||||
"reorderLinearView": "Riordina la vista lineare",
|
||||
"editMode": "Modifica nell'editor del flusso di lavoro",
|
||||
"resetToDefaultValue": "Ripristina il valore predefinito",
|
||||
"noFieldsViewMode": "Questo flusso di lavoro non ha campi selezionati da visualizzare. Visualizza il flusso di lavoro completo per configurare i valori.",
|
||||
@@ -1168,18 +1165,14 @@
|
||||
"specialDesc": "Questa invocazione comporta una gestione speciale nell'applicazione. Ad esempio, i nodi Lotto vengono utilizzati per mettere in coda più grafici da un singolo flusso di lavoro.",
|
||||
"internalDesc": "Questa invocazione è utilizzata internamente da Invoke. Potrebbe subire modifiche significative durante gli aggiornamenti dell'app e potrebbe essere rimossa in qualsiasi momento.",
|
||||
"addItem": "Aggiungi elemento",
|
||||
"generateValues": "Genera valori",
|
||||
"generatorNoValues": "vuoto",
|
||||
"linearDistribution": "Distribuzione lineare",
|
||||
"parseString": "Analizza stringa",
|
||||
"splitOn": "Diviso su",
|
||||
"noBatchGroup": "nessun gruppo",
|
||||
"generatorLoading": "caricamento",
|
||||
"generatorLoadFromFile": "Carica da file",
|
||||
"dynamicPromptsRandom": "Prompt dinamici (casuali)",
|
||||
"dynamicPromptsCombinatorial": "Prompt dinamici (combinatori)",
|
||||
"floatRangeGenerator": "Generatore di intervalli di numeri in virgola mobile",
|
||||
"integerRangeGenerator": "Generatore di intervalli di numeri interi",
|
||||
"uniformRandomDistribution": "Distribuzione casuale uniforme",
|
||||
"generatorNRandomValues_one": "{{count}} valore casuale",
|
||||
"generatorNRandomValues_many": "{{count}} valori casuali",
|
||||
@@ -1190,7 +1183,6 @@
|
||||
"loadWorkflowDesc2": "Il flusso di lavoro corrente presenta modifiche non salvate.",
|
||||
"downloadWorkflowError": "Errore durante lo scaricamento del flusso di lavoro",
|
||||
"deletedMissingNodeFieldFormElement": "Campo modulo mancante eliminato: nodo {{nodeId}} campo {{fieldName}}",
|
||||
"loadingTemplates": "Caricamento {{name}}",
|
||||
"unableToUpdateNode": "Aggiornamento del nodo non riuscito: nodo {{node}} di tipo {{type}} (potrebbe essere necessario eliminarlo e ricrearlo)",
|
||||
"description": "Descrizione",
|
||||
"generatorImagesCategory": "Categoria",
|
||||
@@ -1218,7 +1210,23 @@
|
||||
"alignmentUL": "In alto a sinistra",
|
||||
"alignmentDL": "In basso a sinistra",
|
||||
"alignmentUR": "In alto a destra"
|
||||
}
|
||||
},
|
||||
"generatorLoading": "caricamento",
|
||||
"addLinearView": "Aggiungi alla vista lineare",
|
||||
"hideLegendNodes": "Nascondi legenda tipo di campo",
|
||||
"mismatchedVersion": "Nodo non valido: il nodo {{node}} di tipo {{type}} ha una versione non corrispondente (provare ad aggiornare?)",
|
||||
"noFieldsLinearview": "Nessun campo aggiunto alla vista lineare",
|
||||
"removeLinearView": "Rimuovi dalla vista lineare",
|
||||
"reorderLinearView": "Riordina vista lineare",
|
||||
"showLegendNodes": "Mostra legenda tipo di campo",
|
||||
"unableToLoadWorkflow": "Impossibile caricare il flusso di lavoro",
|
||||
"unknownTemplate": "Modello sconosciuto",
|
||||
"unknownInput": "Input sconosciuto: {{name}}",
|
||||
"loadingTemplates": "Caricamento in corso {{name}}",
|
||||
"versionUnknown": " Versione sconosciuta",
|
||||
"generateValues": "Genera valori",
|
||||
"floatRangeGenerator": "Generatore di intervallo di numeri decimali",
|
||||
"integerRangeGenerator": "Generatore di intervallo di numeri interi"
|
||||
},
|
||||
"boards": {
|
||||
"autoAddBoard": "Aggiungi automaticamente bacheca",
|
||||
@@ -1260,20 +1268,20 @@
|
||||
"shared": "Bacheche condivise",
|
||||
"addPrivateBoard": "Aggiungi una Bacheca Privata",
|
||||
"noBoards": "Nessuna bacheca {{boardType}}",
|
||||
"hideBoards": "Nascondi bacheche",
|
||||
"viewBoards": "Visualizza bacheche",
|
||||
"deletedPrivateBoardsCannotbeRestored": "Le bacheche e le immagini eliminate non possono essere ripristinate. Selezionando \"Elimina solo bacheca\", le immagini verranno spostate in uno stato privato e non categorizzato per l'autore dell'immagine.",
|
||||
"updateBoardError": "Errore durante l'aggiornamento della bacheca",
|
||||
"uncategorizedImages": "Immagini non categorizzate",
|
||||
"deleteAllUncategorizedImages": "Elimina tutte le immagini non categorizzate",
|
||||
"deletedImagesCannotBeRestored": "Le immagini eliminate non possono essere ripristinate.",
|
||||
"locateInGalery": "Trova nella Galleria",
|
||||
"movingVideosToBoard_one": "Spostamento di {{count}} video sulla bacheca:",
|
||||
"movingVideosToBoard_many": "Spostamento di {{count}} video sulla bacheca:",
|
||||
"movingVideosToBoard_other": "Spostamento di {{count}} video sulla bacheca:",
|
||||
"videosWithCount_one": "{{count}} video",
|
||||
"videosWithCount_many": "{{count}} video",
|
||||
"videosWithCount_other": "{{count}} video"
|
||||
"videosWithCount_other": "{{count}} video",
|
||||
"deletedImagesCannotBeRestored": "Le immagini eliminate non possono essere ripristinate.",
|
||||
"hideBoards": "Nascondi bacheche",
|
||||
"viewBoards": "Visualizza le bacheche"
|
||||
},
|
||||
"queue": {
|
||||
"queueFront": "Aggiungi all'inizio della coda",
|
||||
@@ -1330,7 +1338,6 @@
|
||||
"clearQueueAlertDialog2": "Sei sicuro di voler cancellare la coda?",
|
||||
"item": "Elemento",
|
||||
"graphFailedToQueue": "Impossibile mettere in coda il grafico",
|
||||
"batchFieldValues": "Valori Campi Lotto",
|
||||
"time": "Tempo",
|
||||
"openQueue": "Apri coda",
|
||||
"iterations_one": "Iterazione",
|
||||
@@ -1343,7 +1350,7 @@
|
||||
"generations_many": "Generazioni",
|
||||
"generations_other": "Generazioni",
|
||||
"origin": "Origine",
|
||||
"destination": "Destinazione",
|
||||
"destination": "Dest",
|
||||
"upscaling": "Ampliamento",
|
||||
"canvas": "Tela",
|
||||
"workflows": "Flussi di lavoro",
|
||||
@@ -1359,21 +1366,28 @@
|
||||
"retryItem": "Riesegui elemento",
|
||||
"retryFailed": "Problema riesecuzione elemento",
|
||||
"credits": "Crediti",
|
||||
"cancelAllExceptCurrent": "Annulla tutto tranne quello corrente"
|
||||
"cancelAllExceptCurrent": "Annulla tutto tranne quello corrente",
|
||||
"sortColumn": "Ordina colonna",
|
||||
"sortBy": "Ordina per {{column}}",
|
||||
"sortOrderAscending": "Ascendente",
|
||||
"sortOrderDescending": "Discendente",
|
||||
"createdAt": "Creato",
|
||||
"completedAt": "Completato",
|
||||
"batchFieldValues": "Valori del campo Lotto"
|
||||
},
|
||||
"models": {
|
||||
"noMatchingModels": "Nessun modello corrispondente",
|
||||
"loading": "caricamento",
|
||||
"noMatchingLoRAs": "Nessun LoRA corrispondente",
|
||||
"noModelsAvailable": "Nessun modello disponibile",
|
||||
"selectModel": "Seleziona un modello",
|
||||
"noRefinerModelsInstalled": "Nessun modello affinatore SDXL installato",
|
||||
"noLoRAsInstalled": "Nessun LoRA installato",
|
||||
"addLora": "Aggiungi LoRA",
|
||||
"defaultVAE": "VAE predefinito",
|
||||
"concepts": "Concetti",
|
||||
"lora": "LoRA",
|
||||
"noCompatibleLoRAs": "Nessun LoRA compatibile"
|
||||
"noCompatibleLoRAs": "Nessun LoRA compatibile",
|
||||
"noMatchingLoRAs": "Nessun LoRA corrispondente",
|
||||
"noLoRAsInstalled": "Nessun LoRA installato"
|
||||
},
|
||||
"invocationCache": {
|
||||
"disable": "Disabilita",
|
||||
@@ -1851,19 +1865,19 @@
|
||||
"scheduler": "Campionatore",
|
||||
"noModelsAvailable": "Nessun modello disponibile",
|
||||
"denoisingStrength": "Forza di riduzione del rumore",
|
||||
"concatPromptStyle": "Collega Prompt & Stile",
|
||||
"loading": "Caricamento...",
|
||||
"steps": "Passi",
|
||||
"refinerStart": "Inizio Affinamento",
|
||||
"cfgScale": "Scala CFG",
|
||||
"negStylePrompt": "Prompt Stile negativo",
|
||||
"refiner": "Affinatore",
|
||||
"negAestheticScore": "Punteggio estetico negativo",
|
||||
"refinermodel": "Modello Affinatore",
|
||||
"posAestheticScore": "Punteggio estetico positivo",
|
||||
"posStylePrompt": "Prompt Stile positivo",
|
||||
"freePromptStyle": "Prompt di stile manuale",
|
||||
"refinerSteps": "Passi Affinamento"
|
||||
"refinerSteps": "Passi Affinamento",
|
||||
"concatPromptStyle": "Collegamento di prompt e stile",
|
||||
"freePromptStyle": "Prompt manuale Stile",
|
||||
"negStylePrompt": "Prompt di stile negativo",
|
||||
"posStylePrompt": "Prompt di stile positivo"
|
||||
},
|
||||
"metadata": {
|
||||
"positivePrompt": "Prompt positivo",
|
||||
@@ -1890,8 +1904,6 @@
|
||||
"allPrompts": "Tutti i prompt",
|
||||
"imageDimensions": "Dimensioni dell'immagine",
|
||||
"parameterSet": "Parametro {{parameter}} impostato",
|
||||
"parsingFailed": "Analisi non riuscita",
|
||||
"recallParameter": "Richiama {{label}}",
|
||||
"canvasV2Metadata": "Livelli Tela",
|
||||
"guidance": "Guida",
|
||||
"seamlessXAxis": "Asse X senza giunte",
|
||||
@@ -1902,36 +1914,33 @@
|
||||
"videoModel": "Modello",
|
||||
"videoDuration": "Durata",
|
||||
"videoAspectRatio": "Proporzioni",
|
||||
"videoResolution": "Risoluzione"
|
||||
"videoResolution": "Risoluzione",
|
||||
"parsingFailed": "Analisi non riuscita",
|
||||
"recallParameter": "Richiama {{label}}"
|
||||
},
|
||||
"hrf": {
|
||||
"enableHrf": "Abilita Correzione Alta Risoluzione",
|
||||
"upscaleMethod": "Metodo di ampliamento",
|
||||
"metadata": {
|
||||
"strength": "Forza della Correzione Alta Risoluzione",
|
||||
"enabled": "Correzione Alta Risoluzione Abilitata",
|
||||
"method": "Metodo della Correzione Alta Risoluzione"
|
||||
},
|
||||
"hrf": "Correzione Alta Risoluzione"
|
||||
"hrf": "Correzione Alta Risoluzione",
|
||||
"enableHrf": "Abilita correzione ad alta risoluzione",
|
||||
"upscaleMethod": "Metodo di ampliamento"
|
||||
},
|
||||
"workflows": {
|
||||
"saveWorkflowAs": "Salva flusso di lavoro come",
|
||||
"workflowEditorMenu": "Menu dell'editor del flusso di lavoro",
|
||||
"workflowName": "Nome del flusso di lavoro",
|
||||
"saveWorkflow": "Salva flusso di lavoro",
|
||||
"openWorkflow": "Apri flusso di lavoro",
|
||||
"clearWorkflowSearchFilter": "Cancella il filtro di ricerca del flusso di lavoro",
|
||||
"workflowLibrary": "Libreria flussi di lavoro",
|
||||
"workflowSaved": "Flusso di lavoro salvato",
|
||||
"unnamedWorkflow": "Flusso di lavoro senza nome",
|
||||
"savingWorkflow": "Salvataggio del flusso di lavoro...",
|
||||
"problemLoading": "Problema durante il caricamento dei flussi di lavoro",
|
||||
"loading": "Caricamento dei flussi di lavoro",
|
||||
"searchWorkflows": "Cerca flussi di lavoro",
|
||||
"problemSavingWorkflow": "Problema durante il salvataggio del flusso di lavoro",
|
||||
"deleteWorkflow": "Elimina flusso di lavoro",
|
||||
"workflows": "Flussi di lavoro",
|
||||
"noDescription": "Nessuna descrizione",
|
||||
"newWorkflowCreated": "Nuovo flusso di lavoro creato",
|
||||
"downloadWorkflow": "Salva su file",
|
||||
"uploadWorkflow": "Carica da file",
|
||||
@@ -1948,9 +1957,6 @@
|
||||
"loadWorkflow": "$t(common.load) Flusso di lavoro",
|
||||
"autoLayout": "Schema automatico",
|
||||
"loadFromGraph": "Carica il flusso di lavoro dal grafico",
|
||||
"userWorkflows": "Flussi di lavoro utente",
|
||||
"projectWorkflows": "Flussi di lavoro del progetto",
|
||||
"defaultWorkflows": "Flussi di lavoro predefiniti",
|
||||
"chooseWorkflowFromLibrary": "Scegli il flusso di lavoro dalla libreria",
|
||||
"deleteWorkflow2": "Vuoi davvero eliminare questo flusso di lavoro? Questa operazione non può essere annullata.",
|
||||
"edit": "Modifica",
|
||||
@@ -1958,7 +1964,6 @@
|
||||
"copyShareLink": "Copia Condividi Link",
|
||||
"copyShareLinkForWorkflow": "Copia Condividi Link del Flusso di lavoro",
|
||||
"delete": "Elimina",
|
||||
"openLibrary": "Apri la libreria",
|
||||
"builder": {
|
||||
"resetAllNodeFields": "Reimposta tutti i campi del nodo",
|
||||
"row": "Riga",
|
||||
@@ -1980,9 +1985,7 @@
|
||||
"singleLine": "Linea singola",
|
||||
"multiLine": "Linea Multipla",
|
||||
"both": "Entrambi",
|
||||
"emptyRootPlaceholderViewMode": "Fare clic su Modifica per iniziare a creare un modulo per questo flusso di lavoro.",
|
||||
"textPlaceholder": "Testo vuoto",
|
||||
"workflowBuilderAlphaWarning": "Il generatore di flussi di lavoro è attualmente in versione alpha. Potrebbero esserci cambiamenti radicali prima della versione stabile.",
|
||||
"heading": "Intestazione",
|
||||
"divider": "Divisore",
|
||||
"container": "Contenitore",
|
||||
@@ -2029,25 +2032,36 @@
|
||||
"errorWorkflowHasUnpublishableNodes": "Il flusso di lavoro ha nodi di estrazione lotto, generatore o metadati",
|
||||
"showShuffle": "Mostra Mescola",
|
||||
"shuffle": "Mescola",
|
||||
"removeFromForm": "Rimuovi dal modulo"
|
||||
"removeFromForm": "Rimuovi dal modulo",
|
||||
"emptyRootPlaceholderViewMode": "Fare clic su Modifica per iniziare a creare un modulo per questo flusso di lavoro.",
|
||||
"workflowBuilderAlphaWarning": "Il generatore di flussi di lavoro è attualmente in versione alpha. Potrebbero esserci modifiche sostanziali prima della versione stabile."
|
||||
},
|
||||
"loadMore": "Carica altro",
|
||||
"searchPlaceholder": "Cerca per nome, descrizione o etichetta",
|
||||
"filterByTags": "Filtra per etichetta",
|
||||
"shared": "Condiviso",
|
||||
"browseWorkflows": "Sfoglia i flussi di lavoro",
|
||||
"allLoaded": "Tutti i flussi di lavoro caricati",
|
||||
"saveChanges": "Salva modifiche",
|
||||
"yourWorkflows": "I tuoi flussi di lavoro",
|
||||
"recentlyOpened": "Aperto di recente",
|
||||
"workflowThumbnail": "Miniatura del flusso di lavoro",
|
||||
"private": "Privato",
|
||||
"deselectAll": "Deseleziona tutto",
|
||||
"noRecentWorkflows": "Nessun flusso di lavoro recente",
|
||||
"view": "Visualizza",
|
||||
"recommended": "Consigliato per te",
|
||||
"emptyStringPlaceholder": "<stringa vuota>",
|
||||
"published": "Pubblicato"
|
||||
"published": "Pubblicato",
|
||||
"defaultWorkflows": "Flussi di lavoro predefiniti",
|
||||
"userWorkflows": "Flussi di lavoro dell'utente",
|
||||
"projectWorkflows": "Flussi di lavoro del progetto",
|
||||
"allLoaded": "Tutti i flussi di lavoro caricati",
|
||||
"filterByTags": "Filtra per etichetta",
|
||||
"noRecentWorkflows": "Nessun flusso di lavoro recente",
|
||||
"openWorkflow": "Apri flusso di lavoro",
|
||||
"problemLoading": "Problema nel caricamento dei flussi di lavoro",
|
||||
"noDescription": "Nessuna descrizione",
|
||||
"searchWorkflows": "Ricerca flussi di lavoro",
|
||||
"clearWorkflowSearchFilter": "Cancella filtro di ricerca del flusso di lavoro",
|
||||
"openLibrary": "Apri libreria"
|
||||
},
|
||||
"accordions": {
|
||||
"compositing": {
|
||||
@@ -2074,14 +2088,14 @@
|
||||
"addPromptTrigger": "Aggiungi Trigger nel prompt",
|
||||
"noMatchingTriggers": "Nessun Trigger corrispondente",
|
||||
"discard": "Scarta",
|
||||
"insert": "Inserisci",
|
||||
"replace": "Sostituisci",
|
||||
"resultSubtitle": "Scegli come gestire il prompt espanso:",
|
||||
"resultTitle": "Espansione del prompt completata",
|
||||
"expandingPrompt": "Espansione del prompt...",
|
||||
"uploadImageForPromptGeneration": "Carica l'immagine per la generazione del prompt",
|
||||
"expandCurrentPrompt": "Espandi il prompt corrente",
|
||||
"generateFromImage": "Genera prompt dall'immagine"
|
||||
"generateFromImage": "Genera prompt dall'immagine",
|
||||
"resultTitle": "Espansione del prompt completata",
|
||||
"resultSubtitle": "Scegli come gestire il prompt espanso:",
|
||||
"insert": "Inserisci"
|
||||
},
|
||||
"controlLayers": {
|
||||
"addLayer": "Aggiungi Livello",
|
||||
@@ -2090,7 +2104,6 @@
|
||||
"moveForward": "Sposta avanti",
|
||||
"moveBackward": "Sposta indietro",
|
||||
"autoNegative": "Auto Negativo",
|
||||
"deletePrompt": "Cancella il prompt",
|
||||
"rectangle": "Rettangolo",
|
||||
"addPositivePrompt": "Aggiungi $t(controlLayers.prompt)",
|
||||
"addNegativePrompt": "Aggiungi $t(controlLayers.negativePrompt)",
|
||||
@@ -2107,10 +2120,8 @@
|
||||
"newCanvasFromImage": "Nuova tela da immagine",
|
||||
"globalReferenceImage": "Immagine di riferimento Globale",
|
||||
"copyToClipboard": "Copia negli appunti",
|
||||
"sendingToCanvas": "Effettua le generazioni nella Tela",
|
||||
"clearHistory": "Cancella la cronologia",
|
||||
"inpaintMask": "Maschera Inpaint",
|
||||
"sendToGallery": "Invia alla Galleria",
|
||||
"controlLayer": "Livello di Controllo",
|
||||
"rasterLayer_withCount_one": "$t(controlLayers.rasterLayer)",
|
||||
"rasterLayer_withCount_many": "Livelli Raster",
|
||||
@@ -2124,7 +2135,6 @@
|
||||
"addControlLayer": "Aggiungi $t(controlLayers.controlLayer)",
|
||||
"addInpaintMask": "Aggiungi $t(controlLayers.inpaintMask)",
|
||||
"addRegionalGuidance": "Aggiungi $t(controlLayers.regionalGuidance)",
|
||||
"sendToCanvasDesc": "Premendo Invoke il lavoro in corso viene visualizzato sulla tela.",
|
||||
"addRasterLayer": "Aggiungi $t(controlLayers.rasterLayer)",
|
||||
"clearCaches": "Svuota le cache",
|
||||
"regionIsEmpty": "La regione selezionata è vuota",
|
||||
@@ -2145,12 +2155,7 @@
|
||||
"showHUD": "Mostra HUD",
|
||||
"maskFill": "Riempimento maschera",
|
||||
"addReferenceImage": "Aggiungi $t(controlLayers.referenceImage)",
|
||||
"addGlobalReferenceImage": "Aggiungi $t(controlLayers.globalReferenceImage)",
|
||||
"sendingToGallery": "Inviare generazioni alla Galleria",
|
||||
"sendToGalleryDesc": "Premendo Invoke viene generata e salvata un'immagine unica nella tua galleria.",
|
||||
"sendToCanvas": "Invia alla Tela",
|
||||
"viewProgressInViewer": "Visualizza i progressi e i risultati nel <Btn>Visualizzatore immagini</Btn>.",
|
||||
"viewProgressOnCanvas": "Visualizza i progressi e i risultati nella <Btn>Tela</Btn>.",
|
||||
"saveBboxToGallery": "Salva il riquadro di delimitazione nella Galleria",
|
||||
"cropLayerToBbox": "Ritaglia il livello al riquadro di delimitazione",
|
||||
"savedToGalleryError": "Errore durante il salvataggio nella galleria",
|
||||
@@ -2172,9 +2177,6 @@
|
||||
"pullBboxIntoReferenceImageOk": "Contenuto del riquadro inserito nell'immagine di riferimento",
|
||||
"pullBboxIntoLayerOk": "Riquadro caricato nel livello",
|
||||
"pullBboxIntoReferenceImageError": "Problema nell'inserimento del contenuto del riquadro nell'immagine di riferimento",
|
||||
"globalReferenceImage_withCount_one": "$t(controlLayers.globalReferenceImage)",
|
||||
"globalReferenceImage_withCount_many": "Immagini di riferimento Globali",
|
||||
"globalReferenceImage_withCount_other": "Immagini di riferimento Globali",
|
||||
"controlMode": {
|
||||
"balanced": "Bilanciato (consigliato)",
|
||||
"controlMode": "Modalità di controllo",
|
||||
@@ -2185,7 +2187,6 @@
|
||||
"negativePrompt": "Prompt Negativo",
|
||||
"prompt": "Prompt Positivo",
|
||||
"beginEndStepPercentShort": "Inizio/Fine %",
|
||||
"stagingOnCanvas": "Genera immagini nella",
|
||||
"ipAdapterMethod": {
|
||||
"full": "Stile e Composizione",
|
||||
"style": "Stile (semplice)",
|
||||
@@ -2334,8 +2335,6 @@
|
||||
"value": "Valore (HSV)"
|
||||
}
|
||||
},
|
||||
"controlLayers_withCount_hidden": "Livelli di controllo ({{count}} nascosti)",
|
||||
"regionalGuidance_withCount_hidden": "Guida regionale ({{count}} nascosti)",
|
||||
"fill": {
|
||||
"grid": "Griglia",
|
||||
"crosshatch": "Tratteggio incrociato",
|
||||
@@ -2344,28 +2343,18 @@
|
||||
"solid": "Solido",
|
||||
"vertical": "Verticale",
|
||||
"horizontal": "Orizzontale",
|
||||
"diagonal": "Diagonale"
|
||||
"diagonal": "Diagonale",
|
||||
"bgFillColor": "Colore di sfondo",
|
||||
"fgFillColor": "Colore di primo piano"
|
||||
},
|
||||
"rasterLayers_withCount_hidden": "Livelli raster ({{count}} nascosti)",
|
||||
"inpaintMasks_withCount_hidden": "Maschere Inpaint ({{count}} nascoste)",
|
||||
"regionalGuidance_withCount_visible": "Guide regionali ({{count}})",
|
||||
"locked": "Bloccato",
|
||||
"hidingType": "Nascondere {{type}}",
|
||||
"logDebugInfo": "Registro Info Debug",
|
||||
"inpaintMasks_withCount_visible": "Maschere Inpaint ({{count}})",
|
||||
"layer_one": "Livello",
|
||||
"layer_many": "Livelli",
|
||||
"layer_other": "Livelli",
|
||||
"disableTransparencyEffect": "Disabilita l'effetto trasparenza",
|
||||
"controlLayers_withCount_visible": "Livelli di controllo ({{count}})",
|
||||
"transparency": "Trasparenza",
|
||||
"newCanvasSessionDesc": "Questo cancellerà la tela e tutte le impostazioni, eccetto la selezione del modello. Le generazioni saranno effettuate sulla tela.",
|
||||
"rasterLayers_withCount_visible": "Livelli raster ({{count}})",
|
||||
"globalReferenceImages_withCount_visible": "Immagini di riferimento Globali ({{count}})",
|
||||
"globalReferenceImages_withCount_hidden": "Immagini di riferimento globali ({{count}} nascoste)",
|
||||
"layer_withCount_one": "Livello ({{count}})",
|
||||
"layer_withCount_many": "Livelli ({{count}})",
|
||||
"layer_withCount_other": "Livelli ({{count}})",
|
||||
"unlocked": "Sbloccato",
|
||||
"enableTransparencyEffect": "Abilita l'effetto trasparenza",
|
||||
"replaceLayer": "Sostituisci livello",
|
||||
@@ -2373,9 +2362,6 @@
|
||||
"pullBboxIntoReferenceImage": "Carica l'immagine delimitata nel riquadro",
|
||||
"showProgressOnCanvas": "Mostra i progressi sulla Tela",
|
||||
"weight": "Peso",
|
||||
"newGallerySession": "Nuova sessione Galleria",
|
||||
"newGallerySessionDesc": "Questo cancellerà la tela e tutte le impostazioni, eccetto la selezione del modello. Le generazioni saranno inviate alla galleria.",
|
||||
"newCanvasSession": "Nuova sessione Tela",
|
||||
"deleteSelected": "Elimina selezione",
|
||||
"settings": {
|
||||
"isolatedStagingPreview": "Anteprima di generazione isolata",
|
||||
@@ -2416,8 +2402,8 @@
|
||||
"accept": "Accetta",
|
||||
"saveToGallery": "Salva nella Galleria",
|
||||
"previous": "Precedente",
|
||||
"showResultsOn": "Risultati visualizzati",
|
||||
"showResultsOff": "Risultati nascosti"
|
||||
"showResultsOn": "Visualizzare i risultati",
|
||||
"showResultsOff": "Nascondere i risultati"
|
||||
},
|
||||
"HUD": {
|
||||
"bbox": "Riquadro di delimitazione",
|
||||
@@ -2449,7 +2435,6 @@
|
||||
"copyBboxToClipboard": "Copia il riquadro di delimitazione negli appunti",
|
||||
"newResizedControlLayer": "Nuovo livello di controllo ridimensionato"
|
||||
},
|
||||
"newImg2ImgCanvasFromImage": "Nuova Immagine da immagine",
|
||||
"copyRasterLayerTo": "Copia $t(controlLayers.rasterLayer) in",
|
||||
"copyControlLayerTo": "Copia $t(controlLayers.controlLayer) in",
|
||||
"copyInpaintMaskTo": "Copia $t(controlLayers.inpaintMask) in",
|
||||
@@ -2457,7 +2442,6 @@
|
||||
"dragToMove": "Trascina un punto per spostarlo",
|
||||
"clickToAdd": "Fare clic sul livello per aggiungere un punto",
|
||||
"clickToRemove": "Clicca su un punto per rimuoverlo",
|
||||
"help3": "Inverte la selezione per selezionare tutto tranne l'oggetto di destinazione.",
|
||||
"pointType": "Tipo punto",
|
||||
"apply": "Applica",
|
||||
"reset": "Reimposta",
|
||||
@@ -2469,8 +2453,16 @@
|
||||
"neutral": "Neutro",
|
||||
"saveAs": "Salva come",
|
||||
"process": "Elabora",
|
||||
"help1": "Seleziona un singolo oggetto di destinazione. Aggiungi i punti <Bold>Includi</Bold> e <Bold>Escludi</Bold> per indicare quali parti del livello fanno parte dell'oggetto di destinazione.",
|
||||
"help2": "Inizia con un punto <Bold>Include</Bold> all'interno dell'oggetto di destinazione. Aggiungi altri punti per perfezionare la selezione. Meno punti in genere producono risultati migliori."
|
||||
"desc": "Seleziona un singolo oggetto di destinazione. Una volta completata la selezione, fai clic su <Bold>Applica</Bold> per eliminare tutto ciò che si trova al di fuori dell'area selezionata, oppure salva la selezione come nuovo livello.",
|
||||
"visualModeDesc": "La modalità visiva utilizza input di tipo riquadro e punto per selezionare un oggetto.",
|
||||
"visualMode1": "Fai clic e trascina per disegnare un riquadro attorno all'oggetto che desideri selezionare. Puoi ottenere risultati migliori disegnando il riquadro un po' più grande o più piccolo dell'oggetto.",
|
||||
"visualMode2": "Fare clic per aggiungere un punto di <Bold>iinclusione</Bold>i verde oppure fare clic tenendo premuto Maiusc per aggiungere un punto di <Bold>iesclusione</Bold>i rosso per indicare al modello cosa includere o escludere.",
|
||||
"visualMode3": "I punti possono essere utilizzati per perfezionare una selezione di caselle oppure in modo indipendente.",
|
||||
"promptModeDesc": "La modalità Prompt utilizza l'input di testo per selezionare un oggetto.",
|
||||
"promptMode1": "Digitare una breve descrizione dell'oggetto che si desidera selezionare.",
|
||||
"promptMode2": "Utilizzare un linguaggio semplice, evitando descrizioni complesse o oggetti multipli.",
|
||||
"model": "Modello",
|
||||
"prompt": "Prompt di selezione"
|
||||
},
|
||||
"convertControlLayerTo": "Converti $t(controlLayers.controlLayer) in",
|
||||
"newRasterLayer": "Nuovo $t(controlLayers.rasterLayer)",
|
||||
@@ -2481,7 +2473,6 @@
|
||||
"convertRegionalGuidanceTo": "Converti $t(controlLayers.regionalGuidance) in",
|
||||
"newControlLayer": "Nuovo $t(controlLayers.controlLayer)",
|
||||
"newInpaintMask": "Nuova $t(controlLayers.inpaintMask)",
|
||||
"replaceCurrent": "Sostituisci corrente",
|
||||
"mergeDown": "Unire in basso",
|
||||
"mergingLayers": "Unione dei livelli",
|
||||
"controlLayerEmptyState": "<UploadButton>Carica un'immagine</UploadButton>, trascina un'immagine dalla galleria su questo livello, <PullBboxButton>trascina il riquadro di delimitazione in questo livello</PullBboxButton> oppure disegna sulla tela per iniziare.",
|
||||
@@ -2495,7 +2486,6 @@
|
||||
"newSession": "Nuova sessione",
|
||||
"resetCanvasLayers": "Ripristina livelli Tela",
|
||||
"referenceImageRegional": "Immagine di riferimento (regionale)",
|
||||
"referenceImageGlobal": "Immagine di riferimento (globale)",
|
||||
"warnings": {
|
||||
"controlAdapterNoModelSelected": "nessun modello selezionato per il livello di controllo",
|
||||
"controlAdapterNoControl": "nessun controllo selezionato/disegnato",
|
||||
@@ -2521,7 +2511,6 @@
|
||||
"pasteToBbox": "Riquadro di delimitazione",
|
||||
"pasteToCanvas": "Tela",
|
||||
"pasteToCanvasDesc": "Nuovo livello (nella Tela)",
|
||||
"pastedTo": "Incollato su {{destination}}",
|
||||
"regionCopiedToClipboard": "{{region}} Copiato negli appunti",
|
||||
"errors": {
|
||||
"unableToFindImage": "Impossibile trovare l'immagine",
|
||||
@@ -2544,16 +2533,66 @@
|
||||
"showNonRasterLayers": "Mostra livelli non raster (Shift+H)",
|
||||
"hideNonRasterLayers": "Nascondi livelli non raster (Shift+H)",
|
||||
"referenceImageEmptyStateWithCanvasOptions": "<UploadButton>Carica un'immagine</UploadButton>, trascina un'immagine dalla galleria su questa immagine di riferimento o <PullBboxButton>trascina il riquadro di delimitazione in questa immagine di riferimento</PullBboxButton> per iniziare.",
|
||||
"uploadOrDragAnImage": "Trascina un'immagine dalla galleria o <UploadButton>carica un'immagine</UploadButton>.",
|
||||
"autoSwitch": {
|
||||
"off": "Spento",
|
||||
"switchOnStart": "All'inizio",
|
||||
"switchOnFinish": "Alla fine",
|
||||
"off": "Spento"
|
||||
"switchOnFinish": "Alla fine"
|
||||
},
|
||||
"invertMask": "Inverti maschera",
|
||||
"fitBboxToMasks": "Adatta il riquadro di delimitazione alle maschere",
|
||||
"maxRefImages": "Max Immagini di rif.to",
|
||||
"useAsReferenceImage": "Usa come immagine di riferimento"
|
||||
"useAsReferenceImage": "Usa come immagine di riferimento",
|
||||
"globalReferenceImage_withCount_one": "$t(controlLayers.globalReferenceImage)",
|
||||
"globalReferenceImage_withCount_many": "Immagini di riferimento globali",
|
||||
"globalReferenceImage_withCount_other": "Immagini di riferimento globali",
|
||||
"layer_withCount_one": "Livello ({{count}})",
|
||||
"layer_withCount_many": "Livelli ({{count}})",
|
||||
"layer_withCount_other": "Livelli ({{count}})",
|
||||
"addAdjustments": "Aggiungi regolazioni",
|
||||
"removeAdjustments": "Rimuovi regolazioni",
|
||||
"adjustments": {
|
||||
"simple": "Semplice",
|
||||
"curves": "Curve",
|
||||
"heading": "Regolazioni",
|
||||
"expand": "Espandi regolazioni",
|
||||
"collapse": "Comprimi regolazioni",
|
||||
"brightness": "Luminosità",
|
||||
"contrast": "Contrasto",
|
||||
"saturation": "Saturazione",
|
||||
"temperature": "Temperatura",
|
||||
"tint": "Tinta",
|
||||
"sharpness": "Nitidezza",
|
||||
"reset": "Reimposta"
|
||||
},
|
||||
"deletePrompt": "Elimina prompt",
|
||||
"addGlobalReferenceImage": "Aggiungi $t(controlLayers.globalReferenceImage)",
|
||||
"referenceImageGlobal": "Immagine di riferimento (globale)",
|
||||
"sendingToGallery": "Invia generazioni alla Galleria",
|
||||
"sendToGallery": "Invia alla Galleria",
|
||||
"sendToGalleryDesc": "Premendo Invoke viene generata e salvata un'immagine unica nella tua galleria.",
|
||||
"newImg2ImgCanvasFromImage": "Nuovo immagine-a-immagine da Immagine",
|
||||
"sendToCanvasDesc": "Premendo Invoke il lavoro in corso viene visualizzato sulla tela.",
|
||||
"viewProgressOnCanvas": "Visualizza i progressi e gli output nel <Btn>Visualizzatore immagini</Btn>.",
|
||||
"regionalGuidance_withCount_hidden": "Guida regionale ({{count}} nascosti)",
|
||||
"controlLayers_withCount_hidden": "Livelli di controllo ({{count}} nascosti)",
|
||||
"rasterLayers_withCount_hidden": "Livelli raster ({{count}} nascosti)",
|
||||
"globalReferenceImages_withCount_hidden": "Immagini di riferimento globali ({{count}} nascoste)",
|
||||
"inpaintMasks_withCount_hidden": "Maschere Inpaint ({{count}} nascoste)",
|
||||
"regionalGuidance_withCount_visible": "Guida regionale ({{count}})",
|
||||
"controlLayers_withCount_visible": "Livelli di controllo ({{count}})",
|
||||
"rasterLayers_withCount_visible": "Livelli raster ({{count}})",
|
||||
"globalReferenceImages_withCount_visible": "Immagini di riferimento globali ({{count}})",
|
||||
"inpaintMasks_withCount_visible": "Maschere Inpaint ({{count}})",
|
||||
"pastedTo": "Incollato su {{destination}}",
|
||||
"stagingOnCanvas": "Predisponi le immagini su",
|
||||
"newGallerySession": "Nuova sessione della Galleria",
|
||||
"newGallerySessionDesc": "Questo cancellerà la tela e tutte le impostazioni, ad eccezione della selezione del modello. Le generazioni verranno inviate alla galleria.",
|
||||
"newCanvasSession": "Nuova sessione Tela",
|
||||
"newCanvasSessionDesc": "Questo cancellerà la tela e tutte le impostazioni, ad eccezione della selezione del modello. Le generazioni verranno predisposte sulla tela.",
|
||||
"replaceCurrent": "Sostituisci l'attuale",
|
||||
"uploadOrDragAnImage": "Trascina un'immagine dalla galleria o <UploadButton>carica un'immagine</UploadButton>.",
|
||||
"sendingToCanvas": "Predisponi le generazioni sulla Tela",
|
||||
"viewProgressInViewer": "Visualizza i progressi e gli output nel <Btn>Visualizzatore immagini</Btn>."
|
||||
},
|
||||
"ui": {
|
||||
"tabs": {
|
||||
@@ -2647,13 +2686,13 @@
|
||||
"canvasCalloutLink": "Per ulteriori funzionalità, vai su Tela."
|
||||
},
|
||||
"videoTitle": "Genera video da prompt testuale.",
|
||||
"video": {
|
||||
"startingFrameCalloutTitle": "Aggiungi un fotogramma iniziale",
|
||||
"startingFrameCalloutDesc": "Aggiungi un'immagine per controllare il primo fotogramma del tuo video."
|
||||
},
|
||||
"addStartingFrame": {
|
||||
"title": "Aggiungi un fotogramma iniziale",
|
||||
"description": "Aggiungi un'immagine per controllare il primo fotogramma del tuo video."
|
||||
},
|
||||
"video": {
|
||||
"startingFrameCalloutTitle": "Aggiungi un fotogramma iniziale",
|
||||
"startingFrameCalloutDesc": "Aggiungi un'immagine per controllare il primo fotogramma del tuo video."
|
||||
}
|
||||
},
|
||||
"panels": {
|
||||
@@ -2750,11 +2789,11 @@
|
||||
"whatsNewInInvoke": "Novità in Invoke",
|
||||
"readReleaseNotes": "Leggi le note di rilascio",
|
||||
"watchRecentReleaseVideos": "Guarda i video su questa versione",
|
||||
"watchUiUpdatesOverview": "Guarda le novità dell'interfaccia",
|
||||
"items": [
|
||||
"Tela: Color Picker non campiona l'alfa, il riquadro di delimitazione rispetta il blocco delle proporzioni quando si ridimensiona il pulsante Mescola per i campi numerici nel generatore di flusso di lavoro, nasconde i cursori delle dimensioni dei pixel quando si utilizza un modello che non li supporta",
|
||||
"Flussi di lavoro: aggiunto un pulsante Mescola ai campi di input numerici"
|
||||
]
|
||||
"Seleziona oggetto v2: selezione degli oggetti migliorata con input di punti e riquadri o prompt di testo.",
|
||||
"Regolazioni del livello raster: regola facilmente la luminosità, il contrasto, la saturazione, le curve e altro ancora del livello."
|
||||
],
|
||||
"watchUiUpdatesOverview": "Guarda la panoramica degli aggiornamenti dell'interfaccia utente"
|
||||
},
|
||||
"system": {
|
||||
"logLevel": {
|
||||
@@ -2803,6 +2842,9 @@
|
||||
"clearSucceeded": "Cache del modello cancellata",
|
||||
"clearFailed": "Problema durante la cancellazione della cache del modello"
|
||||
},
|
||||
"lora": {
|
||||
"weight": "Peso"
|
||||
},
|
||||
"video": {
|
||||
"noVideoSelected": "Nessun video selezionato",
|
||||
"selectFromGallery": "Seleziona un video dalla galleria per riprodurlo"
|
||||
|
||||
@@ -27,7 +27,6 @@
|
||||
"openInNewTab": "新しいタブで開く",
|
||||
"controlNet": "コントロールネット",
|
||||
"linear": "リニア",
|
||||
"imageFailedToLoad": "画像が読み込めません",
|
||||
"modelManager": "モデルマネージャー",
|
||||
"learnMore": "もっと学ぶ",
|
||||
"random": "ランダム",
|
||||
@@ -56,7 +55,6 @@
|
||||
"details": "詳細",
|
||||
"inpaint": "inpaint",
|
||||
"delete": "削除",
|
||||
"nextPage": "次のページ",
|
||||
"copy": "コピー",
|
||||
"error": "エラー",
|
||||
"file": "ファイル",
|
||||
@@ -64,13 +62,10 @@
|
||||
"input": "インプット",
|
||||
"format": "形式",
|
||||
"installed": "インストール済み",
|
||||
"localSystem": "ローカルシステム",
|
||||
"outputs": "アウトプット",
|
||||
"prevPage": "前のページ",
|
||||
"unknownError": "未知のエラー",
|
||||
"orderBy": "並び順:",
|
||||
"enabled": "有効",
|
||||
"notInstalled": "未 $t(common.installed)",
|
||||
"positivePrompt": "ポジティブプロンプト",
|
||||
"negativePrompt": "ネガティブプロンプト",
|
||||
"selected": "選択済み",
|
||||
@@ -96,7 +91,6 @@
|
||||
"close": "閉じる",
|
||||
"warnings": "警告",
|
||||
"dontShowMeThese": "次回から表示しない",
|
||||
"goTo": "移動",
|
||||
"generating": "生成中",
|
||||
"loadingModel": "モデルをロード中",
|
||||
"layout": "レイアウト",
|
||||
@@ -107,7 +101,6 @@
|
||||
"min": "最小",
|
||||
"max": "最大",
|
||||
"values": "値",
|
||||
"resetToDefaults": "デフォルトに戻す",
|
||||
"row": "行",
|
||||
"column": "列",
|
||||
"board": "ボード",
|
||||
@@ -131,7 +124,6 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "画像のサイズ",
|
||||
"gallerySettings": "ギャラリーの設定",
|
||||
"noImagesInGallery": "表示する画像がありません",
|
||||
"autoSwitchNewImages": "新しい画像に自動切替",
|
||||
"copy": "コピー",
|
||||
"image": "画像",
|
||||
@@ -145,7 +137,6 @@
|
||||
"deleteImage_other": "画像 {{count}} 枚を削除",
|
||||
"deleteImagePermanent": "削除された画像は復元できません。",
|
||||
"download": "ダウンロード",
|
||||
"unableToLoad": "ギャラリーをロードできません",
|
||||
"bulkDownloadRequested": "ダウンロード準備中",
|
||||
"bulkDownloadRequestedDesc": "ダウンロードの準備中です。しばらくお待ちください。",
|
||||
"bulkDownloadRequestFailed": "ダウンロード準備中に問題が発生",
|
||||
@@ -160,7 +151,6 @@
|
||||
"compareImage": "比較画像",
|
||||
"openInViewer": "ビューアで開く",
|
||||
"selectForCompare": "比較対象として選択",
|
||||
"selectAnImageToCompare": "比較する画像を選択",
|
||||
"slider": "スライダー",
|
||||
"sideBySide": "横並び",
|
||||
"hover": "ホバー",
|
||||
@@ -172,8 +162,6 @@
|
||||
"compareHelp4": "<Kbd>[Z</Kbd>]または<Kbd>[Esc</Kbd>]を押して終了します。",
|
||||
"compareHelp2": "<Kbd>M</Kbd> キーを押して比較モードを切り替えます。",
|
||||
"move": "移動",
|
||||
"openViewer": "ビューアを開く",
|
||||
"closeViewer": "ビューアを閉じる",
|
||||
"exitSearch": "画像検索を終了",
|
||||
"oldestFirst": "最古から",
|
||||
"showStarredImagesFirst": "スター付き画像を最初に",
|
||||
@@ -182,7 +170,6 @@
|
||||
"searchImages": "メタデータで検索",
|
||||
"gallery": "ギャラリー",
|
||||
"newestFirst": "最新から",
|
||||
"jump": "ジャンプ",
|
||||
"go": "進む",
|
||||
"sortDirection": "並び替え順",
|
||||
"displayBoardSearch": "ボード検索",
|
||||
@@ -605,7 +592,6 @@
|
||||
"scanResults": "結果をスキャン",
|
||||
"scanPlaceholder": "ローカルフォルダへのパス",
|
||||
"typePhraseHere": "ここにフレーズを入力",
|
||||
"ipAdapters": "IPアダプター",
|
||||
"modelImageUpdated": "モデル画像アップデート",
|
||||
"installAll": "全てインストール",
|
||||
"installRepo": "リポジトリをインストール",
|
||||
@@ -647,7 +633,6 @@
|
||||
"spandrelImageToImage": "Image to Image(スパンドレル)",
|
||||
"starterBundles": "スターターバンドル",
|
||||
"starterModels": "スターターモデル",
|
||||
"starterModelsInModelManager": "スターターモデルがモデルマネージャーで見つかりました",
|
||||
"modelImageDeleteFailed": "モデル画像の削除失敗",
|
||||
"urlForbidden": "このモデルにアクセスできません",
|
||||
"urlForbiddenErrorMessage": "このモデルを配布しているサイトからリクエスト権限が必要かもしれません.",
|
||||
@@ -656,12 +641,10 @@
|
||||
"inplaceInstall": "定位置にインストール",
|
||||
"fileSize": "ファイルサイズ",
|
||||
"modelPickerFallbackNoModelsInstalled2": "<LinkComponent>モデルマネージャー</LinkComponent> にアクセスしてモデルをインストールしてください.",
|
||||
"filterModels": "フィルターモデル",
|
||||
"modelPickerFallbackNoModelsInstalled": "モデルがインストールされていません.",
|
||||
"manageModels": "モデル管理",
|
||||
"hfTokenReset": "ハギングフェイストークンリセット",
|
||||
"relatedModels": "関連のあるモデル",
|
||||
"showOnlyRelatedModels": "関連している",
|
||||
"installedModelsCount": "{{total}} モデルのうち {{installed}} 個がインストールされています。",
|
||||
"allNModelsInstalled": "{{count}} 個のモデルがすべてインストールされています",
|
||||
"nToInstall": "{{count}}個をインストールする",
|
||||
@@ -678,12 +661,8 @@
|
||||
"scanFolderDescription": "ローカルフォルダをスキャンしてモデルを自動的に検出し、インストールします。",
|
||||
"recommendedModels": "推奨モデル",
|
||||
"exploreStarter": "または、利用可能なすべてのスターターモデルを参照してください",
|
||||
"quickStart": "クイックスタートバンドル",
|
||||
"bundleDescription": "各バンドルには各モデルファミリーの必須モデルと、開始するための厳選されたベースモデルが含まれています。",
|
||||
"browseAll": "または、利用可能なすべてのモデルを参照してください。",
|
||||
"stableDiffusion15": "Stable Diffusion1.5",
|
||||
"sdxl": "SDXL",
|
||||
"fluxDev": "FLUX.1 dev"
|
||||
"sdxl": "SDXL"
|
||||
}
|
||||
},
|
||||
"parameters": {
|
||||
@@ -699,12 +678,10 @@
|
||||
"scaleBeforeProcessing": "処理前のスケール",
|
||||
"scaledWidth": "幅のスケール",
|
||||
"scaledHeight": "高さのスケール",
|
||||
"downloadImage": "画像をダウンロード",
|
||||
"usePrompt": "プロンプトを使用",
|
||||
"useSeed": "シード値を使用",
|
||||
"useAll": "すべてを使用",
|
||||
"info": "情報",
|
||||
"showOptionsPanel": "サイドパネルを表示 (O or T)",
|
||||
"iterations": "生成回数",
|
||||
"general": "基本設定",
|
||||
"setToOptimalSize": "サイズをモデルに最適化",
|
||||
@@ -718,7 +695,6 @@
|
||||
"collectionNumberLTExclusiveMin": "{{value}} <= {{exclusiveMinimum}} (最小値を除く)",
|
||||
"missingInputForField": "入力の欠落",
|
||||
"noModelSelected": "モデルが選択されていません",
|
||||
"emptyBatches": "空のバッチ",
|
||||
"collectionStringTooLong": "長すぎます,最大{{maxLength}}",
|
||||
"batchNodeCollectionSizeMismatchNoGroupId": "バッチグループのコレクションサイズが合いません",
|
||||
"invoke": "呼び出す",
|
||||
@@ -730,7 +706,6 @@
|
||||
"missingNodeTemplate": "ノードテンプレートの欠落",
|
||||
"batchNodeNotConnected": "バッチノードが: {{label}}につながっていない",
|
||||
"collectionNumberLTMin": "{{value}} < {{minimum}} (最小増加)",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), スケーリングされたbboxの高さは{{height}}です",
|
||||
"fluxModelMultipleControlLoRAs": "コントロールLoRAは1度に1つしか使用できません",
|
||||
"noPrompts": "プロンプトが生成されません",
|
||||
"noNodesInGraph": "グラフにノードがありません",
|
||||
@@ -738,7 +713,6 @@
|
||||
"canvasIsFiltering": "キャンバスがビジー状態(フィルタリング)",
|
||||
"canvasIsCompositing": "キャンバスがビジー状態(合成)",
|
||||
"systemDisconnected": "システムが切断されました",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), 拡大縮小されたbboxの幅は{{width}}です",
|
||||
"canvasIsTransforming": "キャンバスがビジー状態(変換)",
|
||||
"canvasIsRasterizing": "キャンバスがビジー状態(ラスタライズ)",
|
||||
"modelIncompatibleBboxHeight": "Bboxの高さは{{height}}ですが,{{model}}は{{multiple}}の倍数が必要です",
|
||||
@@ -746,8 +720,6 @@
|
||||
"modelIncompatibleBboxWidth": "Bboxの幅は{{width}}ですが, {{model}}は{{multiple}}の倍数が必要です",
|
||||
"modelIncompatibleScaledBboxWidth": "bboxの幅は{{width}}ですが,{{model}}は{{multiple}}の倍数が必要です",
|
||||
"canvasIsSelectingObject": "キャンバスがビジー状態(オブジェクトの選択)",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), bboxの幅は{{width}}です",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), bboxの高さは{{height}}です",
|
||||
"noFLUXVAEModelSelected": "FLUX生成にVAEモデルが選択されていません",
|
||||
"noT5EncoderModelSelected": "FLUX生成にT5エンコーダモデルが選択されていません",
|
||||
"modelDisabledForTrial": "{{modelName}} を使用した生成はトライアルアカウントではご利用いただけません.アカウント設定にアクセスしてアップグレードしてください。",
|
||||
@@ -825,8 +797,6 @@
|
||||
"enableHighlightFocusedRegions": "重点領域を強調表示",
|
||||
"clearIntermediatesDesc1": "中間物をクリアすると、キャンバスとコントロールネットの状態がリセットされます.",
|
||||
"showProgressInViewer": "ビューアで進行状況画像を表示する",
|
||||
"modelDescriptionsDisabled": "ドロップダウンのモデル説明が無効になっています",
|
||||
"modelDescriptionsDisabledDesc": "ドロップダウンのモデル説明が無効になっています.設定で有効にしてください.",
|
||||
"clearIntermediatesDisabled": "中間物をクリアするにはキューが空でなければなりません",
|
||||
"clearIntermediatesDesc2": "中間画像は生成時に生成される副産物であり、ギャラリーに表示される結果画像とは異なります.中間画像を削除するとディスク容量が解放されます.",
|
||||
"intermediatesClearedFailed": "中間物をクリアする問題",
|
||||
@@ -857,11 +827,9 @@
|
||||
"imagesWillBeAddedTo": "アップロードされた画像はボード {{boardName}} のアセットに追加されます.",
|
||||
"layerCopiedToClipboard": "レイヤーがクリップボードにコピーされました",
|
||||
"pasteFailed": "貼り付け失敗",
|
||||
"imageSavingFailed": "画像保存に失敗しました",
|
||||
"importSuccessful": "インポートが成功しました",
|
||||
"problemDownloadingImage": "画像をダウンロードできません",
|
||||
"modelAddedSimple": "モデルがキューに追加されました",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "PNG、JPEG、または WEBP 画像は最大 1 つにする必要があります.",
|
||||
"outOfMemoryErrorDesc": "現在の生成設定はシステム容量を超えています.設定を調整してもう一度お試しください.",
|
||||
"parametersSet": "パラメーターが呼び出されました",
|
||||
"modelImportCanceled": "モデルのインポートがキャンセルされました",
|
||||
@@ -876,14 +844,11 @@
|
||||
"linkCopied": "リンクがコピーされました",
|
||||
"unableToLoadImage": "画像をロードできません",
|
||||
"unableToLoadImageMetadata": "画像のメタデータをロードできません",
|
||||
"imageSaved": "画像が保存されました",
|
||||
"importFailed": "インポートに失敗しました",
|
||||
"invalidUpload": "無効なアップロードです",
|
||||
"outOfMemoryError": "メモリ不足エラー",
|
||||
"parameterSetDesc": "{{parameter}}を呼び出し",
|
||||
"errorCopied": "エラーがコピーされました",
|
||||
"sentToCanvas": "キャンバスに送信",
|
||||
"setControlImage": "コントロール画像としてセット",
|
||||
"workflowLoaded": "ワークフローがロードされました",
|
||||
"unableToCopy": "コピーできません",
|
||||
"unableToCopyDesc": "あなたのブラウザはクリップボードアクセスをサポートしていません.Firefoxユーザーの場合は、以下の手順で修正できる可能性があります. ",
|
||||
@@ -897,32 +862,23 @@
|
||||
"parameterNotSetDescWithMessage": "{{parameter}}: {{message}}を呼び出せません",
|
||||
"problemCopyingLayer": "レイヤーをコピーできません",
|
||||
"problemSavingLayer": "レイヤー保存ができません",
|
||||
"setNodeField": "ノードフィールドとしてセット",
|
||||
"layerSavedToAssets": "レイヤーがアセットに保存されました",
|
||||
"outOfMemoryErrorDescLocal": "OOM を削減するには、<LinkComponent>低 VRAM ガイド</LinkComponent> に従ってください.",
|
||||
"parameterNotSet": "パラメーターが呼び出されていません",
|
||||
"addedToBoard": "{{name}} 個の資産をボードに追加しました",
|
||||
"addedToUncategorized": "$t(boards.uncategorized)個のアセットがボードに追加されました",
|
||||
"problemDeletingWorkflow": "ワークフローが削除された問題",
|
||||
"imageNotLoadedDesc": "画像を見つけられません",
|
||||
"parameterNotSetDesc": "{{parameter}}を呼び出せません",
|
||||
"chatGPT4oIncompatibleGenerationMode": "ChatGPT 4oは,テキストから画像への生成と画像から画像への生成のみをサポートしています.インペインティングおよび,アウトペインティングタスクには他のモデルを使用してください.",
|
||||
"imagenIncompatibleGenerationMode": "Google {{model}} はテキストから画像への変換のみをサポートしています. 画像から画像への変換, インペインティング,アウトペインティングのタスクには他のモデルを使用してください.",
|
||||
"noRasterLayers": "ラスターレイヤーが見つかりません",
|
||||
"noRasterLayersDesc": "PSDにエクスポートするには、少なくとも1つのラスターレイヤーを作成します",
|
||||
"noActiveRasterLayers": "アクティブなラスターレイヤーがありません",
|
||||
"noActiveRasterLayersDesc": "PSD にエクスポートするには、少なくとも 1 つのラスター レイヤーを有効にします",
|
||||
"noVisibleRasterLayers": "表示されるラスター レイヤーがありません",
|
||||
"noVisibleRasterLayersDesc": "PSD にエクスポートするには、少なくとも 1 つのラスター レイヤーを有効にします",
|
||||
"invalidCanvasDimensions": "キャンバスのサイズが無効です",
|
||||
"canvasTooLarge": "キャンバスが大きすぎます",
|
||||
"canvasTooLargeDesc": "キャンバスのサイズがPSDエクスポートの最大許容サイズを超えています。キャンバス全体の幅と高さを小さくしてから、もう一度お試しください。",
|
||||
"failedToProcessLayers": "レイヤーの処理に失敗しました",
|
||||
"psdExportSuccess": "PSDエクスポート完了",
|
||||
"psdExportSuccessDesc": "{{count}} 個のレイヤーを PSD ファイルに正常にエクスポートしました",
|
||||
"problemExportingPSD": "PSD のエクスポート中に問題が発生しました",
|
||||
"canvasManagerNotAvailable": "キャンバスマネージャーは利用できません",
|
||||
"noValidLayerAdapters": "有効なレイヤーアダプタが見つかりません",
|
||||
"fluxKontextIncompatibleGenerationMode": "Flux Kontext はテキストから画像への変換のみをサポートしています。画像から画像への変換、インペインティング、アウトペインティングのタスクには他のモデルを使用してください。",
|
||||
"promptGenerationStarted": "プロンプト生成が開始されました",
|
||||
"uploadAndPromptGenerationFailed": "画像のアップロードとプロンプトの生成に失敗しました",
|
||||
@@ -954,7 +910,6 @@
|
||||
"positivePrompt": "ポジティブプロンプト",
|
||||
"strength": "Image to Image 強度",
|
||||
"recallParameters": "パラメータを再使用",
|
||||
"recallParameter": "{{label}} を再使用",
|
||||
"imageDimensions": "画像サイズ",
|
||||
"imageDetails": "画像の詳細",
|
||||
"model": "モデル",
|
||||
@@ -969,7 +924,6 @@
|
||||
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
||||
"canvasV2Metadata": "キャンバス",
|
||||
"guidance": "手引き",
|
||||
"parsingFailed": "解析に失敗しました",
|
||||
"seamlessXAxis": "シームレスX軸",
|
||||
"seamlessYAxis": "シームレスY軸",
|
||||
"parameterSet": "パラメーター {{parameter}} が設定されました",
|
||||
@@ -1027,7 +981,6 @@
|
||||
"clearQueueAlertDialog2": "キューをクリアしてもよろしいですか?",
|
||||
"item": "項目",
|
||||
"graphFailedToQueue": "グラフをキューに追加できませんでした",
|
||||
"batchFieldValues": "バッチの詳細",
|
||||
"openQueue": "キューを開く",
|
||||
"time": "時間",
|
||||
"completedIn": "完了まで",
|
||||
@@ -1057,14 +1010,12 @@
|
||||
"models": {
|
||||
"noMatchingModels": "一致するモデルがありません",
|
||||
"loading": "読み込み中",
|
||||
"noMatchingLoRAs": "一致するLoRAがありません",
|
||||
"noModelsAvailable": "使用可能なモデルがありません",
|
||||
"selectModel": "モデルを選択してください",
|
||||
"concepts": "コンセプト",
|
||||
"addLora": "LoRAを追加",
|
||||
"lora": "LoRA",
|
||||
"defaultVAE": "デフォルトVAE",
|
||||
"noLoRAsInstalled": "インストールされているLoRAはありません",
|
||||
"noRefinerModelsInstalled": "インストールされているSDXLリファイナーモデルはありません",
|
||||
"noCompatibleLoRAs": "互換性のあるLoRAはありません"
|
||||
},
|
||||
@@ -1074,7 +1025,6 @@
|
||||
"addNodeToolTip": "ノードを追加 (Shift+A, Space)",
|
||||
"missingTemplate": "Invalid node: タイプ {{type}} のノード {{node}} にテンプレートがありません(未インストール?)",
|
||||
"loadWorkflow": "ワークフローを読み込み",
|
||||
"hideLegendNodes": "フィールドタイプの凡例を非表示",
|
||||
"float": "浮動小数点",
|
||||
"integer": "整数",
|
||||
"nodeTemplate": "ノードテンプレート",
|
||||
@@ -1118,13 +1068,11 @@
|
||||
"enum": "Enum",
|
||||
"arithmeticSequence": "等差数列",
|
||||
"linearDistribution": "線形分布",
|
||||
"addLinearView": "ライナービューに追加",
|
||||
"animatedEdges": "アニメーションエッジ",
|
||||
"uniformRandomDistribution": "一様ランダム分布",
|
||||
"noBatchGroup": "グループなし",
|
||||
"parseString": "文字列の解析",
|
||||
"generatorImagesFromBoard": "ボードからの画像",
|
||||
"generatorLoading": "読み込み中",
|
||||
"missingNode": "呼び出しノードがありません",
|
||||
"missingSourceOrTargetNode": "ソースまたはターゲットノードがありません",
|
||||
"missingSourceOrTargetHandle": "ソースまたはターゲットハンドルがありません",
|
||||
@@ -1145,7 +1093,6 @@
|
||||
"missingInvocationTemplate": "呼び出しテンプレートがありません",
|
||||
"nodePack": "ノードパック",
|
||||
"targetNodeFieldDoesNotExist": "無効なエッジ:ターゲット/インプットフィールド{{node}}.{{field}} が存在しません",
|
||||
"mismatchedVersion": "無効なノード:ノード {{node}} のタイプ {{type}} はバージョンとミスマッチしています (アップデートを試されますか?)",
|
||||
"dynamicPromptsCombinatorial": "ダイナミックプロンプト(組み合わせ)",
|
||||
"cannotMixAndMatchCollectionItemTypes": "コレクション・アイテムの種類を組み合わせることはできません",
|
||||
"missingFieldTemplate": "フィールドテンプレートがありません",
|
||||
@@ -1155,7 +1102,6 @@
|
||||
"collectionOrScalarFieldType": "{{name}} (単数またはコレクション)",
|
||||
"unableToUpdateNode": "ノードアップロード失敗:ノード {{node}} のタイプ {{type}} (削除か再生成が必要かもしれません)",
|
||||
"deletedInvalidEdge": "無効なエッジを削除しました{{source}} -> {{target}}",
|
||||
"noFieldsLinearview": "線形ビューに追加されたフィールドがありません",
|
||||
"collectionFieldType": "{{name}} (コレクション)",
|
||||
"colorCodeEdgesHelp": "接続されたフィールドによるカラーコードエッジ",
|
||||
"showEdgeLabelsHelp": "エッジのラベルを表示,接続されているノードを示す",
|
||||
@@ -1170,7 +1116,6 @@
|
||||
"loadWorkflowDesc2": "現在のワークフローは保存されていない変更があります.",
|
||||
"clearWorkflowDesc": "このワークフローをクリアして新しいワークフローにしますか?",
|
||||
"updateNode": "ノードをアップデート",
|
||||
"versionUnknown": " バージョン不明",
|
||||
"graph": "グラフ",
|
||||
"workflowContact": "お問い合わせ",
|
||||
"outputFieldTypeParseError": "出力フィールド {{node}}.{{field}} の型を解析できません({{message}})",
|
||||
@@ -1189,36 +1134,28 @@
|
||||
"unableToExtractSchemaNameFromRef": "参照からスキーマ名を抽出できません",
|
||||
"unableToUpdateNodes_other": "{{count}} 個のノードをアップデートできません",
|
||||
"workflowSettings": "ワークフローエディター設定",
|
||||
"generateValues": "値を生成",
|
||||
"floatRangeGenerator": "浮動小数点レンジ生成器",
|
||||
"integerRangeGenerator": "整数レンジ生成器",
|
||||
"specialDesc": "この呼び出しは,アプリ内で特別な処理を行います.例えば,バッチノードは1つのワークフローから複数のグラフをキューに入れるために使用されます.",
|
||||
"modelAccessError": "モデル {{key}}が見つからないので,デフォルトにリセットします",
|
||||
"betaDesc": "この呼び出しはベータ版です.安定するまでは,アプリのアップデートの際に変更される可能性があります.この呼び出しは長期的にサポートする予定です.",
|
||||
"internalDesc": "この呼び出しはInvokeによって内部的に使用されます.アプリの更新時に変更される可能性があり,いつでも削除される可能性があります.",
|
||||
"noFieldsViewMode": "このワークフローには表示する選択フィールドがありません.値を設定するためにはワークフロー全体を表示します.",
|
||||
"clearWorkflow": "ワークフローをクリア",
|
||||
"removeLinearView": "線形ビューから削除",
|
||||
"snapToGrid": "グリッドにスナップ",
|
||||
"showMinimapnodes": "ミニマップを表示",
|
||||
"reorderLinearView": "線形ビューの並び替え",
|
||||
"description": "説明",
|
||||
"notesDescription": "ワークフローに関するメモを追加する",
|
||||
"newWorkflowDesc2": "現在のワークフローに保存されていない変更があります.",
|
||||
"unknownField": "不明なフィールド",
|
||||
"unexpectedField_withName": "予期しないフィールド\"{{name}}\"",
|
||||
"loadingTemplates": "読み込み中 {{name}}",
|
||||
"validateConnectionsHelp": "無効な接続が行われたり,無効なグラフが呼び出されたりしないようにします",
|
||||
"validateConnections": "接続とグラフを確認する",
|
||||
"saveToGallery": "ギャラリーに保存",
|
||||
"newWorkflowDesc": "新しいワークフローを作りますか?",
|
||||
"unknownFieldType": "$t(nodes.unknownField)型: {{type}}",
|
||||
"unsupportedArrayItemType": "サポートされていない配列項目型です \"{{type}}\"",
|
||||
"unableToLoadWorkflow": "ワークフローが読み込めません",
|
||||
"unableToValidateWorkflow": "ワークフローを確認できません",
|
||||
"unknownErrorValidatingWorkflow": "ワークフローの確認で不明なエラーが発生",
|
||||
"clearWorkflowDesc2": "現在のワークフローは保存されていない変更があります.",
|
||||
"showLegendNodes": "フィールドタイプの凡例を表示",
|
||||
"unsupportedMismatchedUnion": "CollectionOrScalar型とベース型{{firstType}}および{{secondType}}が不一致です",
|
||||
"updateApp": "アプリケーションをアップデート",
|
||||
"noGraph": "グラフなし",
|
||||
@@ -1236,10 +1173,8 @@
|
||||
"workflowDescription": "短い説明",
|
||||
"workflowValidation": "ワークフロー検証エラー",
|
||||
"noOutputRecorded": "記録されたアウトプットがありません",
|
||||
"unknownTemplate": "不明なテンプレート",
|
||||
"nodeOpacity": "ノードの不透明度",
|
||||
"unableToParseFieldType": "フィールドタイプを解析できません",
|
||||
"unknownInput": "不明な入力: {{name}}"
|
||||
"unableToParseFieldType": "フィールドタイプを解析できません"
|
||||
},
|
||||
"boards": {
|
||||
"autoAddBoard": "自動追加するボード",
|
||||
@@ -1263,7 +1198,6 @@
|
||||
"deleteBoardOnly": "ボードのみ削除",
|
||||
"deletedBoardsCannotbeRestored": "削除したボードと画像は復元できません。「ボードのみ削除」を選択すると、画像は未分類の状態になります。",
|
||||
"movingImagesToBoard_other": "{{count}} の画像をボードに移動:",
|
||||
"hideBoards": "ボードを隠す",
|
||||
"assetsWithCount_other": "{{count}} のアセット",
|
||||
"addPrivateBoard": "プライベートボードを追加",
|
||||
"addSharedBoard": "共有ボードを追加",
|
||||
@@ -1278,10 +1212,8 @@
|
||||
"selectedForAutoAdd": "自動追加に選択済み",
|
||||
"deletedPrivateBoardsCannotbeRestored": "削除されたボードと画像は復元できません。「ボードのみ削除」を選択すると、画像は作成者に対して非公開の未分類状態になります。",
|
||||
"noBoards": "{{boardType}} ボードがありません",
|
||||
"viewBoards": "ボードを表示",
|
||||
"uncategorizedImages": "分類されていない画像",
|
||||
"deleteAllUncategorizedImages": "分類されていないすべての画像を削除",
|
||||
"deletedImagesCannotBeRestored": "削除した画像は復元できません."
|
||||
"deleteAllUncategorizedImages": "分類されていないすべての画像を削除"
|
||||
},
|
||||
"invocationCache": {
|
||||
"invocationCache": "呼び出しキャッシュ",
|
||||
@@ -1753,9 +1685,7 @@
|
||||
"strength": "高解像修復の強度",
|
||||
"enabled": "高解像修復が有効"
|
||||
},
|
||||
"enableHrf": "高解像修復を有効",
|
||||
"hrf": "高解像修復",
|
||||
"upscaleMethod": "アップスケール手法"
|
||||
"hrf": "高解像修復"
|
||||
},
|
||||
"prompt": {
|
||||
"addPromptTrigger": "プロンプトトリガーを追加",
|
||||
@@ -1765,10 +1695,7 @@
|
||||
"expandCurrentPrompt": "現在のプロンプトを展開",
|
||||
"uploadImageForPromptGeneration": "プロンプト生成用の画像をアップロードする",
|
||||
"expandingPrompt": "プロンプトを展開しています...",
|
||||
"resultTitle": "プロンプト拡張完了",
|
||||
"resultSubtitle": "拡張プロンプトの処理方法を選択します:",
|
||||
"replace": "交換する",
|
||||
"insert": "挿入する",
|
||||
"discard": "破棄する"
|
||||
},
|
||||
"ui": {
|
||||
@@ -1834,11 +1761,9 @@
|
||||
}
|
||||
},
|
||||
"controlLayers": {
|
||||
"globalReferenceImage_withCount_other": "全域参照画像",
|
||||
"regionalReferenceImage": "領域参照画像",
|
||||
"saveLayerToAssets": "レイヤーをアセットに保存",
|
||||
"global": "全域",
|
||||
"inpaintMasks_withCount_hidden": "インペイントマスク ({{count}} hidden)",
|
||||
"opacity": "透明度",
|
||||
"canvasContextMenu": {
|
||||
"newRegionalGuidance": "新規領域ガイダンス",
|
||||
@@ -1890,7 +1815,6 @@
|
||||
"duplicate": "複製",
|
||||
"addLayer": "レイヤーを追加",
|
||||
"rasterLayer": "ラスターレイヤー",
|
||||
"inpaintMasks_withCount_visible": "({{count}}) インペイントマスク",
|
||||
"regional": "領域",
|
||||
"rectangle": "矩形",
|
||||
"moveBackward": "背面へ移動",
|
||||
@@ -2092,7 +2016,6 @@
|
||||
"autoNegative": "オートネガティブ",
|
||||
"enableAutoNegative": "オートネガティブを有効にする",
|
||||
"disableAutoNegative": "オートネガティブを無効にする",
|
||||
"deletePrompt": "プロンプトを削除",
|
||||
"deleteReferenceImage": "参照画像を削除",
|
||||
"showHUD": "HUDを表示",
|
||||
"maskFill": "マスク塗りつぶし",
|
||||
@@ -2104,41 +2027,22 @@
|
||||
"addControlLayer": "$t(controlLayers.controlLayer)を追加します",
|
||||
"addInpaintMask": "$t(controlLayers.inpaintMask)を追加します",
|
||||
"addRegionalGuidance": "$t(controlLayers.regionalGuidance)を追加します",
|
||||
"addGlobalReferenceImage": "$t(controlLayers.globalReferenceImage)を追加します",
|
||||
"addDenoiseLimit": "$t(controlLayers.denoiseLimit)を追加します",
|
||||
"controlLayer": "コントロールレイヤー",
|
||||
"inpaintMask": "インペイントマスク",
|
||||
"referenceImageRegional": "参考画像(地域別)",
|
||||
"referenceImageGlobal": "参考画像(グローバル)",
|
||||
"asRasterLayer": "$t(controlLayers.rasterLayer) として",
|
||||
"asRasterLayerResize": "$t(controlLayers.rasterLayer) として (リサイズ)",
|
||||
"asControlLayer": "$t(controlLayers.controlLayer) として",
|
||||
"asControlLayerResize": "$t(controlLayers.controlLayer) として (リサイズ)",
|
||||
"referenceImage": "参照画像",
|
||||
"sendingToCanvas": "キャンバスに生成をのせる",
|
||||
"sendingToGallery": "生成をギャラリーに送る",
|
||||
"sendToGallery": "ギャラリーに送る",
|
||||
"sendToGalleryDesc": "Invokeを押すとユニークな画像が生成され、ギャラリーに保存されます。",
|
||||
"sendToCanvas": "キャンバスに送る",
|
||||
"newLayerFromImage": "画像から新規レイヤー",
|
||||
"newCanvasFromImage": "画像から新規キャンバス",
|
||||
"newImg2ImgCanvasFromImage": "画像からの新規 Img2Img",
|
||||
"copyToClipboard": "クリップボードにコピー",
|
||||
"sendToCanvasDesc": "Invokeを押すと、進行中の作品がキャンバス上にステージされます。",
|
||||
"viewProgressInViewer": "<Btn>画像ビューア</Btn>で進行状況と出力を表示します。",
|
||||
"viewProgressOnCanvas": "<Btn>キャンバス</Btn> で進行状況とステージ出力を表示します。",
|
||||
"rasterLayer_withCount_other": "ラスターレイヤー",
|
||||
"controlLayer_withCount_other": "コントロールレイヤー",
|
||||
"regionalGuidance_withCount_hidden": "地域ガイダンス({{count}} 件非表示)",
|
||||
"controlLayers_withCount_hidden": "コントロールレイヤー({{count}} 個非表示)",
|
||||
"rasterLayers_withCount_hidden": "ラスター レイヤー ({{count}} 個非表示)",
|
||||
"globalReferenceImages_withCount_hidden": "グローバル参照画像({{count}} 枚非表示)",
|
||||
"regionalGuidance_withCount_visible": "地域ガイダンス ({{count}})",
|
||||
"controlLayers_withCount_visible": "コントロールレイヤー ({{count}})",
|
||||
"rasterLayers_withCount_visible": "ラスターレイヤー({{count}})",
|
||||
"globalReferenceImages_withCount_visible": "グローバル参照画像 ({{count}})",
|
||||
"layer_other": "レイヤー",
|
||||
"layer_withCount_other": "レイヤー ({{count}})",
|
||||
"convertRasterLayerTo": "$t(controlLayers.rasterLayer) を変換する",
|
||||
"convertControlLayerTo": "$t(controlLayers.controlLayer) を変換する",
|
||||
"convertRegionalGuidanceTo": "$t(controlLayers.regionalGuidance) を変換する",
|
||||
@@ -2156,7 +2060,6 @@
|
||||
"pasteToBboxDesc": "新しいレイヤー(Bbox内)",
|
||||
"pasteToCanvas": "キャンバス",
|
||||
"pasteToCanvasDesc": "新しいレイヤー(キャンバス内)",
|
||||
"pastedTo": "{{destination}} に貼り付けました",
|
||||
"transparency": "透明性",
|
||||
"enableTransparencyEffect": "透明効果を有効にする",
|
||||
"disableTransparencyEffect": "透明効果を無効にする",
|
||||
@@ -2169,7 +2072,6 @@
|
||||
"locked": "ロックされています",
|
||||
"unlocked": "ロック解除",
|
||||
"deleteSelected": "選択項目を削除",
|
||||
"stagingOnCanvas": "ステージング画像",
|
||||
"replaceLayer": "レイヤーの置き換え",
|
||||
"pullBboxIntoLayer": "Bboxをレイヤーに引き込む",
|
||||
"pullBboxIntoReferenceImage": "Bboxを参照画像に取り込む",
|
||||
@@ -2177,17 +2079,11 @@
|
||||
"useImage": "画像を使う",
|
||||
"negativePrompt": "ネガティブプロンプト",
|
||||
"beginEndStepPercentShort": "開始/終了 %",
|
||||
"newGallerySession": "新しいギャラリーセッション",
|
||||
"newGallerySessionDesc": "これにより、キャンバスとモデル選択以外のすべての設定がクリアされます。生成した画像はギャラリーに送信されます。",
|
||||
"newCanvasSession": "新規キャンバスセッション",
|
||||
"newCanvasSessionDesc": "これにより、キャンバスとモデル選択以外のすべての設定がクリアされます。生成はキャンバス上でステージングされます。",
|
||||
"resetCanvasLayers": "キャンバスレイヤーをリセット",
|
||||
"resetGenerationSettings": "生成設定をリセット",
|
||||
"replaceCurrent": "現在のものを置き換える",
|
||||
"controlLayerEmptyState": "<UploadButton>画像をアップロード</UploadButton>、<GalleryButton>ギャラリー</GalleryButton>からこのレイヤーに画像をドラッグ、<PullBboxButton>境界ボックスをこのレイヤーにプル</PullBboxButton>、またはキャンバスに描画して開始します。",
|
||||
"referenceImageEmptyStateWithCanvasOptions": "開始するには、<UploadButton>画像をアップロード</UploadButton>するか、<GalleryButton>ギャラリー</GalleryButton>からこの参照画像に画像をドラッグするか、<PullBboxButton>境界ボックスをこの参照画像にプル</PullBboxButton>します。",
|
||||
"referenceImageEmptyState": "開始するには、<UploadButton>画像をアップロード</UploadButton>するか、<GalleryButton>ギャラリー</GalleryButton>からこの参照画像に画像をドラッグします。",
|
||||
"uploadOrDragAnImage": "ギャラリーから画像をドラッグするか、<UploadButton>画像をアップロード</UploadButton>します。",
|
||||
"imageNoise": "画像ノイズ",
|
||||
"denoiseLimit": "ノイズ除去制限",
|
||||
"warnings": {
|
||||
@@ -2253,9 +2149,6 @@
|
||||
"saveAs": "名前を付けて保存",
|
||||
"cancel": "キャンセル",
|
||||
"process": "プロセス",
|
||||
"help1": "ターゲットオブジェクトを1つ選択します。<Bold>含める</Bold>ポイントと<Bold>除外</Bold>ポイントを追加して、レイヤーのどの部分がターゲットオブジェクトの一部であるかを示します。",
|
||||
"help2": "対象オブジェクト内に<Bold>含める</Bold>ポイントを1つ選択するところから始めます。ポイントを追加して選択範囲を絞り込みます。ポイントが少ないほど、通常はより良い結果が得られます。",
|
||||
"help3": "選択を反転して、ターゲットオブジェクト以外のすべてを選択します。",
|
||||
"clickToAdd": "レイヤーをクリックしてポイントを追加します",
|
||||
"dragToMove": "ポイントをドラッグして移動します",
|
||||
"clickToRemove": "ポイントをクリックして削除します"
|
||||
@@ -2356,12 +2249,8 @@
|
||||
"loading": "ロード中...",
|
||||
"steps": "ステップ",
|
||||
"refiner": "Refiner",
|
||||
"negStylePrompt": "ネガティブスタイルプロンプト",
|
||||
"noModelsAvailable": "利用できるモデルがありません",
|
||||
"posStylePrompt": "ポジティブスタイルプロンプト",
|
||||
"cfgScale": "CFGスケール",
|
||||
"concatPromptStyle": "リンキングプロンプトとスタイル",
|
||||
"freePromptStyle": "手動スタイルプロンプト",
|
||||
"posAestheticScore": "ポジティブ美的スコア",
|
||||
"refinerSteps": "リファイナーステップ",
|
||||
"refinerStart": "リファイナースタート",
|
||||
@@ -2379,8 +2268,6 @@
|
||||
"name": "名前",
|
||||
"descending": "降順",
|
||||
"searchPlaceholder": "名前、説明、タグで検索",
|
||||
"projectWorkflows": "プロジェクトワークフロー",
|
||||
"searchWorkflows": "ワークフローを検索",
|
||||
"updated": "アップデート",
|
||||
"published": "公表",
|
||||
"builder": {
|
||||
@@ -2406,10 +2293,8 @@
|
||||
"addToForm": "フォームに追加",
|
||||
"headingPlaceholder": "空の見出し",
|
||||
"nodeFieldTooltip": "ノード フィールドを追加するには、ワークフロー エディターのフィールドにある小さなプラス記号ボタンをクリックするか、フィールド名をフォームにドラッグします。",
|
||||
"workflowBuilderAlphaWarning": "ワークフロービルダーは現在アルファ版です。安定版リリースまでに互換性に影響する変更が発生する可能性があります。",
|
||||
"component": "コンポーネント",
|
||||
"textPlaceholder": "空のテキスト",
|
||||
"emptyRootPlaceholderViewMode": "このワークフローのフォームの作成を開始するには、[編集] をクリックします。",
|
||||
"addOption": "オプションを追加",
|
||||
"singleLine": "単線",
|
||||
"numberInput": "数値入力",
|
||||
@@ -2460,20 +2345,15 @@
|
||||
"convertGraph": "グラフを変換",
|
||||
"downloadWorkflow": "ファイルに保存",
|
||||
"saveWorkflow": "ワークフローを保存",
|
||||
"userWorkflows": "ユーザーワークフロー",
|
||||
"yourWorkflows": "あなたのワークフロー",
|
||||
"edit": "編集",
|
||||
"workflowLibrary": "ワークフローライブラリ",
|
||||
"workflowSaved": "ワークフローが保存されました",
|
||||
"clearWorkflowSearchFilter": "ワークフロー検索フィルタをクリア",
|
||||
"workflowCleared": "ワークフローが作成されました",
|
||||
"autoLayout": "オートレイアウト",
|
||||
"view": "ビュー",
|
||||
"saveChanges": "変更を保存",
|
||||
"noDescription": "説明なし",
|
||||
"recommended": "あなたへのおすすめ",
|
||||
"noRecentWorkflows": "最近のワークフローがありません",
|
||||
"problemLoading": "ワークフローのローディングに関する問題",
|
||||
"newWorkflowCreated": "新しいワークフローが作成されました",
|
||||
"noWorkflows": "ワークフローがありません",
|
||||
"copyShareLink": "共有リンクをコピー",
|
||||
@@ -2481,21 +2361,16 @@
|
||||
"workflowThumbnail": "ワークフローサムネイル",
|
||||
"loadWorkflow": "$t(common.load) ワークフロー",
|
||||
"shared": "共有",
|
||||
"openWorkflow": "ワークフローを開く",
|
||||
"emptyStringPlaceholder": "<空の文字列>",
|
||||
"browseWorkflows": "ワークフローを閲覧する",
|
||||
"saveWorkflowAs": "ワークフローとして保存",
|
||||
"private": "プライベート",
|
||||
"deselectAll": "すべて選択解除",
|
||||
"delete": "削除",
|
||||
"openLibrary": "ライブラリを開く",
|
||||
"loadMore": "もっと読み込む",
|
||||
"saveWorkflowToProject": "ワークフローをプロジェクトに保存",
|
||||
"created": "作成されました",
|
||||
"workflowEditorMenu": "ワークフローエディターメニュー",
|
||||
"defaultWorkflows": "デフォルトワークフロー",
|
||||
"allLoaded": "すべてのワークフローが読み込まれました",
|
||||
"filterByTags": "タグでフィルター",
|
||||
"recentlyOpened": "最近開いた",
|
||||
"opened": "オープン",
|
||||
"deleteWorkflow": "ワークフローを削除",
|
||||
@@ -2541,7 +2416,6 @@
|
||||
"perIterationDesc": "それぞれのいてレーションに別のシードを使う"
|
||||
},
|
||||
"showDynamicPrompts": "ダイナミックプロンプトを表示する",
|
||||
"promptsToGenerate": "生成するプロンプト",
|
||||
"dynamicPrompts": "ダイナミックプロンプト",
|
||||
"loading": "ダイナミックプロンプトを生成...",
|
||||
"maxPrompts": "最大プロンプト"
|
||||
@@ -2567,8 +2441,7 @@
|
||||
"キャンバス: SDXL のアスペクト比がスマートになり、スクロールによるズームが改善されました。"
|
||||
],
|
||||
"readReleaseNotes": "リリースノートを読む",
|
||||
"watchRecentReleaseVideos": "最近のリリースビデオを見る",
|
||||
"watchUiUpdatesOverview": "Watch UI アップデートの概要"
|
||||
"watchRecentReleaseVideos": "最近のリリースビデオを見る"
|
||||
},
|
||||
"supportVideos": {
|
||||
"supportVideos": "サポートビデオ",
|
||||
|
||||
@@ -27,7 +27,6 @@
|
||||
"save": "저장",
|
||||
"created": "생성됨",
|
||||
"error": "에러",
|
||||
"prevPage": "이전 페이지",
|
||||
"ipAdapter": "IP 어댑터",
|
||||
"installed": "설치됨",
|
||||
"accept": "수락",
|
||||
@@ -42,7 +41,6 @@
|
||||
"outputs": "결과물",
|
||||
"unknownError": "알려지지 않은 에러",
|
||||
"linear": "선형",
|
||||
"imageFailedToLoad": "이미지를 로드할 수 없음",
|
||||
"direction": "방향",
|
||||
"data": "데이터",
|
||||
"somethingWentWrong": "뭔가 잘못됐어요",
|
||||
@@ -52,7 +50,6 @@
|
||||
"orderBy": "정렬 기준",
|
||||
"copyError": "$t(gallery.copy) 에러",
|
||||
"learnMore": "더 알아보기",
|
||||
"nextPage": "다음 페이지",
|
||||
"saveAs": "다른 이름으로 저장",
|
||||
"loading": "불러오는 중",
|
||||
"random": "랜덤",
|
||||
@@ -60,18 +57,15 @@
|
||||
"postprocessing": "후처리",
|
||||
"advanced": "고급",
|
||||
"input": "입력",
|
||||
"details": "세부사항",
|
||||
"notInstalled": "설치되지 않음"
|
||||
"details": "세부사항"
|
||||
},
|
||||
"gallery": {
|
||||
"galleryImageSize": "이미지 크기",
|
||||
"gallerySettings": "갤러리 설정",
|
||||
"deleteSelection": "선택 항목 삭제",
|
||||
"featuresWillReset": "이 이미지를 삭제하면 해당 기능이 즉시 재설정됩니다.",
|
||||
"noImagesInGallery": "보여줄 이미지가 없음",
|
||||
"autoSwitchNewImages": "새로운 이미지로 자동 전환",
|
||||
"loading": "불러오는 중",
|
||||
"unableToLoad": "갤러리를 로드할 수 없음",
|
||||
"image": "이미지",
|
||||
"drop": "드랍",
|
||||
"downloadSelection": "선택 항목 다운로드",
|
||||
@@ -151,8 +145,6 @@
|
||||
"loadWorkflow": "Workflow 불러오기",
|
||||
"noOutputRecorded": "기록된 출력 없음",
|
||||
"colorCodeEdgesHelp": "연결된 필드에 따른 색상 코드 선",
|
||||
"hideLegendNodes": "필드 유형 범례 숨기기",
|
||||
"addLinearView": "Linear View에 추가",
|
||||
"float": "실수",
|
||||
"targetNodeFieldDoesNotExist": "잘못된 모서리: 대상/입력 필드 {{node}}. {{field}}이(가) 없습니다",
|
||||
"animatedEdges": "애니메이션 모서리",
|
||||
@@ -160,7 +152,6 @@
|
||||
"nodeTemplate": "노드 템플릿",
|
||||
"nodeOpacity": "노드 불투명도",
|
||||
"sourceNodeDoesNotExist": "잘못된 모서리: 소스/출력 노드 {{node}}이(가) 없습니다",
|
||||
"noFieldsLinearview": "Linear View에 추가된 필드 없음",
|
||||
"nodeSearch": "노드 검색",
|
||||
"inputMayOnlyHaveOneConnection": "입력에 하나의 연결만 있을 수 있습니다",
|
||||
"notes": "메모",
|
||||
@@ -195,7 +186,6 @@
|
||||
"notesDescription": "Workflow에 대한 메모 추가",
|
||||
"colorCodeEdges": "색상-코드 선",
|
||||
"targetNodeDoesNotExist": "잘못된 모서리: 대상/입력 노드 {{node}}이(가) 없습니다",
|
||||
"mismatchedVersion": "잘못된 노드: {{type}} 유형의 {{node}} 노드에 일치하지 않는 버전이 있습니다(업데이트 해보시겠습니까?)",
|
||||
"addNodeToolTip": "노드 추가(Shift+A, Space)",
|
||||
"collectionOrScalarFieldType": "{{name}} 컬렉션|Scalar",
|
||||
"nodeVersion": "노드 버전",
|
||||
@@ -242,7 +232,6 @@
|
||||
"next": "다음",
|
||||
"cancelBatch": "Batch 취소",
|
||||
"back": "back",
|
||||
"batchFieldValues": "Batch 필드 값들",
|
||||
"cancel": "취소",
|
||||
"session": "세션",
|
||||
"time": "시간",
|
||||
@@ -296,8 +285,6 @@
|
||||
"cacheSize": "캐시 크기"
|
||||
},
|
||||
"hrf": {
|
||||
"enableHrf": "이용 가능한 고해상도 고정",
|
||||
"upscaleMethod": "업스케일 방법",
|
||||
"metadata": {
|
||||
"strength": "고해상도 고정 강도",
|
||||
"enabled": "고해상도 고정 사용",
|
||||
@@ -308,12 +295,10 @@
|
||||
"models": {
|
||||
"noMatchingModels": "일치하는 모델 없음",
|
||||
"loading": "로딩중",
|
||||
"noMatchingLoRAs": "일치하는 LoRA 없음",
|
||||
"noModelsAvailable": "사용 가능한 모델이 없음",
|
||||
"addLora": "LoRA 추가",
|
||||
"selectModel": "모델 선택",
|
||||
"noRefinerModelsInstalled": "SDXL Refiner 모델이 설치되지 않음",
|
||||
"noLoRAsInstalled": "설치된 LoRA 없음"
|
||||
"noRefinerModelsInstalled": "SDXL Refiner 모델이 설치되지 않음"
|
||||
},
|
||||
"boards": {
|
||||
"autoAddBoard": "자동 추가 Board",
|
||||
|
||||
@@ -30,12 +30,10 @@
|
||||
"ipAdapter": "IP-adapter",
|
||||
"auto": "Autom.",
|
||||
"controlNet": "ControlNet",
|
||||
"imageFailedToLoad": "Kan afbeelding niet laden",
|
||||
"learnMore": "Meer informatie",
|
||||
"advanced": "Uitgebreid",
|
||||
"file": "Bestand",
|
||||
"installed": "Geïnstalleerd",
|
||||
"notInstalled": "Niet $t(common.installed)",
|
||||
"simple": "Eenvoudig",
|
||||
"somethingWentWrong": "Er ging iets mis",
|
||||
"add": "Voeg toe",
|
||||
@@ -43,14 +41,12 @@
|
||||
"details": "Details",
|
||||
"outputs": "Uitvoeren",
|
||||
"save": "Bewaar",
|
||||
"nextPage": "Volgende pagina",
|
||||
"blue": "Blauw",
|
||||
"alpha": "Alfa",
|
||||
"red": "Rood",
|
||||
"editor": "Editor",
|
||||
"folder": "Map",
|
||||
"format": "structuur",
|
||||
"goTo": "Ga naar",
|
||||
"template": "Sjabloon",
|
||||
"input": "Invoer",
|
||||
"safetensors": "Safetensors",
|
||||
@@ -62,7 +58,6 @@
|
||||
"negativePrompt": "Negatieve prompt",
|
||||
"selected": "Geselecteerd",
|
||||
"orderBy": "Sorteer op",
|
||||
"prevPage": "Vorige pagina",
|
||||
"beta": "Bèta",
|
||||
"copyError": "$t(gallery.copy) Fout",
|
||||
"toResolve": "Op te lossen",
|
||||
@@ -79,21 +74,18 @@
|
||||
"delete": "Verwijder",
|
||||
"direction": "Richting",
|
||||
"error": "Fout",
|
||||
"localSystem": "Lokaal systeem",
|
||||
"unknownError": "Onbekende fout"
|
||||
},
|
||||
"gallery": {
|
||||
"galleryImageSize": "Afbeeldingsgrootte",
|
||||
"gallerySettings": "Instellingen galerij",
|
||||
"autoSwitchNewImages": "Wissel autom. naar nieuwe afbeeldingen",
|
||||
"noImagesInGallery": "Geen afbeeldingen om te tonen",
|
||||
"deleteImage_one": "Verwijder afbeelding",
|
||||
"deleteImage_other": "",
|
||||
"deleteImagePermanent": "Verwijderde afbeeldingen kunnen niet worden hersteld.",
|
||||
"autoAssignBoardOnClick": "Ken automatisch bord toe bij klikken",
|
||||
"featuresWillReset": "Als je deze afbeelding verwijdert, dan worden deze functies onmiddellijk teruggezet.",
|
||||
"loading": "Bezig met laden",
|
||||
"unableToLoad": "Kan galerij niet laden",
|
||||
"downloadSelection": "Download selectie",
|
||||
"currentlyInUse": "Deze afbeelding is momenteel in gebruik door de volgende functies:",
|
||||
"copy": "Kopieer",
|
||||
@@ -199,12 +191,10 @@
|
||||
"scaledHeight": "Geschaalde H",
|
||||
"infillMethod": "Infill-methode",
|
||||
"tileSize": "Grootte tegel",
|
||||
"downloadImage": "Download afbeelding",
|
||||
"usePrompt": "Hergebruik invoertekst",
|
||||
"useSeed": "Hergebruik seed",
|
||||
"useAll": "Hergebruik alles",
|
||||
"info": "Info",
|
||||
"showOptionsPanel": "Toon deelscherm Opties (O of T)",
|
||||
"symmetry": "Symmetrie",
|
||||
"cancel": {
|
||||
"cancel": "Annuleer"
|
||||
@@ -293,15 +283,12 @@
|
||||
"baseModelChangedCleared_one": "Basismodel is gewijzigd: {{count}} niet-compatibel submodel weggehaald of uitgeschakeld",
|
||||
"baseModelChangedCleared_other": "Basismodel is gewijzigd: {{count}} niet-compatibele submodellen weggehaald of uitgeschakeld",
|
||||
"loadedWithWarnings": "Werkstroom geladen met waarschuwingen",
|
||||
"setControlImage": "Ingesteld als controle-afbeelding",
|
||||
"setNodeField": "Ingesteld als knooppuntveld",
|
||||
"imageUploaded": "Afbeelding geüpload",
|
||||
"addedToBoard": "Toegevoegd aan bord",
|
||||
"workflowLoaded": "Werkstroom geladen",
|
||||
"modelAddedSimple": "Model toegevoegd aan wachtrij",
|
||||
"imageUploadFailed": "Fout bij uploaden afbeelding",
|
||||
"workflowDeleted": "Werkstroom verwijderd",
|
||||
"invalidUpload": "Ongeldige upload",
|
||||
"problemRetrievingWorkflow": "Fout bij ophalen van werkstroom",
|
||||
"parameters": "Parameters",
|
||||
"modelImportCanceled": "Importeren model geannuleerd",
|
||||
@@ -325,17 +312,14 @@
|
||||
"zoomOutNodes": "Uitzoomen",
|
||||
"fitViewportNodes": "Aanpassen aan beeld",
|
||||
"hideMinimapnodes": "Minimap verbergen",
|
||||
"showLegendNodes": "Typelegende veld tonen",
|
||||
"zoomInNodes": "Inzoomen",
|
||||
"showMinimapnodes": "Minimap tonen",
|
||||
"hideLegendNodes": "Typelegende veld verbergen",
|
||||
"reloadNodeTemplates": "Herlaad knooppuntsjablonen",
|
||||
"loadWorkflow": "Laad werkstroom",
|
||||
"downloadWorkflow": "Download JSON van werkstroom",
|
||||
"scheduler": "Planner",
|
||||
"missingTemplate": "Ongeldig knooppunt: knooppunt {{node}} van het soort {{type}} heeft een ontbrekend sjabloon (niet geïnstalleerd?)",
|
||||
"workflowDescription": "Korte beschrijving",
|
||||
"versionUnknown": " Versie onbekend",
|
||||
"noNodeSelected": "Geen knooppunt gekozen",
|
||||
"addNode": "Voeg knooppunt toe",
|
||||
"unableToValidateWorkflow": "Kan werkstroom niet valideren",
|
||||
@@ -349,9 +333,7 @@
|
||||
"integer": "Geheel getal",
|
||||
"nodeTemplate": "Sjabloon knooppunt",
|
||||
"nodeOpacity": "Dekking knooppunt",
|
||||
"unableToLoadWorkflow": "Fout bij laden werkstroom",
|
||||
"snapToGrid": "Lijn uit op raster",
|
||||
"noFieldsLinearview": "Geen velden toegevoegd aan lineaire weergave",
|
||||
"nodeSearch": "Zoek naar knooppunten",
|
||||
"updateNode": "Werk knooppunt bij",
|
||||
"version": "Versie",
|
||||
@@ -370,9 +352,7 @@
|
||||
"edge": "Rand",
|
||||
"animatedEdgesHelp": "Animeer gekozen randen en randen verbonden met de gekozen knooppunten",
|
||||
"cannotDuplicateConnection": "Kan geen dubbele verbindingen maken",
|
||||
"unknownTemplate": "Onbekend sjabloon",
|
||||
"noWorkflow": "Geen werkstroom",
|
||||
"removeLinearView": "Verwijder uit lineaire weergave",
|
||||
"workflowTags": "Labels",
|
||||
"fullyContainNodesHelp": "Knooppunten moeten zich volledig binnen het keuzevak bevinden om te worden gekozen",
|
||||
"workflowValidation": "Validatiefout werkstroom",
|
||||
@@ -397,14 +377,11 @@
|
||||
"unknownField": "Onbekend veld",
|
||||
"colorCodeEdges": "Kleurgecodeerde randen",
|
||||
"unknownNode": "Onbekend knooppunt",
|
||||
"mismatchedVersion": "Ongeldig knooppunt: knooppunt {{node}} van het soort {{type}} heeft een niet-overeenkomende versie (probeer het bij te werken?)",
|
||||
"addNodeToolTip": "Voeg knooppunt toe (Shift+A, spatie)",
|
||||
"loadingNodes": "Bezig met laden van knooppunten...",
|
||||
"snapToGridHelp": "Lijn knooppunten uit op raster bij verplaatsing",
|
||||
"workflowSettings": "Instellingen werkstroomeditor",
|
||||
"addLinearView": "Voeg toe aan lineaire weergave",
|
||||
"nodePack": "Knooppuntpakket",
|
||||
"unknownInput": "Onbekende invoer: {{name}}",
|
||||
"sourceNodeFieldDoesNotExist": "Ongeldige rand: bron-/uitvoerveld {{node}}.{{field}} bestaat niet",
|
||||
"collectionFieldType": "Verzameling {{name}}",
|
||||
"deletedInvalidEdge": "Ongeldige hoek {{source}} -> {{target}} verwijderd",
|
||||
@@ -419,7 +396,6 @@
|
||||
"sourceNodeDoesNotExist": "Ongeldige rand: bron-/uitvoerknooppunt {{node}} bestaat niet",
|
||||
"unsupportedArrayItemType": "niet-ondersteunde soort van het array-onderdeel \"{{type}}\"",
|
||||
"targetNodeFieldDoesNotExist": "Ongeldige rand: doel-/invoerveld {{node}}.{{field}} bestaat niet",
|
||||
"reorderLinearView": "Herorden lineaire weergave",
|
||||
"newWorkflowDesc": "Een nieuwe werkstroom aanmaken?",
|
||||
"collectionOrScalarFieldType": "Verzameling|scalair {{name}}",
|
||||
"newWorkflow": "Nieuwe werkstroom",
|
||||
@@ -734,27 +710,21 @@
|
||||
"refinerStart": "Startwaarde verfijning",
|
||||
"scheduler": "Planner",
|
||||
"cfgScale": "CFG-schaal",
|
||||
"negStylePrompt": "Negatieve-stijlprompt",
|
||||
"noModelsAvailable": "Geen modellen beschikbaar",
|
||||
"refiner": "Verfijning",
|
||||
"negAestheticScore": "Negatieve esthetische score",
|
||||
"denoisingStrength": "Sterkte ontruising",
|
||||
"refinermodel": "Verfijningsmodel",
|
||||
"posAestheticScore": "Positieve esthetische score",
|
||||
"concatPromptStyle": "Koppelen van prompt en stijl",
|
||||
"loading": "Bezig met laden...",
|
||||
"steps": "Stappen",
|
||||
"posStylePrompt": "Positieve-stijlprompt",
|
||||
"freePromptStyle": "Handmatige stijlprompt",
|
||||
"refinerSteps": "Aantal stappen verfijner"
|
||||
},
|
||||
"models": {
|
||||
"noMatchingModels": "Geen overeenkomend modellen",
|
||||
"loading": "bezig met laden",
|
||||
"noMatchingLoRAs": "Geen overeenkomende LoRA's",
|
||||
"noModelsAvailable": "Geen modellen beschikbaar",
|
||||
"selectModel": "Kies een model",
|
||||
"noLoRAsInstalled": "Geen LoRA's geïnstalleerd",
|
||||
"noRefinerModelsInstalled": "Geen SDXL-verfijningsmodellen geïnstalleerd",
|
||||
"defaultVAE": "Standaard-VAE",
|
||||
"lora": "LoRA",
|
||||
@@ -822,14 +792,12 @@
|
||||
}
|
||||
},
|
||||
"hrf": {
|
||||
"upscaleMethod": "Opschaalmethode",
|
||||
"metadata": {
|
||||
"strength": "Sterkte oplossing voor hoge resolutie",
|
||||
"method": "Methode oplossing voor hoge resolutie",
|
||||
"enabled": "Oplossing voor hoge resolutie ingeschakeld"
|
||||
},
|
||||
"hrf": "Oplossing voor hoge resolutie",
|
||||
"enableHrf": "Schakel oplossing in voor hoge resolutie"
|
||||
"hrf": "Oplossing voor hoge resolutie"
|
||||
},
|
||||
"prompt": {
|
||||
"addPromptTrigger": "Voeg prompttrigger toe",
|
||||
|
||||
@@ -41,11 +41,9 @@
|
||||
"somethingWentWrong": "Coś poszło nie tak",
|
||||
"green": "Zielony",
|
||||
"red": "Czerwony",
|
||||
"imageFailedToLoad": "Nie można załadować obrazu",
|
||||
"saveAs": "Zapisz jako",
|
||||
"outputs": "Wyjścia",
|
||||
"data": "Dane",
|
||||
"localSystem": "System Lokalny",
|
||||
"t2iAdapter": "Adapter T2I",
|
||||
"selected": "Zaznaczone",
|
||||
"warnings": "Ostrzeżenia",
|
||||
@@ -64,12 +62,10 @@
|
||||
"openInViewer": "Otwórz podgląd",
|
||||
"safetensors": "Bezpieczniki",
|
||||
"ok": "Ok",
|
||||
"goTo": "Idź do",
|
||||
"loadingImage": "wczytywanie zdjęcia",
|
||||
"input": "Wejście",
|
||||
"view": "Podgląd",
|
||||
"learnMore": "Dowiedz się więcej",
|
||||
"notInstalled": "Nie $t(common.installed)",
|
||||
"loadingModel": "Wczytywanie modelu",
|
||||
"postprocessing": "Przetwarzanie końcowe",
|
||||
"random": "Losowo",
|
||||
@@ -83,10 +79,8 @@
|
||||
"delete": "Usuń",
|
||||
"template": "Szablon",
|
||||
"txt2img": "Tekst na obraz",
|
||||
"prevPage": "Poprzednia strona",
|
||||
"file": "Plik",
|
||||
"toResolve": "Do rozwiązania",
|
||||
"nextPage": "Następna strona",
|
||||
"unknownError": "Nieznany błąd",
|
||||
"placeholderSelectAModel": "Wybierz model",
|
||||
"new": "Nowy",
|
||||
@@ -99,7 +93,6 @@
|
||||
"galleryImageSize": "Rozmiar obrazów",
|
||||
"gallerySettings": "Ustawienia galerii",
|
||||
"autoSwitchNewImages": "Przełączaj na nowe obrazy",
|
||||
"noImagesInGallery": "Brak obrazów w galerii",
|
||||
"gallery": "Galeria",
|
||||
"alwaysShowImageSizeBadge": "Zawsze pokazuj odznakę wielkości obrazu",
|
||||
"assetsTab": "Pliki, które wrzuciłeś do użytku w twoich projektach.",
|
||||
@@ -128,12 +121,10 @@
|
||||
"scaledHeight": "Sk. do wys.",
|
||||
"infillMethod": "Metoda wypełniania",
|
||||
"tileSize": "Rozmiar kafelka",
|
||||
"downloadImage": "Pobierz obraz",
|
||||
"usePrompt": "Skopiuj sugestie",
|
||||
"useSeed": "Skopiuj inicjator",
|
||||
"useAll": "Skopiuj wszystko",
|
||||
"info": "Informacje",
|
||||
"showOptionsPanel": "Pokaż panel ustawień"
|
||||
"info": "Informacje"
|
||||
},
|
||||
"settings": {
|
||||
"models": "Modele",
|
||||
@@ -186,8 +177,6 @@
|
||||
"selectedForAutoAdd": "Wybrany do automatycznego dodania",
|
||||
"deleteBoard": "Usuń tablicę",
|
||||
"clearSearch": "Usuń historię",
|
||||
"hideBoards": "Ukryj tablice",
|
||||
"viewBoards": "Zobacz tablice",
|
||||
"addSharedBoard": "Dodaj udostępnioną tablicę",
|
||||
"boards": "Tablice",
|
||||
"addPrivateBoard": "Dodaj prywatną tablicę",
|
||||
@@ -233,8 +222,7 @@
|
||||
"strength": "Moc poprawki wysokiej rozdzielczości",
|
||||
"method": "Metoda High Resolution Fix"
|
||||
},
|
||||
"hrf": "Poprawka \"Wysoka rozdzielczość\"",
|
||||
"enableHrf": "Włącz poprawkę wysokiej rozdzielczości"
|
||||
"hrf": "Poprawka \"Wysoka rozdzielczość\""
|
||||
},
|
||||
"queue": {
|
||||
"cancelTooltip": "Anuluj aktualną pozycję",
|
||||
@@ -296,7 +284,6 @@
|
||||
"completed": "Zakończono",
|
||||
"item": "Pozycja",
|
||||
"failed": "Niepowodzenie",
|
||||
"batchFieldValues": "Masowe Wartości pól",
|
||||
"graphFailedToQueue": "NIe udało się dodać tabeli do kolejki",
|
||||
"workflows": "Przepływy pracy",
|
||||
"next": "Następny",
|
||||
|
||||
@@ -17,8 +17,7 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "Tamanho da Imagem",
|
||||
"gallerySettings": "Configurações de Galeria",
|
||||
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente",
|
||||
"noImagesInGallery": "Sem Imagens na Galeria"
|
||||
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente"
|
||||
},
|
||||
"modelManager": {
|
||||
"modelManager": "Gerente de Modelo",
|
||||
@@ -74,12 +73,10 @@
|
||||
"scaledHeight": "A Escalada",
|
||||
"infillMethod": "Método de Preenchimento",
|
||||
"tileSize": "Tamanho do Ladrilho",
|
||||
"downloadImage": "Baixar Imagem",
|
||||
"usePrompt": "Usar Prompt",
|
||||
"useSeed": "Usar Seed",
|
||||
"useAll": "Usar Todos",
|
||||
"info": "Informações",
|
||||
"showOptionsPanel": "Mostrar Painel de Opções",
|
||||
"symmetry": "Simetria",
|
||||
"copyImage": "Copiar imagem",
|
||||
"denoisingStrength": "A força de remoção de ruído",
|
||||
|
||||
@@ -17,7 +17,6 @@
|
||||
"gallery": {
|
||||
"gallerySettings": "Configurações de Galeria",
|
||||
"autoSwitchNewImages": "Trocar para Novas Imagens Automaticamente",
|
||||
"noImagesInGallery": "Sem Imagens na Galeria",
|
||||
"galleryImageSize": "Tamanho da Imagem"
|
||||
},
|
||||
"modelManager": {
|
||||
@@ -69,7 +68,6 @@
|
||||
"tileSize": "Tamanho do Ladrilho",
|
||||
"symmetry": "Simetria",
|
||||
"usePrompt": "Usar Prompt",
|
||||
"showOptionsPanel": "Mostrar Painel de Opções",
|
||||
"strength": "Força",
|
||||
"upscaling": "Redimensionando",
|
||||
"scaleBeforeProcessing": "Escala Antes do Processamento",
|
||||
@@ -81,7 +79,6 @@
|
||||
"scaledHeight": "A Escalada",
|
||||
"infillMethod": "Método de Preenchimento",
|
||||
"copyImage": "Copiar imagem",
|
||||
"downloadImage": "Descarregar Imagem",
|
||||
"useSeed": "Usar Seed",
|
||||
"useAll": "Usar Todos",
|
||||
"info": "Informações"
|
||||
|
||||
@@ -38,7 +38,6 @@
|
||||
"save": "Сохранить",
|
||||
"created": "Создано",
|
||||
"error": "Ошибка",
|
||||
"prevPage": "Предыдущая страница",
|
||||
"simple": "Простой",
|
||||
"ipAdapter": "IP Adapter",
|
||||
"installed": "Установлено",
|
||||
@@ -49,7 +48,6 @@
|
||||
"template": "Шаблон",
|
||||
"outputs": "результаты",
|
||||
"unknownError": "Неизвестная ошибка",
|
||||
"imageFailedToLoad": "Невозможно загрузить изображение",
|
||||
"direction": "Направление",
|
||||
"data": "Данные",
|
||||
"somethingWentWrong": "Что-то пошло не так",
|
||||
@@ -58,11 +56,9 @@
|
||||
"orderBy": "Сортировать по",
|
||||
"copyError": "Ошибка $t(gallery.copy)",
|
||||
"learnMore": "Узнать больше",
|
||||
"nextPage": "Следущая страница",
|
||||
"saveAs": "Сохранить как",
|
||||
"input": "Вход",
|
||||
"details": "Детали",
|
||||
"notInstalled": "Нет $t(common.installed)",
|
||||
"or": "или",
|
||||
"aboutHeading": "Владей своей творческой силой",
|
||||
"red": "Красный",
|
||||
@@ -71,7 +67,6 @@
|
||||
"alpha": "Альфа",
|
||||
"toResolve": "Чтоб решить",
|
||||
"copy": "Копировать",
|
||||
"localSystem": "Локальная система",
|
||||
"aboutDesc": "Используя Invoke для работы? Проверьте это:",
|
||||
"add": "Добавить",
|
||||
"beta": "Бета",
|
||||
@@ -79,7 +74,6 @@
|
||||
"positivePrompt": "Позитивный запрос",
|
||||
"negativePrompt": "Негативный запрос",
|
||||
"editor": "Редактор",
|
||||
"goTo": "Перейти к",
|
||||
"tab": "Вкладка",
|
||||
"enabled": "Включено",
|
||||
"disabled": "Отключено",
|
||||
@@ -101,7 +95,6 @@
|
||||
"galleryImageSize": "Размер изображений",
|
||||
"gallerySettings": "Настройка галереи",
|
||||
"autoSwitchNewImages": "Автоматически выбирать новые",
|
||||
"noImagesInGallery": "Изображений нет",
|
||||
"deleteImagePermanent": "Удаленные изображения невозможно восстановить.",
|
||||
"deleteImage_one": "Удалить изображение",
|
||||
"deleteImage_few": "Удалить {{count}} изображения",
|
||||
@@ -110,7 +103,6 @@
|
||||
"deleteSelection": "Удалить выделенное",
|
||||
"featuresWillReset": "Если вы удалите это изображение, эти функции будут немедленно сброшены.",
|
||||
"loading": "Загрузка",
|
||||
"unableToLoad": "Невозможно загрузить галерею",
|
||||
"image": "изображение",
|
||||
"drop": "перебросить",
|
||||
"downloadSelection": "Скачать выделенное",
|
||||
@@ -136,7 +128,6 @@
|
||||
"compareHelp4": "Нажмите <Kbd>Z</Kbd> или <Kbd>Esc</Kbd> для выхода.",
|
||||
"compareImage": "Сравнить изображение",
|
||||
"viewerImage": "Изображение просмотрщика",
|
||||
"selectAnImageToCompare": "Выберите изображение для сравнения",
|
||||
"slider": "Слайдер",
|
||||
"sideBySide": "Бок о бок",
|
||||
"compareHelp1": "Удерживайте <Kbd>Alt</Kbd> при нажатии на изображение в галерее или при помощи клавиш со стрелками, чтобы изменить сравниваемое изображение.",
|
||||
@@ -154,11 +145,8 @@
|
||||
"exitBoardSearch": "Выйти из поиска досок",
|
||||
"go": "Перейти",
|
||||
"exitSearch": "Выйти из поиска изображений",
|
||||
"jump": "Пыгнуть",
|
||||
"move": "Двигать",
|
||||
"gallery": "Галерея",
|
||||
"openViewer": "Открыть просмотрщик",
|
||||
"closeViewer": "Закрыть просмотрщик",
|
||||
"imagesTab": "Изображения, созданные и сохраненные в Invoke.",
|
||||
"assetsTab": "Файлы, которые вы загрузили для использования в своих проектах.",
|
||||
"boardsSettings": "Настройки доски",
|
||||
@@ -574,8 +562,6 @@
|
||||
"noModelsInstalled": "Нет установленных моделей",
|
||||
"noModelsInstalledDesc1": "Установите модели с помощью",
|
||||
"noMatchingModels": "Нет подходящих моделей",
|
||||
"ipAdapters": "IP адаптеры",
|
||||
"starterModelsInModelManager": "Стартовые модели можно найти в Менеджере моделей",
|
||||
"learnMoreAboutSupportedModels": "Подробнее о поддерживаемых моделях",
|
||||
"t5Encoder": "T5 энкодер",
|
||||
"spandrelImageToImage": "Image to Image (Spandrel)",
|
||||
@@ -612,12 +598,10 @@
|
||||
"scaledHeight": "Масштаб В",
|
||||
"infillMethod": "Способ заполнения",
|
||||
"tileSize": "Размер области",
|
||||
"downloadImage": "Скачать",
|
||||
"usePrompt": "Использовать запрос",
|
||||
"useSeed": "Использовать сид",
|
||||
"useAll": "Использовать все",
|
||||
"info": "Метаданные",
|
||||
"showOptionsPanel": "Показать панель настроек",
|
||||
"cancel": {
|
||||
"cancel": "Отмена"
|
||||
},
|
||||
@@ -643,10 +627,6 @@
|
||||
"missingFieldTemplate": "Отсутствует шаблон поля",
|
||||
"addingImagesTo": "Добавление изображений в",
|
||||
"invoke": "Создать",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), ширина рамки {{width}}",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), высота рамки {{height}}",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), масштабированная высота рамки {{height}}",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16) масштабированная ширина рамки {{width}}",
|
||||
"noFLUXVAEModelSelected": "Для генерации FLUX не выбрана модель VAE",
|
||||
"noT5EncoderModelSelected": "Для генерации FLUX не выбрана модель T5 энкодера",
|
||||
"canvasIsFiltering": "Холст фильтруется",
|
||||
@@ -732,9 +712,6 @@
|
||||
"baseModelChangedCleared_few": "Очищено или отключено {{count}} несовместимых подмодели",
|
||||
"baseModelChangedCleared_many": "Очищено или отключено {{count}} несовместимых подмоделей",
|
||||
"loadedWithWarnings": "Рабочий процесс загружен с предупреждениями",
|
||||
"setControlImage": "Установить как контрольное изображение",
|
||||
"setNodeField": "Установить как поле узла",
|
||||
"invalidUpload": "Неверная загрузка",
|
||||
"imageUploaded": "Изображение загружено",
|
||||
"addedToBoard": "Добавлено в активы доски {{name}}",
|
||||
"workflowLoaded": "Рабочий процесс загружен",
|
||||
@@ -763,21 +740,14 @@
|
||||
"sentToCanvas": "Отправить на холст",
|
||||
"unableToLoadImage": "Невозможно загрузить изображение",
|
||||
"unableToLoadImageMetadata": "Невозможно загрузить метаданные изображения",
|
||||
"imageSaved": "Изображение сохранено",
|
||||
"stylePresetLoaded": "Предустановка стиля загружена",
|
||||
"imageNotLoadedDesc": "Не удалось найти изображение",
|
||||
"imageSavingFailed": "Не удалось сохранить изображение",
|
||||
"problemCopyingLayer": "Не удалось скопировать слой",
|
||||
"unableToLoadStylePreset": "Невозможно загрузить предустановку стиля",
|
||||
"layerCopiedToClipboard": "Слой скопирован в буфер обмена",
|
||||
"sentToUpscale": "Отправить на увеличение",
|
||||
"layerSavedToAssets": "Слой сохранен в активах",
|
||||
"linkCopied": "Ссылка скопирована",
|
||||
"addedToUncategorized": "Добавлено в активы доски $t(boards.uncategorized)",
|
||||
"imagesWillBeAddedTo": "Загруженные изображения будут добавлены в активы доски {{boardName}}.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_one": "Должно быть не более {{count}} изображения в формате PNG или JPEG.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_few": "Должно быть не более {{count}} изображений в формате PNG или JPEG.",
|
||||
"uploadFailedInvalidUploadDesc_withCount_many": "Должно быть не более {{count}} изображений в формате PNG или JPEG."
|
||||
"imagesWillBeAddedTo": "Загруженные изображения будут добавлены в активы доски {{boardName}}."
|
||||
},
|
||||
"accessibility": {
|
||||
"uploadImage": "Загрузить изображение",
|
||||
@@ -799,15 +769,12 @@
|
||||
"zoomInNodes": "Увеличьте масштаб",
|
||||
"zoomOutNodes": "Уменьшите масштаб",
|
||||
"fitViewportNodes": "Уместить вид",
|
||||
"showLegendNodes": "Показать тип поля",
|
||||
"hideMinimapnodes": "Скрыть миникарту",
|
||||
"hideLegendNodes": "Скрыть тип поля",
|
||||
"showMinimapnodes": "Показать миникарту",
|
||||
"loadWorkflow": "Загрузить рабочий процесс",
|
||||
"reloadNodeTemplates": "Перезагрузить шаблоны узлов",
|
||||
"downloadWorkflow": "Скачать JSON рабочего процесса",
|
||||
"addNode": "Добавить узел",
|
||||
"addLinearView": "Добавить в линейный вид",
|
||||
"animatedEdges": "Анимированные ребра",
|
||||
"animatedEdgesHelp": "Анимация выбранных ребер и ребер, соединенных с выбранными узлами",
|
||||
"boolean": "Логические значения",
|
||||
@@ -819,7 +786,6 @@
|
||||
"workflowDescription": "Краткое описание",
|
||||
"inputFieldTypeParseError": "Невозможно разобрать тип поля ввода {{node}}.{{field}} ({{message}})",
|
||||
"unsupportedAnyOfLength": "слишком много элементов объединения ({{count}})",
|
||||
"versionUnknown": " Версия неизвестна",
|
||||
"unsupportedArrayItemType": "неподдерживаемый тип элемента массива \"{{type}}\"",
|
||||
"noNodeSelected": "Узел не выбран",
|
||||
"unableToValidateWorkflow": "Невозможно проверить рабочий процесс",
|
||||
@@ -837,10 +803,8 @@
|
||||
"nodeTemplate": "Шаблон узла",
|
||||
"nodeOpacity": "Непрозрачность узла",
|
||||
"sourceNodeDoesNotExist": "Недопустимое ребро: исходный/выходной узел {{node}} не существует",
|
||||
"unableToLoadWorkflow": "Невозможно загрузить рабочий процесс",
|
||||
"unableToExtractEnumOptions": "невозможно извлечь параметры перечисления",
|
||||
"snapToGrid": "Привязка к сетке",
|
||||
"noFieldsLinearview": "Нет полей, добавленных в линейный вид",
|
||||
"unableToParseFieldType": "невозможно проанализировать тип поля",
|
||||
"nodeSearch": "Поиск узлов",
|
||||
"updateNode": "Обновить узел",
|
||||
@@ -861,9 +825,7 @@
|
||||
"edge": "Край",
|
||||
"sourceNodeFieldDoesNotExist": "Неверный край: поле источника/вывода {{node}}.{{field}} не существует",
|
||||
"cannotDuplicateConnection": "Невозможно создать дубликаты соединений",
|
||||
"unknownTemplate": "Неизвестный шаблон",
|
||||
"noWorkflow": "Нет рабочего процесса",
|
||||
"removeLinearView": "Удалить из линейного вида",
|
||||
"workflowTags": "Теги",
|
||||
"fullyContainNodesHelp": "Чтобы узлы были выбраны, они должны полностью находиться в поле выбора",
|
||||
"unableToGetWorkflowVersion": "Не удалось получить версию схемы рабочего процесса",
|
||||
@@ -896,7 +858,6 @@
|
||||
"colorCodeEdges": "Ребра с цветовой кодировкой",
|
||||
"unknownNode": "Неизвестный узел",
|
||||
"targetNodeDoesNotExist": "Недопустимое ребро: целевой/входной узел {{node}} не существует",
|
||||
"mismatchedVersion": "Недопустимый узел: узел {{node}} типа {{type}} имеет несоответствующую версию (попробовать обновить?)",
|
||||
"unknownFieldType": "$t(nodes.unknownField) тип: {{type}}",
|
||||
"collectionOrScalarFieldType": "{{name}} (Один или коллекция)",
|
||||
"betaDesc": "Этот вызов находится в бета-версии. Пока он не станет стабильным, в нем могут происходить изменения при обновлении приложений. Мы планируем поддерживать этот вызов в течение длительного времени.",
|
||||
@@ -905,14 +866,12 @@
|
||||
"snapToGridHelp": "Привязка узлов к сетке при перемещении",
|
||||
"workflowSettings": "Настройки редактора рабочих процессов",
|
||||
"deletedInvalidEdge": "Удалено недопустимое ребро {{source}} -> {{target}}",
|
||||
"unknownInput": "Неизвестный вход: {{name}}",
|
||||
"newWorkflow": "Новый рабочий процесс",
|
||||
"newWorkflowDesc": "Создать новый рабочий процесс?",
|
||||
"clearWorkflow": "Очистить рабочий процесс",
|
||||
"newWorkflowDesc2": "Текущий рабочий процесс имеет несохраненные изменения.",
|
||||
"clearWorkflowDesc": "Очистить этот рабочий процесс и создать новый?",
|
||||
"clearWorkflowDesc2": "Текущий рабочий процесс имеет несохраненные измерения.",
|
||||
"reorderLinearView": "Изменить порядок линейного просмотра",
|
||||
"viewMode": "Использовать в линейном представлении",
|
||||
"editMode": "Открыть в редакторе узлов",
|
||||
"resetToDefaultValue": "Сбросить к стандартному значкнию",
|
||||
@@ -974,8 +933,6 @@
|
||||
"addPrivateBoard": "Добавить личную доску",
|
||||
"private": "Личные доски",
|
||||
"shared": "Общие доски",
|
||||
"hideBoards": "Скрыть доски",
|
||||
"viewBoards": "Просмотреть доски",
|
||||
"noBoards": "Нет досок {{boardType}}",
|
||||
"deletedPrivateBoardsCannotbeRestored": "Удаленные доски не могут быть восстановлены. Выбор «Удалить только доску» переведет изображения в приватное состояние без категории для создателя изображения.",
|
||||
"updateBoardError": "Ошибка обновления доски"
|
||||
@@ -1404,8 +1361,6 @@
|
||||
"noRecallParameters": "Параметры для вызова не найдены",
|
||||
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
||||
"parameterSet": "Параметр {{parameter}} установлен",
|
||||
"parsingFailed": "Не удалось выполнить синтаксический анализ",
|
||||
"recallParameter": "Отозвать {{label}}",
|
||||
"allPrompts": "Все запросы",
|
||||
"imageDimensions": "Размеры изображения",
|
||||
"canvasV2Metadata": "Холст",
|
||||
@@ -1456,7 +1411,6 @@
|
||||
"next": "Следующий",
|
||||
"cancelBatch": "Отменить пакет",
|
||||
"back": "задний",
|
||||
"batchFieldValues": "Пакетные значения полей",
|
||||
"cancel": "Отмена",
|
||||
"session": "Сессия",
|
||||
"time": "Время",
|
||||
@@ -1491,18 +1445,14 @@
|
||||
"refinerStart": "Запуск доработчика",
|
||||
"scheduler": "Планировщик",
|
||||
"cfgScale": "Шкала точности (CFG)",
|
||||
"negStylePrompt": "Негативный запрос стиля",
|
||||
"noModelsAvailable": "Нет доступных моделей",
|
||||
"refiner": "Доработчик",
|
||||
"negAestheticScore": "Отрицательная эстетическая оценка",
|
||||
"denoisingStrength": "Шумоподавление",
|
||||
"refinermodel": "Дорабатывающая модель",
|
||||
"posAestheticScore": "Положительная эстетическая оценка",
|
||||
"concatPromptStyle": "Связывание запроса и стиля",
|
||||
"loading": "Загрузка...",
|
||||
"steps": "Шаги",
|
||||
"posStylePrompt": "Запрос стиля",
|
||||
"freePromptStyle": "Ручной запрос стиля",
|
||||
"refinerSteps": "Шаги доработчика"
|
||||
},
|
||||
"invocationCache": {
|
||||
@@ -1527,20 +1477,15 @@
|
||||
"workflowEditorMenu": "Меню редактора рабочего процесса",
|
||||
"workflowName": "Имя рабочего процесса",
|
||||
"saveWorkflow": "Сохранить рабочий процесс",
|
||||
"openWorkflow": "Открытый рабочий процесс",
|
||||
"clearWorkflowSearchFilter": "Очистить фильтр поиска рабочих процессов",
|
||||
"workflowLibrary": "Библиотека",
|
||||
"downloadWorkflow": "Сохранить в файл",
|
||||
"workflowSaved": "Рабочий процесс сохранен",
|
||||
"unnamedWorkflow": "Безымянный рабочий процесс",
|
||||
"savingWorkflow": "Сохранение рабочего процесса...",
|
||||
"problemLoading": "Проблема с загрузкой рабочих процессов",
|
||||
"loading": "Загрузка рабочих процессов",
|
||||
"searchWorkflows": "Поиск рабочих процессов",
|
||||
"problemSavingWorkflow": "Проблема с сохранением рабочего процесса",
|
||||
"deleteWorkflow": "Удалить рабочий процесс",
|
||||
"workflows": "Рабочие процессы",
|
||||
"noDescription": "Без описания",
|
||||
"uploadWorkflow": "Загрузить из файла",
|
||||
"newWorkflowCreated": "Создан новый рабочий процесс",
|
||||
"saveWorkflowToProject": "Сохранить рабочий процесс в проект",
|
||||
@@ -1556,9 +1501,6 @@
|
||||
"convertGraph": "Конвертировать график",
|
||||
"loadFromGraph": "Загрузка рабочего процесса из графика",
|
||||
"autoLayout": "Автоматическое расположение",
|
||||
"userWorkflows": "Пользовательские рабочие процессы",
|
||||
"projectWorkflows": "Рабочие процессы проекта",
|
||||
"defaultWorkflows": "Стандартные рабочие процессы",
|
||||
"deleteWorkflow2": "Вы уверены, что хотите удалить этот рабочий процесс? Это нельзя отменить.",
|
||||
"chooseWorkflowFromLibrary": "Выбрать рабочий процесс из библиотеки",
|
||||
"edit": "Редактировать",
|
||||
@@ -1568,8 +1510,6 @@
|
||||
"delete": "Удалить"
|
||||
},
|
||||
"hrf": {
|
||||
"enableHrf": "Включить исправление высокого разрешения",
|
||||
"upscaleMethod": "Метод увеличения",
|
||||
"metadata": {
|
||||
"strength": "Сила исправления высокого разрешения",
|
||||
"enabled": "Исправление высокого разрешения включено",
|
||||
@@ -1580,12 +1520,10 @@
|
||||
"models": {
|
||||
"noMatchingModels": "Нет подходящих моделей",
|
||||
"loading": "загрузка",
|
||||
"noMatchingLoRAs": "Нет подходящих LoRA",
|
||||
"noModelsAvailable": "Нет доступных моделей",
|
||||
"addLora": "Добавить LoRA",
|
||||
"selectModel": "Выберите модель",
|
||||
"noRefinerModelsInstalled": "Дорабатывающие модели SDXL не установлены",
|
||||
"noLoRAsInstalled": "Нет установленных LoRA",
|
||||
"lora": "LoRA",
|
||||
"defaultVAE": "Стандартное VAE",
|
||||
"concepts": "LoRA"
|
||||
@@ -1620,7 +1558,6 @@
|
||||
"moveForward": "Переместить вперёд",
|
||||
"moveBackward": "Переместить назад",
|
||||
"autoNegative": "Авто негатив",
|
||||
"deletePrompt": "Удалить запрос",
|
||||
"rectangle": "Прямоугольник",
|
||||
"addNegativePrompt": "Добавить $t(controlLayers.negativePrompt)",
|
||||
"regionalGuidance": "Региональная точность",
|
||||
@@ -1790,7 +1727,6 @@
|
||||
},
|
||||
"addReferenceImage": "Добавить $t(controlLayers.referenceImage)",
|
||||
"inpaintMask": "Маска перерисовки",
|
||||
"sendToGalleryDesc": "При нажатии кнопки Invoke создается изображение и сохраняется в вашей галерее.",
|
||||
"sendToCanvas": "Отправить на холст",
|
||||
"regionalGuidance_withCount_one": "$t(controlLayers.regionalGuidance)",
|
||||
"regionalGuidance_withCount_few": "Региональных точности",
|
||||
@@ -1802,7 +1738,6 @@
|
||||
"inpaintMask_withCount_one": "$t(controlLayers.inpaintMask)",
|
||||
"inpaintMask_withCount_few": "Маски перерисовки",
|
||||
"inpaintMask_withCount_many": "Масок перерисовки",
|
||||
"globalReferenceImages_withCount_visible": "Глобальные эталонные изображения ({{count}})",
|
||||
"controlMode": {
|
||||
"prompt": "Запрос",
|
||||
"controlMode": "Режим контроля",
|
||||
@@ -1838,7 +1773,6 @@
|
||||
"pullBboxIntoReferenceImage": "Поместить рамку в эталонное изображение",
|
||||
"enableAutoNegative": "Включить авто негатив",
|
||||
"maskFill": "Заполнение маски",
|
||||
"viewProgressInViewer": "Просматривайте прогресс и результаты в <Btn>Просмотрщике изображений</Btn>.",
|
||||
"tool": {
|
||||
"move": "Двигать",
|
||||
"bbox": "Ограничительная рамка",
|
||||
@@ -1849,18 +1783,10 @@
|
||||
"colorPicker": "Подборщик цветов"
|
||||
},
|
||||
"rasterLayer": "Растровый слой",
|
||||
"sendingToCanvas": "Постановка генераций на холст",
|
||||
"rasterLayers_withCount_visible": "Растровые слои ({{count}})",
|
||||
"regionalGuidance_withCount_hidden": "Региональная точность ({{count}} скрыто)",
|
||||
"enableTransparencyEffect": "Включить эффект прозрачности",
|
||||
"hidingType": "Скрыть {{type}}",
|
||||
"addRegionalGuidance": "Добавить $t(controlLayers.regionalGuidance)",
|
||||
"sendingToGallery": "Отправка генераций в галерею",
|
||||
"viewProgressOnCanvas": "Просматривайте прогресс и результаты этапов на <Btn>Холсте</Btn>.",
|
||||
"controlLayers_withCount_hidden": "Контрольные слои ({{count}} скрыто)",
|
||||
"rasterLayers_withCount_hidden": "Растровые слои ({{count}} скрыто)",
|
||||
"deleteSelected": "Удалить выбранное",
|
||||
"stagingOnCanvas": "Постановка изображений на",
|
||||
"pullBboxIntoLayer": "Поместить рамку в слой",
|
||||
"locked": "Заблокировано",
|
||||
"replaceLayer": "Заменить слой",
|
||||
@@ -1869,16 +1795,10 @@
|
||||
"addRasterLayer": "Добавить $t(controlLayers.rasterLayer)",
|
||||
"addControlLayer": "Добавить $t(controlLayers.controlLayer)",
|
||||
"addInpaintMask": "Добавить $t(controlLayers.inpaintMask)",
|
||||
"inpaintMasks_withCount_hidden": "Маски перерисовки ({{count}} скрыто)",
|
||||
"regionalGuidance_withCount_visible": "Региональная точность ({{count}})",
|
||||
"newGallerySessionDesc": "Это очистит холст и все настройки, кроме выбранной модели. Генерации будут отправлены в галерею.",
|
||||
"newCanvasSession": "Новая сессия холста",
|
||||
"newCanvasSessionDesc": "Это очистит холст и все настройки, кроме выбора модели. Генерации будут размещены на холсте.",
|
||||
"cropLayerToBbox": "Обрезать слой по ограничительной рамке",
|
||||
"clipToBbox": "Обрезка штрихов в рамке",
|
||||
"outputOnlyMaskedRegions": "Вывод только маскированных областей",
|
||||
"duplicate": "Дублировать",
|
||||
"inpaintMasks_withCount_visible": "Маски перерисовки ({{count}})",
|
||||
"layer_one": "Слой",
|
||||
"layer_few": "Слоя",
|
||||
"layer_many": "Слоев",
|
||||
@@ -1897,33 +1817,20 @@
|
||||
},
|
||||
"disableAutoNegative": "Отключить авто негатив",
|
||||
"deleteReferenceImage": "Удалить эталонное изображение",
|
||||
"controlLayers_withCount_visible": "Контрольные слои ({{count}})",
|
||||
"rasterLayer_withCount_one": "$t(controlLayers.rasterLayer)",
|
||||
"rasterLayer_withCount_few": "Растровых слоя",
|
||||
"rasterLayer_withCount_many": "Растровых слоев",
|
||||
"transparency": "Прозрачность",
|
||||
"weight": "Вес",
|
||||
"newGallerySession": "Новая сессия галереи",
|
||||
"sendToCanvasDesc": "Нажатие кнопки Invoke отображает вашу текущую работу на холсте.",
|
||||
"globalReferenceImages_withCount_hidden": "Глобальные эталонные изображения ({{count}} скрыто)",
|
||||
"layer_withCount_one": "Слой ({{count}})",
|
||||
"layer_withCount_few": "Слои ({{count}})",
|
||||
"layer_withCount_many": "Слои ({{count}})",
|
||||
"disableTransparencyEffect": "Отключить эффект прозрачности",
|
||||
"showingType": "Показать {{type}}",
|
||||
"dynamicGrid": "Динамическая сетка",
|
||||
"logDebugInfo": "Писать отладочную информацию",
|
||||
"unlocked": "Разблокировано",
|
||||
"showProgressOnCanvas": "Показать прогресс на холсте",
|
||||
"globalReferenceImage_withCount_one": "$t(controlLayers.globalReferenceImage)",
|
||||
"globalReferenceImage_withCount_few": "Глобальных эталонных изображения",
|
||||
"globalReferenceImage_withCount_many": "Глобальных эталонных изображений",
|
||||
"regionalReferenceImage": "Региональное эталонное изображение",
|
||||
"globalReferenceImage": "Глобальное эталонное изображение",
|
||||
"sendToGallery": "Отправить в галерею",
|
||||
"referenceImage": "Эталонное изображение",
|
||||
"addGlobalReferenceImage": "Добавить $t(controlLayers.globalReferenceImage)",
|
||||
"newImg2ImgCanvasFromImage": "Новое img2img из изображения"
|
||||
"referenceImage": "Эталонное изображение"
|
||||
},
|
||||
"ui": {
|
||||
"tabs": {
|
||||
|
||||
@@ -28,7 +28,6 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "Bildstorlek",
|
||||
"gallerySettings": "Galleriinställningar",
|
||||
"noImagesInGallery": "Inga bilder i galleriet",
|
||||
"autoSwitchNewImages": "Ändra automatiskt till nya bilder"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -36,12 +36,10 @@
|
||||
"communityLabel": "Topluluk",
|
||||
"back": "Geri",
|
||||
"areYouSure": "Emin misiniz?",
|
||||
"notInstalled": "$t(common.installed) Değil",
|
||||
"openInNewTab": "Yeni Sekmede Aç",
|
||||
"aboutHeading": "Yaratıcı Gücünüzün Sahibi Olun",
|
||||
"load": "Yükle",
|
||||
"loading": "Yükleniyor",
|
||||
"localSystem": "Yerel Sistem",
|
||||
"inpaint": "içboyama",
|
||||
"modelManager": "Model Yöneticisi",
|
||||
"orderBy": "Sırala",
|
||||
@@ -65,11 +63,8 @@
|
||||
"format": "biçim",
|
||||
"details": "Ayrıntılar",
|
||||
"error": "Hata",
|
||||
"imageFailedToLoad": "Görsel Yüklenemedi",
|
||||
"safetensors": "Safetensors",
|
||||
"upload": "Yükle",
|
||||
"nextPage": "Sonraki Sayfa",
|
||||
"prevPage": "Önceki Sayfa",
|
||||
"dontAskMeAgain": "Bir daha sorma",
|
||||
"delete": "Kaldır",
|
||||
"direction": "Yön",
|
||||
@@ -181,7 +176,6 @@
|
||||
"session": "Oturum",
|
||||
"batchQueued": "Toplu İş Sıraya Alındı",
|
||||
"notReady": "Sıraya Alınamadı",
|
||||
"batchFieldValues": "Toplu İş Değişkenleri",
|
||||
"graphFailedToQueue": "Çizge sıraya alınamadı",
|
||||
"graphQueued": "Çizge sıraya alındı"
|
||||
},
|
||||
@@ -207,12 +201,10 @@
|
||||
"image": "görsel",
|
||||
"galleryImageSize": "Görsel Boyutu",
|
||||
"copy": "Kopyala",
|
||||
"noImagesInGallery": "Gösterilecek Görsel Yok",
|
||||
"autoSwitchNewImages": "Yeni Görseli Biter Bitmez Gör",
|
||||
"currentlyInUse": "Bu görsel şurada kullanımda:",
|
||||
"deleteImage_one": "Görseli Sil",
|
||||
"deleteImage_other": "",
|
||||
"unableToLoad": "Galeri Yüklenemedi",
|
||||
"downloadSelection": "Seçileni İndir",
|
||||
"dropOrUpload": "$t(gallery.drop) ya da Yükle",
|
||||
"dropToUpload": "Yüklemek için $t(gallery.drop)",
|
||||
@@ -220,13 +212,11 @@
|
||||
},
|
||||
"hrf": {
|
||||
"hrf": "Yüksek Çözünürlük Kürü",
|
||||
"enableHrf": "Yüksek Çözünürlük Kürünü Aç",
|
||||
"metadata": {
|
||||
"enabled": "Yüksek Çözünürlük Kürü Açık",
|
||||
"strength": "Yüksek Çözünürlük Kürü Etkisi",
|
||||
"method": "Yüksek Çözünürlük Kürü Yöntemi"
|
||||
},
|
||||
"upscaleMethod": "Büyütme Yöntemi"
|
||||
}
|
||||
},
|
||||
"hotkeys": {
|
||||
"noHotkeysFound": "Kısayol Tuşu Bulanamadı",
|
||||
@@ -256,7 +246,6 @@
|
||||
"unknownErrorValidatingWorkflow": "İş akışını doğrulamada bilinmeyen bir sorun",
|
||||
"unableToGetWorkflowVersion": "İş akışı sürümüne ulaşılamadı",
|
||||
"newWorkflowDesc2": "Geçerli iş akışında kaydedilmemiş değişiklikler var.",
|
||||
"unableToLoadWorkflow": "İş Akışı Yüklenemedi",
|
||||
"cannotConnectInputToInput": "Giriş girişe bağlanamaz",
|
||||
"zoomInNodes": "Yakınlaştır",
|
||||
"boolean": "Boole Değeri",
|
||||
@@ -267,16 +256,12 @@
|
||||
"cannotDuplicateConnection": "Kopya bağlantılar yaratılamaz"
|
||||
},
|
||||
"workflows": {
|
||||
"searchWorkflows": "İş Akışlarında Ara",
|
||||
"workflowName": "İş Akışı Adı",
|
||||
"problemSavingWorkflow": "İş Akışını Kaydetmede Sorun",
|
||||
"saveWorkflow": "İş Akışını Kaydet",
|
||||
"uploadWorkflow": "Dosyadan Yükle",
|
||||
"newWorkflowCreated": "Yeni İş Akışı Yaratıldı",
|
||||
"problemLoading": "İş Akışlarını Yüklemede Sorun",
|
||||
"loading": "İş Akışları Yükleniyor",
|
||||
"noDescription": "Tanımsız",
|
||||
"clearWorkflowSearchFilter": "İş Akışı Aramasını Resetle",
|
||||
"workflowEditorMenu": "İş Akışı Düzenleyici Menüsü",
|
||||
"downloadWorkflow": "İndir",
|
||||
"saveWorkflowAs": "İş Akışını Farklı Kaydet",
|
||||
@@ -328,7 +313,6 @@
|
||||
"noiseThreshold": "Gürültü Eşiği",
|
||||
"seed": "Tohum",
|
||||
"imageActions": "Görsel İşlemleri",
|
||||
"showOptionsPanel": "Yan Paneli Göster (O ya da T)",
|
||||
"shuffle": "Kar",
|
||||
"usePrompt": "İstemi Kullan",
|
||||
"setToOptimalSizeTooSmall": "$t(parameters.setToOptimalSize) (çok küçük olabilir)",
|
||||
@@ -346,7 +330,6 @@
|
||||
"perlinNoise": "Perlin Gürültüsü",
|
||||
"scaledWidth": "Ölçekli En",
|
||||
"seamlessXAxis": "Dikişsiz Döşeme X Ekseni",
|
||||
"downloadImage": "Görseli İndir",
|
||||
"type": "Tür"
|
||||
},
|
||||
"modelManager": {
|
||||
@@ -399,11 +382,9 @@
|
||||
"defaultVAE": "Varsayılan VAE",
|
||||
"lora": "LoRA",
|
||||
"noModelsAvailable": "Model yok",
|
||||
"noMatchingLoRAs": "Uygun LoRA Yok",
|
||||
"noMatchingModels": "Uygun Model Yok",
|
||||
"loading": "yükleniyor",
|
||||
"selectModel": "Model Seçin",
|
||||
"noLoRAsInstalled": "LoRA Yok"
|
||||
"selectModel": "Model Seçin"
|
||||
},
|
||||
"settings": {
|
||||
"generation": "Oluşturma"
|
||||
@@ -411,7 +392,6 @@
|
||||
"sdxl": {
|
||||
"cfgScale": "CFG Ölçeği",
|
||||
"loading": "Yükleniyor...",
|
||||
"denoisingStrength": "Arındırma Ölçüsü",
|
||||
"concatPromptStyle": "İstem ve Stili Bitiştir"
|
||||
"denoisingStrength": "Arındırma Ölçüsü"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -22,8 +22,7 @@
|
||||
"gallery": {
|
||||
"galleryImageSize": "Розмір зображень",
|
||||
"gallerySettings": "Налаштування галереї",
|
||||
"autoSwitchNewImages": "Автоматично вибирати нові",
|
||||
"noImagesInGallery": "Зображень немає"
|
||||
"autoSwitchNewImages": "Автоматично вибирати нові"
|
||||
},
|
||||
"modelManager": {
|
||||
"modelManager": "Менеджер моделей",
|
||||
@@ -80,12 +79,10 @@
|
||||
"scaledHeight": "Масштаб В",
|
||||
"infillMethod": "Засіб заповнення",
|
||||
"tileSize": "Розмір області",
|
||||
"downloadImage": "Завантажити",
|
||||
"usePrompt": "Використати запит",
|
||||
"useSeed": "Використати сід",
|
||||
"useAll": "Використати все",
|
||||
"info": "Метадані",
|
||||
"showOptionsPanel": "Показати панель налаштувань",
|
||||
"general": "Основне",
|
||||
"denoisingStrength": "Сила шумоподавлення",
|
||||
"copyImage": "Копіювати зображення",
|
||||
|
||||
@@ -20,8 +20,6 @@
|
||||
"addBoard": "Thêm Bảng",
|
||||
"downloadBoard": "Tải Xuống Bảng",
|
||||
"movingImagesToBoard_other": "Di chuyển {{count}} ảnh vào Bảng:",
|
||||
"viewBoards": "Xem Bảng",
|
||||
"hideBoards": "Ẩn Bảng",
|
||||
"noBoards": "Không Có Bảng Thuộc Loại {{boardType}}",
|
||||
"noMatching": "Không Có Bảng Tương Ứng",
|
||||
"searchBoard": "Tìm Bảng...",
|
||||
@@ -55,8 +53,12 @@
|
||||
"assetsWithCount_other": "{{count}} tài nguyên",
|
||||
"uncategorizedImages": "Ảnh Chưa Sắp Xếp",
|
||||
"deleteAllUncategorizedImages": "Xoá Tất Cả Ảnh Chưa Sắp Xếp",
|
||||
"deletedImagesCannotBeRestored": "Ảnh đã xoá không thể phục hồi lại.",
|
||||
"locateInGalery": "Vị Trí Ở Thư Viện Ảnh"
|
||||
"locateInGalery": "Vị Trí Ở Thư Viện Ảnh",
|
||||
"deletedImagesCannotBeRestored": "Ảnh đã xóa không thể khôi phục lại.",
|
||||
"hideBoards": "Ẩn Bảng",
|
||||
"movingVideosToBoard_other": "Di chuyển {{count}} video vào bảng:",
|
||||
"viewBoards": "Xem Bảng",
|
||||
"videosWithCount_other": "{{count}} video"
|
||||
},
|
||||
"gallery": {
|
||||
"swapImages": "Đổi Hình Ảnh",
|
||||
@@ -84,33 +86,27 @@
|
||||
"galleryImageSize": "Kích Thước Ảnh",
|
||||
"downloadSelection": "Tải xuống Phần Được Lựa Chọn",
|
||||
"bulkDownloadRequested": "Chuẩn Bị Tải Xuống",
|
||||
"unableToLoad": "Không Thể Tải Thư viện Ảnh",
|
||||
"newestFirst": "Mới Nhất Trước",
|
||||
"showStarredImagesFirst": "Hiển Thị Ảnh Gắn Sao Trước",
|
||||
"bulkDownloadRequestedDesc": "Yêu cầu tải xuống đang được chuẩn bị. Vui lòng chờ trong giây lát.",
|
||||
"starImage": "Gắn Sao Cho Ảnh",
|
||||
"openViewer": "Mở Trình Xem",
|
||||
"starImage": "Gắn Sao",
|
||||
"viewerImage": "Trình Xem Ảnh",
|
||||
"sideBySide": "Cạnh Nhau",
|
||||
"alwaysShowImageSizeBadge": "Luôn Hiển Thị Kích Thước Ảnh",
|
||||
"autoAssignBoardOnClick": "Tự Động Gán Vào Bảng Khi Nhấp Chuột",
|
||||
"jump": "Nhảy Đến",
|
||||
"go": "Đi",
|
||||
"autoSwitchNewImages": "Tự Động Đổi Sang Hình Ảnh Mới",
|
||||
"featuresWillReset": "Nếu bạn xoá hình ảnh này, những tính năng đó sẽ lập tức được khởi động lại.",
|
||||
"openInViewer": "Mở Trong Trình Xem",
|
||||
"searchImages": "Tìm Theo Metadata",
|
||||
"selectForCompare": "Chọn Để So Sánh",
|
||||
"closeViewer": "Đóng Trình Xem",
|
||||
"move": "Di Chuyển",
|
||||
"displayBoardSearch": "Tìm Kiếm Bảng",
|
||||
"displaySearch": "Tìm Kiếm Hình Ảnh",
|
||||
"selectAnImageToCompare": "Chọn Ảnh Để So Sánh",
|
||||
"slider": "Thanh Trượt",
|
||||
"gallerySettings": "Cài Đặt Thư Viện Ảnh",
|
||||
"image": "hình ảnh",
|
||||
"noImageSelected": "Không Có Ảnh Được Chọn",
|
||||
"noImagesInGallery": "Không Có Ảnh Để Hiển Thị",
|
||||
"assetsTab": "Tài liệu bạn đã tải lên để dùng cho dự án của mình.",
|
||||
"imagesTab": "Ảnh bạn vừa được tạo và lưu trong Invoke.",
|
||||
"loading": "Đang Tải",
|
||||
@@ -118,13 +114,24 @@
|
||||
"exitCompare": "Ngừng So Sánh",
|
||||
"stretchToFit": "Kéo Dài Cho Vừa Vặn",
|
||||
"sortDirection": "Cách Sắp Xếp",
|
||||
"unstarImage": "Ngừng Gắn Sao Cho Ảnh",
|
||||
"unstarImage": "Bỏ Gắn Sao",
|
||||
"compareHelp2": "Nhấn <Kbd>M</Kbd> để tuần hoàn trong chế độ so sánh.",
|
||||
"boardsSettings": "Thiết Lập Bảng",
|
||||
"imagesSettings": "Cài Đặt Ảnh Trong Thư Viện Ảnh",
|
||||
"assets": "Tài Nguyên",
|
||||
"images": "Hình Ảnh",
|
||||
"useForPromptGeneration": "Dùng Để Tạo Sinh Lệnh"
|
||||
"useForPromptGeneration": "Dùng Để Tạo Sinh Lệnh",
|
||||
"deleteVideo_other": "Xóa {{count}} Video",
|
||||
"deleteVideoPermanent": "Video đã xóa không thể khôi phục lại.",
|
||||
"jump": "Nhảy Đến",
|
||||
"noVideoSelected": "Không Có Video Được Chọn",
|
||||
"noImagesInGallery": "Không Có Ảnh Để Hiển Thị",
|
||||
"unableToLoad": "Không Thể Tải Thư Viện Ảnh",
|
||||
"selectAnImageToCompare": "Chọn Ảnh Để So Sánh",
|
||||
"openViewer": "Mở Trình Xem",
|
||||
"closeViewer": "Đóng Trình Xem",
|
||||
"videos": "Video",
|
||||
"videosTab": "Video bạn tạo và được lưu trong Invoke."
|
||||
},
|
||||
"common": {
|
||||
"ipAdapter": "IP Adapter",
|
||||
@@ -135,14 +142,12 @@
|
||||
"clipboard": "Clipboard",
|
||||
"learnMore": "Tìm Hiểu Thêm",
|
||||
"openInViewer": "Mở Trong Trình Xem",
|
||||
"nextPage": "Trang Sau",
|
||||
"alpha": "Alpha",
|
||||
"edit": "Sửa",
|
||||
"nodes": "Workflow",
|
||||
"format": "Định Dạng",
|
||||
"delete": "Xoá",
|
||||
"details": "Chi Tiết",
|
||||
"imageFailedToLoad": "Không Thể Tải Hình Ảnh",
|
||||
"img2img": "Hình ảnh sang Hình ảnh",
|
||||
"upload": "Tải Lên",
|
||||
"somethingWentWrong": "Có vấn đề phát sinh",
|
||||
@@ -158,7 +163,7 @@
|
||||
"dontAskMeAgain": "Không hỏi lại",
|
||||
"error": "Lỗi",
|
||||
"or": "hoặc",
|
||||
"installed": "Đã Tải Xuống",
|
||||
"installed": "Được Tải Xuống Sẵn",
|
||||
"simple": "Cơ Bản",
|
||||
"linear": "Tuyến Tính",
|
||||
"safetensors": "Safetensors",
|
||||
@@ -180,19 +185,15 @@
|
||||
"on": "Bật",
|
||||
"checkpoint": "Checkpoint",
|
||||
"txt2img": "Từ Ngữ Sang Hình Ảnh",
|
||||
"prevPage": "Trang Trước",
|
||||
"unknown": "Không Rõ",
|
||||
"githubLabel": "Github",
|
||||
"folder": "Thư mục",
|
||||
"goTo": "Đến",
|
||||
"hotkeysLabel": "Phím Tắt",
|
||||
"loadingImage": "Đang Tải Hình ảnh",
|
||||
"localSystem": "Hệ Thống Máy Chủ",
|
||||
"input": "Đầu Vào",
|
||||
"languagePickerLabel": "Ngôn Ngữ",
|
||||
"openInNewTab": "Mở Trong Tab Mới",
|
||||
"outpaint": "outpaint",
|
||||
"notInstalled": "Chưa $t(common.installed)",
|
||||
"save": "Lưu",
|
||||
"saveAs": "Lưu Như",
|
||||
"auto": "Tự Động",
|
||||
@@ -234,7 +235,6 @@
|
||||
"end": "Kết Thúc",
|
||||
"min": "Tối Thiểu",
|
||||
"max": "Tối Đa",
|
||||
"resetToDefaults": "Đặt Lại Về Mặc Định",
|
||||
"seed": "Hạt Giống",
|
||||
"combinatorial": "Tổ Hợp",
|
||||
"column": "Cột",
|
||||
@@ -256,7 +256,14 @@
|
||||
"options_withCount_other": "{{count}} thiết lập",
|
||||
"removeNegativePrompt": "Xóa Lệnh Tiêu Cực",
|
||||
"addNegativePrompt": "Thêm Lệnh Tiêu Cực",
|
||||
"selectYourModel": "Chọn Model"
|
||||
"selectYourModel": "Chọn Model",
|
||||
"goTo": "Đi Đến",
|
||||
"imageFailedToLoad": "Không Thể Tải Ảnh",
|
||||
"localSystem": "Hệ Thống Máy Chủ",
|
||||
"notInstalled": "Chưa $t(common.installed)",
|
||||
"prevPage": "Trang Trước",
|
||||
"nextPage": "Trang Sau",
|
||||
"resetToDefaults": "Tải Lại Mặc Định"
|
||||
},
|
||||
"prompt": {
|
||||
"addPromptTrigger": "Thêm Trigger Cho Lệnh",
|
||||
@@ -266,11 +273,11 @@
|
||||
"expandCurrentPrompt": "Mở Rộng Lệnh Hiện Tại",
|
||||
"uploadImageForPromptGeneration": "Tải Ảnh Để Tạo Sinh Lệnh",
|
||||
"expandingPrompt": "Đang mở rộng lệnh...",
|
||||
"replace": "Thay Thế",
|
||||
"discard": "Huỷ Bỏ",
|
||||
"resultTitle": "Mở Rộng Lệnh Hoàn Tất",
|
||||
"resultSubtitle": "Chọn phương thức mở rộng lệnh:",
|
||||
"replace": "Thay Thế",
|
||||
"insert": "Chèn",
|
||||
"discard": "Huỷ Bỏ"
|
||||
"insert": "Chèn"
|
||||
},
|
||||
"queue": {
|
||||
"resume": "Tiếp Tục",
|
||||
@@ -284,7 +291,6 @@
|
||||
"clearQueueAlertDialog2": "Bạn chắc chắn muốn dọn sạch hàng không?",
|
||||
"queueEmpty": "Hàng Trống",
|
||||
"queueBack": "Thêm Vào Hàng",
|
||||
"batchFieldValues": "Giá Trị Vùng Theo Lô",
|
||||
"openQueue": "Mở Queue",
|
||||
"pause": "Dừng Lại",
|
||||
"pauseFailed": "Có Vấn Đề Khi Dừng Lại Bộ Xử Lý",
|
||||
@@ -348,7 +354,13 @@
|
||||
"retryFailed": "Có Vấn Đề Khi Thử Lại Mục",
|
||||
"retryItem": "Thử Lại Mục",
|
||||
"credits": "Nguồn",
|
||||
"cancelAllExceptCurrent": "Huỷ Bỏ Tất Cả Ngoại Trừ Mục Hiện Tại"
|
||||
"cancelAllExceptCurrent": "Huỷ Bỏ Tất Cả Ngoại Trừ Mục Hiện Tại",
|
||||
"createdAt": "Tạo tại",
|
||||
"completedAt": "Hoàn Thành Tại",
|
||||
"sortColumn": "Sắp Xếp Cột",
|
||||
"sortBy": "Sắp Xếp Theo {{column}}",
|
||||
"sortOrderAscending": "Tăng Dần",
|
||||
"sortOrderDescending": "Giảm Dần"
|
||||
},
|
||||
"hotkeys": {
|
||||
"canvas": {
|
||||
@@ -500,6 +512,14 @@
|
||||
"toggleBbox": {
|
||||
"title": "Bật/Tắt Hiển Thị Hộp Giới Hạn",
|
||||
"desc": "Ẩn hoặc hiện hộp giới hạn tạo sinh"
|
||||
},
|
||||
"setFillColorsToDefault": {
|
||||
"title": "Đặt Màu Lại Mặc Định",
|
||||
"desc": "Chỉnh công cụ màu hiện tại về mặc định."
|
||||
},
|
||||
"toggleFillColor": {
|
||||
"title": "Bật/Tắt Màu Lấp Đầy",
|
||||
"desc": "Bật/Tắt công cụ đổ màu hiện tại."
|
||||
}
|
||||
},
|
||||
"workflows": {
|
||||
@@ -697,12 +717,19 @@
|
||||
"title": "Chọn Tab Tạo Sinh",
|
||||
"desc": "Chọn tab Tạo Sinh.",
|
||||
"key": "1"
|
||||
},
|
||||
"selectVideoTab": {
|
||||
"title": "Chọn Thẻ Video",
|
||||
"desc": "Chọn thẻ Video."
|
||||
}
|
||||
},
|
||||
"searchHotkeys": "Tìm Phím tắt",
|
||||
"noHotkeysFound": "Không Tìm Thấy Phím Tắt",
|
||||
"clearSearch": "Làm Sạch Thanh Tìm Kiếm",
|
||||
"hotkeys": "Phím Tắt"
|
||||
"hotkeys": "Phím Tắt",
|
||||
"video": {
|
||||
"title": "Video"
|
||||
}
|
||||
},
|
||||
"modelManager": {
|
||||
"modelConverted": "Model Đã Được Chuyển Đổi",
|
||||
@@ -786,7 +813,6 @@
|
||||
"hfTokenUnableToVerifyErrorMessage": "Không thể xác minh HuggingFace token. Khả năng cao lỗi mạng. Vui lòng thử lại sau.",
|
||||
"inplaceInstall": "Tải Xuống Tại Chỗ",
|
||||
"installRepo": "Tải Xuống Kho Lưu Trữ (Repository)",
|
||||
"ipAdapters": "IP Adapters",
|
||||
"loraModels": "LoRA",
|
||||
"main": "Chính",
|
||||
"modelConversionFailed": "Chuyển Đổi Model Thất Bại",
|
||||
@@ -832,7 +858,6 @@
|
||||
"textualInversions": "Bộ Đảo Ngược Văn Bản",
|
||||
"loraTriggerPhrases": "Từ Ngữ Kích Hoạt Cho LoRA",
|
||||
"width": "Chiều Rộng",
|
||||
"starterModelsInModelManager": "Model khởi đầu có thể tìm thấy ở Trình Quản Lý Model",
|
||||
"clipLEmbed": "CLIP-L Embed",
|
||||
"clipGEmbed": "CLIP-G Embed",
|
||||
"controlLora": "LoRA Điều Khiển Được",
|
||||
@@ -844,13 +869,11 @@
|
||||
"sigLip": "SigLIP",
|
||||
"llavaOnevision": "LLaVA OneVision",
|
||||
"fileSize": "Kích Thước Tệp",
|
||||
"filterModels": "Lọc Model",
|
||||
"modelPickerFallbackNoModelsInstalled2": "Nhấp vào <LinkComponent>Trình Quản Lý Model</LinkComponent> để tải.",
|
||||
"modelPickerFallbackNoModelsInstalled": "Không Có Sẵn Model.",
|
||||
"manageModels": "Quản Lý Model",
|
||||
"hfTokenReset": "Làm Mới HF Token",
|
||||
"relatedModels": "Model Liên Quan",
|
||||
"showOnlyRelatedModels": "Liên Quan",
|
||||
"installedModelsCount": "Đã tải {{installed}} trên {{total}} model.",
|
||||
"allNModelsInstalled": "Đã tải tất cả {{count}} model",
|
||||
"nToInstall": "Còn {{count}} để tải",
|
||||
@@ -867,30 +890,32 @@
|
||||
"scanFolderDescription": "Quét một thư mục trên máy để tự động tra và tải model.",
|
||||
"recommendedModels": "Model Khuyến Nghị",
|
||||
"exploreStarter": "Hoặc duyệt tất cả model khởi đầu có sẵn",
|
||||
"quickStart": "Gói Khởi Đầu Nhanh",
|
||||
"bundleDescription": "Các gói đều bao gồm những model cần thiết cho từng nhánh model và những model cơ sở đã chọn lọc để bắt đầu.",
|
||||
"sdxl": "SDXL",
|
||||
"quickStart": "Gói Khởi Đầu Nhanh",
|
||||
"browseAll": "Hoặc duyệt tất cả model có sẵn:",
|
||||
"stableDiffusion15": "Stable Diffusion 1.5",
|
||||
"sdxl": "SDXL",
|
||||
"fluxDev": "FLUX.1 dev"
|
||||
},
|
||||
"installBundle": "Tải Xuống Gói",
|
||||
"installBundleMsg1": "Bạn có chắc chắn muốn tải xuống gói {{bundleName}}?",
|
||||
"installBundleMsg2": "Gói này sẽ tải xuống {{count}} model sau đây:"
|
||||
"installBundleMsg2": "Gói này sẽ tải xuống {{count}} model sau đây:",
|
||||
"filterModels": "Lọc Model",
|
||||
"ipAdapters": "IP Adapters",
|
||||
"showOnlyRelatedModels": "Liên Quan",
|
||||
"starterModelsInModelManager": "Model Khởi Đầu có thể tìm thấy ở Trình Quản Lý Model"
|
||||
},
|
||||
"metadata": {
|
||||
"guidance": "Hướng Dẫn",
|
||||
"noRecallParameters": "Không tìm thấy tham số",
|
||||
"imageDetails": "Chi Tiết Ảnh",
|
||||
"createdBy": "Được Tạo Bởi",
|
||||
"parsingFailed": "Lỗi Cú Pháp",
|
||||
"canvasV2Metadata": "Layer Canvas",
|
||||
"parameterSet": "Dữ liệu tham số {{parameter}}",
|
||||
"positivePrompt": "Lệnh Tích Cực",
|
||||
"recallParameter": "Gợi Nhớ {{label}}",
|
||||
"seed": "Hạt Giống",
|
||||
"negativePrompt": "Lệnh Tiêu Cực",
|
||||
"noImageDetails": "Không tìm thấy chí tiết ảnh",
|
||||
"noImageDetails": "Không tìm thấy chi tiết ảnh",
|
||||
"strength": "Mức độ mạnh từ ảnh sang ảnh",
|
||||
"Threshold": "Ngưỡng Nhiễu",
|
||||
"width": "Chiều Rộng",
|
||||
@@ -910,7 +935,15 @@
|
||||
"scheduler": "Scheduler",
|
||||
"noMetaData": "Không tìm thấy metadata",
|
||||
"imageDimensions": "Kích Thước Ảnh",
|
||||
"clipSkip": "$t(parameters.clipSkip)"
|
||||
"clipSkip": "$t(parameters.clipSkip)",
|
||||
"videoDetails": "Chi Tiết Video",
|
||||
"noVideoDetails": "Không tìm thấy chi tiết video",
|
||||
"parsingFailed": "Lỗi Cú Pháp",
|
||||
"recallParameter": "Gợi Nhớ {{label}}",
|
||||
"videoModel": "Model",
|
||||
"videoDuration": "Thời Lượng",
|
||||
"videoAspectRatio": "Tỉ Lệ",
|
||||
"videoResolution": "Độ Phân Giải"
|
||||
},
|
||||
"accordions": {
|
||||
"generation": {
|
||||
@@ -956,8 +989,8 @@
|
||||
"method": "Cách Thức Sửa Độ Phân Giải Cao"
|
||||
},
|
||||
"hrf": "Sửa Độ Phân Giải Cao",
|
||||
"enableHrf": "Cho Phép Sửa Độ Phân Giải Cao",
|
||||
"upscaleMethod": "Cách Thức Upscale"
|
||||
"enableHrf": "Bật Chế Độ Chỉnh Sửa Phân Giải Cao",
|
||||
"upscaleMethod": "Phương Thức Upscale"
|
||||
},
|
||||
"nodes": {
|
||||
"validateConnectionsHelp": "Ngăn chặn những kết nối không hợp lý được tạo ra, và đồ thị không hợp lệ bị kích hoạt",
|
||||
@@ -983,9 +1016,7 @@
|
||||
"float": "Số Thực",
|
||||
"missingNode": "Thiếu node kích hoạt",
|
||||
"currentImage": "Hình Ảnh Hiện Tại",
|
||||
"removeLinearView": "Xoá Khỏi Chế Độ Xem Tuyến Tính",
|
||||
"unknownErrorValidatingWorkflow": "Lỗi không rõ khi xác thực workflow",
|
||||
"unableToLoadWorkflow": "Không Thể Tải Workflow",
|
||||
"workflowSettings": "Cài Đặt Biên Tập Workflow",
|
||||
"workflowVersion": "Phiên Bản",
|
||||
"unableToGetWorkflowVersion": "Không thể tìm phiên bản của lược đồ workflow",
|
||||
@@ -995,7 +1026,6 @@
|
||||
"ipAdapter": "IP Adapter",
|
||||
"cannotDuplicateConnection": "Không thể tạo hai kết nối trùng lặp",
|
||||
"workflowValidation": "Lỗi Xác Thực Workflow",
|
||||
"mismatchedVersion": "Node không hợp lệ: node {{node}} thuộc loại {{type}} có phiên bản không khớp (thử cập nhật?)",
|
||||
"sourceNodeFieldDoesNotExist": "Kết nối không phù hợp: nguồn/đầu ra của vùng {{node}}.{{field}} không tồn tại",
|
||||
"targetNodeFieldDoesNotExist": "Kết nối không phù hợp: đích đến/đầu vào của vùng {{node}}.{{field}} không tồn tại",
|
||||
"missingTemplate": "Node không hợp lệ: node {{node}} thuộc loại {{type}} bị thiếu mẫu trình bày (chưa tải?)",
|
||||
@@ -1009,7 +1039,6 @@
|
||||
"edge": "Kết Nối",
|
||||
"graph": "Đồ Thị",
|
||||
"workflowAuthor": "Tác Giả",
|
||||
"addLinearView": "Thêm Vào Chế Độ Xem Tuyến Tính",
|
||||
"showEdgeLabels": "Hiển Thị Tên Kết Nối",
|
||||
"unknownField": "Vùng Dữ Liệu Không Rõ",
|
||||
"executionStateCompleted": "Đã Hoàn Tất",
|
||||
@@ -1039,7 +1068,6 @@
|
||||
"node": "Node",
|
||||
"nodeTemplate": "Mẫu Trình Bày Của Node",
|
||||
"nodeType": "Loại Node",
|
||||
"noFieldsLinearview": "Không có vùng được thêm vào Chế Độ Xem Tuyến Tính",
|
||||
"notes": "Ghi Chú",
|
||||
"updateApp": "Cập Nhật Ứng Dụng",
|
||||
"updateAllNodes": "Cập Nhật Các Node",
|
||||
@@ -1047,7 +1075,6 @@
|
||||
"imageAccessError": "Không thể tìm thấy ảnh {{image_name}}, chuyển về mặc định",
|
||||
"unknownNode": "Node Không Rõ",
|
||||
"unknownNodeType": "Loại Node Không Rõ",
|
||||
"unknownTemplate": "Mẫu Trình Bày Không Rõ",
|
||||
"cannotConnectOutputToOutput": "Không thế kết nối đầu ra với đầu ra",
|
||||
"cannotConnectToSelf": "Không thể kết nối với chính nó",
|
||||
"workflow": "Workflow",
|
||||
@@ -1063,7 +1090,6 @@
|
||||
"fitViewportNodes": "Chế Độ Xem Vừa Khớp",
|
||||
"fullyContainNodes": "Bao Phủ Node Hoàn Toàn Để Chọn",
|
||||
"fullyContainNodesHelp": "Node phải được phủ kín hoàn toàn trong hộp lựa chọn để được lựa chọn",
|
||||
"hideLegendNodes": "Ẩn Vùng Nhập",
|
||||
"hideMinimapnodes": "Ẩn Bản Đồ Thu Nhỏ",
|
||||
"inputMayOnlyHaveOneConnection": "Đầu vào chỉ có thể có một kết nối",
|
||||
"noWorkflows": "Không Có Workflow",
|
||||
@@ -1074,34 +1100,27 @@
|
||||
"problemSettingTitle": "Có Vấn Đề Khi Thiết Lập Tiêu Đề",
|
||||
"resetToDefaultValue": "Đặt lại giá trị mặc định",
|
||||
"reloadNodeTemplates": "Tải Lại Mẫu Trình Bày Node",
|
||||
"reorderLinearView": "Sắp Xếp Lại Chế Độ Xem Tuyến Tính",
|
||||
"viewMode": "Dùng Chế Độ Xem Tuyến Tính",
|
||||
"newWorkflowDesc": "Tạo workflow mới?",
|
||||
"string": "Chuỗi Ký Tự",
|
||||
"version": "Phiên Bản",
|
||||
"versionUnknown": " Phiên Bản Không Rõ",
|
||||
"workflowContact": "Thông Tin Liên Lạc",
|
||||
"workflowName": "Tên",
|
||||
"saveToGallery": "Lưu Vào Thư Viện Ảnh",
|
||||
"connectionWouldCreateCycle": "Kết nối này sẽ tạo ra vòng lặp",
|
||||
"addNode": "Thêm Node",
|
||||
"unsupportedAnyOfLength": "quá nhiều dữ liệu hợp nhất: {{count}}",
|
||||
"unknownInput": "Đầu Vào Không Rõ: {{name}}",
|
||||
"validateConnections": "Xác Thực Kết Nối Và Đồ Thị",
|
||||
"workflowNotes": "Ghi Chú",
|
||||
"workflowTags": "Nhãn",
|
||||
"editMode": "Chỉnh sửa trong Trình Biên Tập Workflow",
|
||||
"edit": "Chỉnh Sửa",
|
||||
"executionStateInProgress": "Đang Xử Lý",
|
||||
"showLegendNodes": "Hiển Thị Vùng Nhập",
|
||||
"outputFieldTypeParseError": "Không thể phân tích loại dữ liệu đầu ra của {{node}}.{{field}} ({{message}})",
|
||||
"modelAccessError": "Không thể tìm thấy model {{key}}, chuyển về mặc định",
|
||||
"internalDesc": "Trình kích hoạt này được dùng bên trong bởi Invoke. Nó có thể phá hỏng thay đổi trong khi cập nhật ứng dụng và có thể bị xoá bất cứ lúc nào.",
|
||||
"specialDesc": "Trình kích hoạt này có một số xử lý đặc biệt trong ứng dụng. Ví dụ, Node Hàng Loạt được dùng để xếp vào nhiều đồ thị từ một workflow.",
|
||||
"addItem": "Thêm Mục",
|
||||
"generateValues": "Cho Ra Giá Trị",
|
||||
"floatRangeGenerator": "Phạm Vị Tạo Ra Số Thực",
|
||||
"integerRangeGenerator": "Phạm Vị Tạo Ra Số Nguyên",
|
||||
"linearDistribution": "Phân Bố Tuyến Tính",
|
||||
"uniformRandomDistribution": "Phân Bố Ngẫu Nhiên Đồng Nhất",
|
||||
"parseString": "Phân Tích Chuỗi",
|
||||
@@ -1110,7 +1129,6 @@
|
||||
"splitOn": "Tách Ở",
|
||||
"arithmeticSequence": "Cấp Số Cộng",
|
||||
"generatorNRandomValues_other": "{{count}} giá trị ngẫu nhiên",
|
||||
"generatorLoading": "đang tải",
|
||||
"generatorLoadFromFile": "Tải Từ Tệp",
|
||||
"dynamicPromptsRandom": "Dynamic Prompts (Ngẫu Nhiên)",
|
||||
"dynamicPromptsCombinatorial": "Dynamic Prompts (Tổ Hợp)",
|
||||
@@ -1120,7 +1138,6 @@
|
||||
"description": "Mô Tả",
|
||||
"loadWorkflowDesc": "Tải workflow?",
|
||||
"loadWorkflowDesc2": "Workflow hiện tại của bạn có những điều chỉnh chưa được lưu.",
|
||||
"loadingTemplates": "Đang Tải {{name}}",
|
||||
"nodeName": "Tên Node",
|
||||
"unableToUpdateNode": "Cập nhật node thất bại: node {{node}} thuộc dạng {{type}} (có thể cần xóa và tạo lại)",
|
||||
"downloadWorkflowError": "Lỗi tải xuống workflow",
|
||||
@@ -1146,7 +1163,23 @@
|
||||
"alignmentDL": "Dưới Cùng Bên Trái",
|
||||
"alignmentUR": "Trên Cùng Bên Phải",
|
||||
"alignmentDR": "Dưới Cùng Bên Phải"
|
||||
}
|
||||
},
|
||||
"generatorLoading": "đang tải",
|
||||
"addLinearView": "Thêm Vào Chế Độ Xem Tuyến Tính (Linear View)",
|
||||
"hideLegendNodes": "Ẩn Vùng Nhập",
|
||||
"mismatchedVersion": "Node không hợp lệ: node {{node}} thuộc loại {{type}} có phiên bản không khớp (thử cập nhật?)",
|
||||
"noFieldsLinearview": "Không có vùng được thêm vào Chế Độ Xem Tuyến Tính",
|
||||
"removeLinearView": "Xoá Khỏi Chế Độ Xem Tuyến Tính",
|
||||
"reorderLinearView": "Sắp Xếp Lại Chế Độ Xem Tuyến Tính",
|
||||
"showLegendNodes": "Hiển Thị Vùng Nhập",
|
||||
"unableToLoadWorkflow": "Không Thể Tải Workflow",
|
||||
"unknownTemplate": "Mẫu Trình Bày Không Rõ",
|
||||
"unknownInput": "Đầu Vào Không Rõ: {{name}}",
|
||||
"loadingTemplates": "Đang Tải {{name}}",
|
||||
"versionUnknown": " Phiên Bản Không Rõ",
|
||||
"generateValues": "Giá Trị Tạo Sinh",
|
||||
"floatRangeGenerator": "Phạm Vị Tạo Sinh Số Thực",
|
||||
"integerRangeGenerator": "Phạm Vị Tạo Sinh Số Nguyên"
|
||||
},
|
||||
"popovers": {
|
||||
"paramCFGRescaleMultiplier": {
|
||||
@@ -1594,14 +1627,14 @@
|
||||
"concepts": "LoRA",
|
||||
"loading": "đang tải",
|
||||
"lora": "LoRA",
|
||||
"noMatchingLoRAs": "Không có LoRA phù hợp",
|
||||
"noRefinerModelsInstalled": "Chưa có model SDXL Refiner được tải xuống",
|
||||
"noLoRAsInstalled": "Chưa có LoRA được tải xuống",
|
||||
"defaultVAE": "VAE Mặc Định",
|
||||
"noMatchingModels": "Không có Model phù hợp",
|
||||
"noModelsAvailable": "Không có model",
|
||||
"selectModel": "Chọn Model",
|
||||
"noCompatibleLoRAs": "Không Có LoRAs Tương Thích"
|
||||
"noCompatibleLoRAs": "Không Có LoRAs Tương Thích",
|
||||
"noMatchingLoRAs": "Không có LoRA phù hợp",
|
||||
"noLoRAsInstalled": "Chưa có LoRA được tải xuống"
|
||||
},
|
||||
"parameters": {
|
||||
"postProcessing": "Xử Lý Hậu Kỳ (Shift + U)",
|
||||
@@ -1611,9 +1644,7 @@
|
||||
"processImage": "Xử Lý Hình Ảnh",
|
||||
"useSize": "Dùng Kích Thước",
|
||||
"invoke": {
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), chiều rộng hộp giới hạn là {{width}}",
|
||||
"noModelSelected": "Không có model được lựa chọn",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), tỉ lệ chiều dài hộp giới hạn là {{height}}",
|
||||
"canvasIsFiltering": "Canvas đang bận (đang lọc)",
|
||||
"canvasIsRasterizing": "Canvas đang bận (đang raster hoá)",
|
||||
"canvasIsTransforming": "Canvas đang bận (đang biến đổi)",
|
||||
@@ -1627,8 +1658,6 @@
|
||||
"systemDisconnected": "Hệ thống mất kết nối",
|
||||
"invoke": "Kích Hoạt",
|
||||
"missingNodeTemplate": "Thiếu mẫu trình bày node",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), chiều dài hộp giới hạn là {{height}}",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), tỉ lệ chiều rộng hộp giới hạn là {{width}}",
|
||||
"missingInputForField": "thiếu đầu vào",
|
||||
"missingFieldTemplate": "Thiếu vùng mẫu trình bày",
|
||||
"collectionTooFewItems": "quá ít mục, tối thiểu là {{minItems}}",
|
||||
@@ -1643,7 +1672,6 @@
|
||||
"collectionNumberLTExclusiveMin": "{{value}} <= {{exclusiveMinimum}} (giá trị chọn lọc tối thiểu)",
|
||||
"collectionNumberGTExclusiveMax": "{{value}} >= {{exclusiveMaximum}} (giá trị chọn lọc tối đa)",
|
||||
"batchNodeCollectionSizeMismatch": "Kích cỡ tài nguyên không phù hợp với Lô {{batchGroupId}}",
|
||||
"emptyBatches": "lô trống",
|
||||
"batchNodeNotConnected": "Node Hàng Loạt chưa được kết nối: {{label}}",
|
||||
"batchNodeEmptyCollection": "Một vài node hàng loạt có tài nguyên rỗng",
|
||||
"collectionEmpty": "tài nguyên trống",
|
||||
@@ -1654,7 +1682,15 @@
|
||||
"modelIncompatibleScaledBboxWidth": "Chiều rộng hộp giới hạn theo tỉ lệ là {{width}} nhưng {{model}} yêu cầu bội số của {{multiple}}",
|
||||
"modelDisabledForTrial": "Tạo sinh với {{modelName}} là không thể với tài khoản trial. Vào phần thiết lập tài khoản để nâng cấp.",
|
||||
"promptExpansionPending": "Trong quá trình mở rộng lệnh",
|
||||
"promptExpansionResultPending": "Hãy chấp thuận hoặc huỷ bỏ kết quả mở rộng lệnh của bạn"
|
||||
"promptExpansionResultPending": "Hãy chấp thuận hoặc huỷ bỏ kết quả mở rộng lệnh của bạn",
|
||||
"emptyBatches": "lô trống",
|
||||
"noStartingFrameImage": "Chưa có khung hình ảnh đầu",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), chiều rộng hộp giới hạn là {{width}}",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), chiều cao hộp giới hạn là {{height}}",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), tỉ lệ chiều rộng hộp giới hạn là {{width}}",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16), tỉ lệ chiều cao hộp giới hạn là {{height}}",
|
||||
"incompatibleLoRAs": "LoRA không tương thích bị thêm vào",
|
||||
"videoIsDisabled": "Trình tạo sinh Video không được mở cho tài khoản {{accountType}}."
|
||||
},
|
||||
"cfgScale": "Thang CFG",
|
||||
"useSeed": "Dùng Hạt Giống",
|
||||
@@ -1701,7 +1737,6 @@
|
||||
"useAll": "Dùng Tất Cả",
|
||||
"useCpuNoise": "Dùng Độ Nhiễu CPU",
|
||||
"remixImage": "Phối Lại Hình Ảnh",
|
||||
"showOptionsPanel": "Hiển Thị Bảng Bên Cạnh (O hoặc T)",
|
||||
"shuffle": "Xáo Trộn",
|
||||
"setToOptimalSizeTooLarge": "$t(parameters.setToOptimalSize) (lớn quá)",
|
||||
"cfgRescaleMultiplier": "Hệ Số Nhân Thang CFG",
|
||||
@@ -1711,7 +1746,6 @@
|
||||
"lockAspectRatio": "Khoá Tỉ Lệ",
|
||||
"swapDimensions": "Hoán Đổi Kích Thước",
|
||||
"copyImage": "Sao Chép Hình Ảnh",
|
||||
"downloadImage": "Tải Xuống Hình Ảnh",
|
||||
"imageFit": "Căn Chỉnh Ảnh Ban Đầu Thành Kích Thước Đầu Ra",
|
||||
"info": "Thông Tin",
|
||||
"usePrompt": "Dùng Lệnh",
|
||||
@@ -1719,7 +1753,17 @@
|
||||
"tileSize": "Kích Thước Khối",
|
||||
"disabledNoRasterContent": "Đã Tắt (Không Có Nội Dung Dạng Raster)",
|
||||
"modelDisabledForTrial": "Tạo sinh với {{modelName}} là không thể với tài khoản trial. Vào phần <LinkComponent>thiết lập tài khoản</LinkComponent> để nâng cấp.",
|
||||
"useClipSkip": "Dùng CLIP Skip"
|
||||
"useClipSkip": "Dùng CLIP Skip",
|
||||
"duration": "Thời Lượng",
|
||||
"downloadImage": "Tải Xuống Hình Ảnh",
|
||||
"images_withCount_other": "Hình Ảnh",
|
||||
"videos_withCount_other": "Video",
|
||||
"startingFrameImage": "Khung Hình Bắt Đầu",
|
||||
"videoActions": "Hành Động Với Video",
|
||||
"sendToVideo": "Gửi Vào Video",
|
||||
"showOptionsPanel": "Hiển Thị Bảng Bên Cạnh (O hoặc T)",
|
||||
"video": "Video",
|
||||
"resolution": "Độ Phân Giải"
|
||||
},
|
||||
"dynamicPrompts": {
|
||||
"seedBehaviour": {
|
||||
@@ -1745,9 +1789,7 @@
|
||||
"antialiasProgressImages": "Xử Lý Khử Răng Cưa Hình Ảnh",
|
||||
"models": "Models",
|
||||
"informationalPopoversDisabledDesc": "Hộp thoại hỗ trợ thông tin đã tắt. Bật lại trong Cài đặt.",
|
||||
"modelDescriptionsDisabled": "Trình Mô Tả Model Bằng Hộp Thả Đã Tắt",
|
||||
"enableModelDescriptions": "Bật Trình Mô Tả Model Bằng Hộp Thả",
|
||||
"modelDescriptionsDisabledDesc": "Trình mô tả model bằng hộp thả đã tắt. Bật lại trong Cài đặt.",
|
||||
"enableNSFWChecker": "Bật Trình Kiểm Tra NSFW",
|
||||
"clearIntermediatesWithCount_other": "Dọn sạch {{count}} sản phẩm trung gian",
|
||||
"reloadingIn": "Tải lại trong",
|
||||
@@ -1770,7 +1812,9 @@
|
||||
"intermediatesClearedFailed": "Có Vấn Đề Khi Dọn Sạch Sản Phẩm Trung Gian",
|
||||
"enableInvisibleWatermark": "Bật Chế Độ Ẩn Watermark",
|
||||
"showDetailedInvocationProgress": "Hiện Dữ Liệu Xử Lý",
|
||||
"enableHighlightFocusedRegions": "Nhấn Mạnh Khu Vực Chỉ Định"
|
||||
"enableHighlightFocusedRegions": "Nhấn Mạnh Khu Vực Chỉ Định",
|
||||
"modelDescriptionsDisabled": "Trình Mô Tả Model Bằng Hộp Thả Đã Tắt",
|
||||
"modelDescriptionsDisabledDesc": "Trình mô tả model bằng hộp thả đã tắt. Bật lại trong Cài đặt."
|
||||
},
|
||||
"sdxl": {
|
||||
"loading": "Đang Tải...",
|
||||
@@ -1780,15 +1824,15 @@
|
||||
"refinermodel": "Model Refiner",
|
||||
"refinerStart": "Bắt Đầu Refiner",
|
||||
"denoisingStrength": "Sức Mạnh Khử Nhiễu",
|
||||
"posStylePrompt": "Điểm Tích Cực Cho Lệnh Phong Cách",
|
||||
"scheduler": "Scheduler",
|
||||
"refiner": "Refiner",
|
||||
"cfgScale": "Thang CFG",
|
||||
"concatPromptStyle": "Liên Kết Lệnh & Phong Cách",
|
||||
"freePromptStyle": "Viết Lệnh Thủ Công Cho Phong Cách",
|
||||
"negStylePrompt": "Điểm Tiêu Cực Cho Lệnh Phong Cách",
|
||||
"negAestheticScore": "Điểm Khác Tiêu Chuẩn",
|
||||
"noModelsAvailable": "Không có sẵn model"
|
||||
"noModelsAvailable": "Không có sẵn model",
|
||||
"concatPromptStyle": "Liên Kết Lệnh & Phong Cách",
|
||||
"freePromptStyle": "Viết Thủ Công Lệnh Phong Cách",
|
||||
"negStylePrompt": "Điểm Tiêu Cực Cho Lệnh Phong Cách",
|
||||
"posStylePrompt": "Điểm Tích Cực Cho Lệnh Phong Cách"
|
||||
},
|
||||
"controlLayers": {
|
||||
"width": "Chiều Rộng",
|
||||
@@ -1803,7 +1847,6 @@
|
||||
"saveLayerToAssets": "Lưu Layer Vào Khu Tài Nguyên",
|
||||
"canvas": "Canvas",
|
||||
"savedToGalleryOk": "Đã Lưu Vào Thư Viện Ảnh",
|
||||
"addGlobalReferenceImage": "Thêm $t(controlLayers.globalReferenceImage)",
|
||||
"clipToBbox": "Chuyển Nét Thành Hộp Giới Hạn",
|
||||
"moveToFront": "Chuyển Lên Trước",
|
||||
"mergeVisible": "Gộp Layer Đang Hiển Thị",
|
||||
@@ -1848,7 +1891,6 @@
|
||||
"stylePrecise": "Phong Cách (Chính Xác)",
|
||||
"stylePreciseDesc": "Áp dụng cách trình bày chính xác, loại bỏ các chủ thể ảnh hưởng."
|
||||
},
|
||||
"deletePrompt": "Xoá Lệnh",
|
||||
"rasterLayer": "Layer Dạng Raster",
|
||||
"disableAutoNegative": "Tắt Tự Động Đảo Chiều",
|
||||
"controlLayer": "Layer Điều Khiển Được",
|
||||
@@ -1859,8 +1901,6 @@
|
||||
"replaceLayer": "Thay Đổi Layer",
|
||||
"regionalGuidance": "Chỉ Dẫn Khu Vực",
|
||||
"newCanvasFromImage": "Canvas Mới Từ Ảnh",
|
||||
"rasterLayers_withCount_visible": "Layer Dạng Raster ({{count}})",
|
||||
"regionalGuidance_withCount_visible": "Chỉ Dẫn Khu Vực ({{count}})",
|
||||
"convertRasterLayerTo": "Chuyển Đổi $t(controlLayers.rasterLayer) Thành",
|
||||
"convertControlLayerTo": "Chuyển Đổi $t(controlLayers.controlLayer) Thành",
|
||||
"convertInpaintMaskTo": "Chuyển Đổi $t(controlLayers.inpaintMask) Thành",
|
||||
@@ -1871,12 +1911,7 @@
|
||||
"newRasterLayer": "$t(controlLayers.rasterLayer) Mới",
|
||||
"enableAutoNegative": "Bật Tự Động Đảo Chiều",
|
||||
"sendToCanvas": "Chuyển Tới Canvas",
|
||||
"inpaintMasks_withCount_hidden": "Lớp Phủ Inpaint ({{count}} đang ẩn)",
|
||||
"globalReferenceImages_withCount_visible": "Ảnh Mẫu Toàn Vùng ({{count}})",
|
||||
"replaceCurrent": "Thay Đổi Cái Hiện Tại",
|
||||
"controlLayers_withCount_visible": "Layer Điều Khiển Được ({{count}})",
|
||||
"hidingType": "Ẩn {{type}}",
|
||||
"newImg2ImgCanvasFromImage": "Chuyển Đổi Ảnh Sang Ảnh Mới Từ Ảnh",
|
||||
"copyToClipboard": "Sao Chép Vào Clipboard",
|
||||
"logDebugInfo": "Thông Tin Log Gỡ Lỗi",
|
||||
"regionalReferenceImage": "Ảnh Mẫu Khu Vực",
|
||||
@@ -1889,37 +1924,28 @@
|
||||
"horizontal": "Đường Ngang",
|
||||
"crosshatch": "Đường Chéo Song Song (Crosshatch)",
|
||||
"vertical": "Đường Dọc",
|
||||
"solid": "Chắc Chắn"
|
||||
"solid": "Chắc Chắn",
|
||||
"bgFillColor": "Màu Nền",
|
||||
"fgFillColor": "Màu Nổi"
|
||||
},
|
||||
"addControlLayer": "Thêm $t(controlLayers.controlLayer)",
|
||||
"inpaintMask": "Lớp Phủ Inpaint",
|
||||
"dynamicGrid": "Lưới Dynamic",
|
||||
"layer_other": "Layer",
|
||||
"layer_withCount_other": "Layer ({{count}})",
|
||||
"pullBboxIntoLayer": "Chuyển Hộp Giới Hạn Vào Layer",
|
||||
"addInpaintMask": "Thêm $t(controlLayers.inpaintMask)",
|
||||
"addRegionalGuidance": "Thêm $t(controlLayers.regionalGuidance)",
|
||||
"sendToGallery": "Đã Chuyển Tới Thư Viện Ảnh",
|
||||
"unlocked": "Mở Khoá",
|
||||
"addReferenceImage": "Thêm $t(controlLayers.referenceImage)",
|
||||
"sendingToCanvas": "Chuyển Ảnh Tạo Sinh Vào Canvas",
|
||||
"sendingToGallery": "Chuyển Ảnh Tạo Sinh Vào Thư Viện Ảnh",
|
||||
"viewProgressOnCanvas": "Xem quá trình xử lý và ảnh đầu ra trong <Btn>Canvas</Btn>.",
|
||||
"inpaintMask_withCount_other": "Lớp Phủ Inpaint",
|
||||
"regionalGuidance_withCount_other": "Chỉ Dẫn Khu Vực",
|
||||
"controlLayers_withCount_hidden": "Layer Điều Khiển Được ({{count}} đang ẩn)",
|
||||
"globalReferenceImages_withCount_hidden": "Ảnh Mẫu Toàn Vùng ({{count}} đang ẩn)",
|
||||
"rasterLayer_withCount_other": "Layer Dạng Raster",
|
||||
"globalReferenceImage_withCount_other": "Ảnh Mẫu Toàn Vùng",
|
||||
"copyRasterLayerTo": "Sao Chép $t(controlLayers.rasterLayer) Tới",
|
||||
"copyControlLayerTo": "Sao Chép $t(controlLayers.controlLayer) Tới",
|
||||
"newRegionalGuidance": "$t(controlLayers.regionalGuidance) Mới",
|
||||
"newGallerySessionDesc": "Nó sẽ dọn sạch canvas và các thiết lập trừ model được chọn. Các ảnh được tạo sinh sẽ được chuyển đến thư viện ảnh.",
|
||||
"stagingOnCanvas": "Hiển thị hình ảnh lên",
|
||||
"pullBboxIntoReferenceImage": "Chuyển Hộp Giới Hạn Vào Ảnh Mẫu",
|
||||
"maskFill": "Lấp Đầy Lớp Phủ",
|
||||
"addRasterLayer": "Thêm $t(controlLayers.rasterLayer)",
|
||||
"rasterLayers_withCount_hidden": "Layer Dạng Raster ({{count}} đang ẩn)",
|
||||
"referenceImage": "Ảnh Mẫu",
|
||||
"showProgressOnCanvas": "Hiện Quá Trình Xử Lý Lên Canvas",
|
||||
"prompt": "Lệnh",
|
||||
@@ -1934,34 +1960,23 @@
|
||||
},
|
||||
"addPositivePrompt": "Thêm $t(controlLayers.prompt)",
|
||||
"deleteReferenceImage": "Xoá Ảnh Mẫu",
|
||||
"inpaintMasks_withCount_visible": "Lớp Phủ Inpaint ({{count}})",
|
||||
"disableTransparencyEffect": "Tắt Hiệu Ứng Trong Suốt",
|
||||
"newGallerySession": "Phiên Thư Viện Ảnh Mới",
|
||||
"sendToGalleryDesc": "Bấm 'Kích Hoạt' sẽ tiến hành tạo sinh và lưu ảnh vào thư viện ảnh.",
|
||||
"opacity": "Độ Mờ Đục",
|
||||
"rectangle": "Hình Chữ Nhật",
|
||||
"addNegativePrompt": "Thêm $t(controlLayers.negativePrompt)",
|
||||
"globalReferenceImage": "Ảnh Mẫu Toàn Vùng",
|
||||
"sendToCanvasDesc": "Bấm 'Kích Hoạt' sẽ hiển thị công việc đang xử lý của bạn lên canvas.",
|
||||
"viewProgressInViewer": "Xem quá trình xử lý và ảnh đầu ra trong <Btn>Trình Xem Ảnh</Btn>.",
|
||||
"regionalGuidance_withCount_hidden": "Chỉ Dẫn Khu Vực ({{count}} đang ẩn)",
|
||||
"controlLayer_withCount_other": "Layer Điều Khiển Được",
|
||||
"newInpaintMask": "$t(controlLayers.inpaintMask) Mới",
|
||||
"locked": "Khoá",
|
||||
"newCanvasSession": "Phiên Canvas Mới",
|
||||
"transparency": "Độ Trong Suốt",
|
||||
"showingType": "Hiển Thị {{type}}",
|
||||
"newCanvasSessionDesc": "Nó sẽ dọn sạch canvas và các thiết lập trừ model được chọn. Các ảnh được tạo sinh sẽ được chuyển đến canvas.",
|
||||
"selectObject": {
|
||||
"help2": "Bắt đầu mới một điểm <Bold>Bao Gồm</Bold> trong đối tượng được chọn. Cho thêm điểm để tinh chế phần chọn. Ít điểm hơn thường mang lại kết quả tốt hơn.",
|
||||
"invertSelection": "Đảo Ngược Phần Chọn",
|
||||
"include": "Bao Gồm",
|
||||
"exclude": "Loại Trừ",
|
||||
"reset": "Làm Mới",
|
||||
"saveAs": "Lưu Như",
|
||||
"help1": "Chọn một đối tượng. Thêm điểm <Bold>Bao Gồm</Bold> và <Bold>Loại Trừ</Bold> để chỉ ra phần nào trong layer là đối tượng mong muốn.",
|
||||
"dragToMove": "Kéo kiểm để di chuyển nó",
|
||||
"help3": "Đảo ngược phần chọn để chọn mọi thứ trừ đối tượng được chọn.",
|
||||
"clickToAdd": "Nhấp chuột vào layer để thêm điểm",
|
||||
"clickToRemove": "Nhấp chuột vào một điểm để xoá",
|
||||
"selectObject": "Chọn Đối Tượng",
|
||||
@@ -2195,7 +2210,6 @@
|
||||
"newSession": "Phiên Làm Việc Mới",
|
||||
"resetGenerationSettings": "Khởi Động Lại Cài Đặt Tạo Sinh",
|
||||
"referenceImageRegional": "Ảnh Mẫu (Khu Vực)",
|
||||
"referenceImageGlobal": "Ảnh Mẫu (Toàn Vùng)",
|
||||
"warnings": {
|
||||
"problemsFound": "Phát hiện vấn đề",
|
||||
"unsupportedModel": "layer không được hỗ trợ cho model cơ sở này",
|
||||
@@ -2220,7 +2234,6 @@
|
||||
"pasteToBboxDesc": "Layer Mới (Trong Hộp Giới Hạn)",
|
||||
"pasteToCanvas": "Canvas",
|
||||
"pasteToCanvasDesc": "Layer Mới (Trong Canvas)",
|
||||
"pastedTo": "Dán Vào {{destination}}",
|
||||
"regionCopiedToClipboard": "Sao Chép {{region}} Vào Clipboard",
|
||||
"copyRegionError": "Lỗi khi sao chép {{region}}",
|
||||
"errors": {
|
||||
@@ -2240,7 +2253,6 @@
|
||||
"denoiseLimit": "Giới Hạn Khử Nhiễu",
|
||||
"addImageNoise": "Thêm $t(controlLayers.imageNoise)",
|
||||
"referenceImageEmptyStateWithCanvasOptions": "<UploadButton>Tải lên hình ảnh</UploadButton>, kéo ảnh từ thư viện ảnh vào Ảnh Mẫu này, hoặc <PullBboxButton>kéo hộp giới hạn vào Ảnh Mẫu này</PullBboxButton> để bắt đầu.",
|
||||
"uploadOrDragAnImage": "Kéo ảnh từ thư viện ảnh hoặc <UploadButton>tải lên ảnh</UploadButton>.",
|
||||
"exportCanvasToPSD": "Xuất Canvas Thành File PSD",
|
||||
"ruleOfThirds": "Hiển Thị Quy Tắc Một Phần Ba",
|
||||
"showNonRasterLayers": "Hiển Thị Layer Không Thuộc Dạng Raster (Shift + H)",
|
||||
@@ -2253,7 +2265,38 @@
|
||||
"fitBboxToMasks": "Xếp Vừa Hộp Giới Hạn Vào Lớp Phủ",
|
||||
"invertMask": "Đảo Ngược Lớp Phủ",
|
||||
"maxRefImages": "Ảnh Mẫu Tối Đa",
|
||||
"useAsReferenceImage": "Dùng Làm Ảnh Mẫu"
|
||||
"useAsReferenceImage": "Dùng Làm Ảnh Mẫu",
|
||||
"deletePrompt": "Xoá Lệnh",
|
||||
"addGlobalReferenceImage": "Thêm $t(controlLayers.globalReferenceImage)",
|
||||
"referenceImageGlobal": "Ảnh Mẫu (Toàn Vùng)",
|
||||
"sendingToCanvas": "Chuyển Ảnh Tạo Sinh Vào Canvas",
|
||||
"sendingToGallery": "Chuyển Ảnh Tạo Sinh Vào Thư Viện Ảnh",
|
||||
"sendToGallery": "Chuyển Tới Thư Viện Ảnh",
|
||||
"sendToGalleryDesc": "Bấm 'Kích Hoạt' sẽ tiến hành tạo sinh và lưu ảnh vào thư viện ảnh.",
|
||||
"newImg2ImgCanvasFromImage": "Chuyển Đổi Ảnh Sang Ảnh Mới Từ Ảnh",
|
||||
"sendToCanvasDesc": "Bấm 'Kích Hoạt' sẽ hiển thị công việc đang xử lý của bạn lên canvas.",
|
||||
"viewProgressInViewer": "Xem quá trình xử lý và ảnh đầu ra trong <Btn>Trình Xem Ảnh</Btn>.",
|
||||
"viewProgressOnCanvas": "Xem quá trình xử lý và ảnh đầu ra trong <Btn>Canvas</Btn>.",
|
||||
"globalReferenceImage_withCount_other": "$t(controlLayers.globalReferenceImage)",
|
||||
"regionalGuidance_withCount_hidden": "Chỉ Dẫn Khu Vực ({{count}} đang ẩn)",
|
||||
"controlLayers_withCount_hidden": "Layer Điều Khiển Được ({{count}} đang ẩn)",
|
||||
"rasterLayers_withCount_hidden": "Layer Dạng Raster ({{count}} đang ẩn)",
|
||||
"globalReferenceImages_withCount_hidden": "Ảnh Mẫu Toàn Vùng ({{count}} đang ẩn)",
|
||||
"inpaintMasks_withCount_hidden": "Lớp Phủ Inpaint ({{count}} đang ẩn)",
|
||||
"regionalGuidance_withCount_visible": "Chỉ Dẫn Khu Vực ({{count}})",
|
||||
"controlLayers_withCount_visible": "Layer Điều Khiển Được ({{count}})",
|
||||
"rasterLayers_withCount_visible": "Layer Dạng Raster ({{count}})",
|
||||
"globalReferenceImages_withCount_visible": "Ảnh Mẫu Toàn Vùng ({{count}})",
|
||||
"inpaintMasks_withCount_visible": "Lớp Phủ Inpaint ({{count}})",
|
||||
"layer_withCount_other": "Layer ({{count}})",
|
||||
"pastedTo": "Dán Vào {{destination}}",
|
||||
"stagingOnCanvas": "Hiển thị hình ảnh lên",
|
||||
"newGallerySession": "Phiên Thư Viện Ảnh Mới",
|
||||
"newGallerySessionDesc": "Nó sẽ dọn sạch canvas và các thiết lập trừ model được chọn. Các ảnh được tạo sinh sẽ được chuyển đến thư viện ảnh.",
|
||||
"newCanvasSession": "Phiên Canvas Mới",
|
||||
"newCanvasSessionDesc": "Nó sẽ dọn sạch canvas và các thiết lập trừ model được chọn. Các ảnh được tạo sinh sẽ được chuyển đến canvas.",
|
||||
"replaceCurrent": "Thay Đổi Cái Hiện Tại",
|
||||
"uploadOrDragAnImage": "Kéo ảnh từ thư viện ảnh hoặc <UploadButton>tải lên ảnh</UploadButton>."
|
||||
},
|
||||
"stylePresets": {
|
||||
"negativePrompt": "Lệnh Tiêu Cực",
|
||||
@@ -2331,15 +2374,11 @@
|
||||
"toast": {
|
||||
"imageUploadFailed": "Tải Lên Ảnh Thất Bại",
|
||||
"layerCopiedToClipboard": "Sao Chép Layer Vào Clipboard",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "Tối đa là {{count}} ảnh PNG, JPEG hoặc WEBP.",
|
||||
"imageCopied": "Ảnh Đã Được Sao Chép",
|
||||
"sentToUpscale": "Chuyển Vào Upscale",
|
||||
"unableToLoadImage": "Không Thể Tải Hình Ảnh",
|
||||
"unableToLoadStylePreset": "Không Thể Tải Phong Cách Được Cài Đặt Trước",
|
||||
"stylePresetLoaded": "Phong Cách Được Cài Đặt Trước Đã Tải",
|
||||
"imageNotLoadedDesc": "Không thể tìm thấy ảnh",
|
||||
"imageSaved": "Ảnh Đã Lưu",
|
||||
"imageSavingFailed": "Lưu Ảnh Thất Bại",
|
||||
"unableToLoadImageMetadata": "Không Thể Tải Metadata Của Ảnh",
|
||||
"workflowLoaded": "Workflow Đã Tải",
|
||||
"uploadFailed": "Tải Lên Thất Bại",
|
||||
@@ -2351,17 +2390,14 @@
|
||||
"importFailed": "Nhập Vào Thất Bại",
|
||||
"importSuccessful": "Nhập Vào Thành Công",
|
||||
"workflowDeleted": "Workflow Đã Xoá",
|
||||
"setControlImage": "Đặt làm ảnh điều khiển được",
|
||||
"connected": "Kết Nối Đến Server",
|
||||
"imageUploaded": "Ảnh Đã Được Tải Lên",
|
||||
"invalidUpload": "Dữ Liệu Tải Lên Không Hợp Lệ",
|
||||
"modelImportCanceled": "Nhập Vào Model Thất Bại",
|
||||
"parameters": "Tham Số",
|
||||
"parameterSet": "Gợi Lại Tham Số",
|
||||
"parameterSetDesc": "Gợi lại {{parameter}}",
|
||||
"loadedWithWarnings": "Đã Tải Workflow Với Cảnh Báo",
|
||||
"outOfMemoryErrorDesc": "Thiết lập tạo sinh hiện tại đã vượt mức cho phép của thiết bị. Hãy điều chỉnh thiết lập và thử lại.",
|
||||
"setNodeField": "Đặt làm vùng node",
|
||||
"problemRetrievingWorkflow": "Có Vấn Đề Khi Lấy Lại Workflow",
|
||||
"somethingWentWrong": "Có Vấn Đề Phát Sinh",
|
||||
"problemDeletingWorkflow": "Có Vấn Đề Khi Xoá Workflow",
|
||||
@@ -2371,13 +2407,12 @@
|
||||
"errorCopied": "Lỗi Khi Sao Chép",
|
||||
"prunedQueue": "Cắt Bớt Hàng Đợi",
|
||||
"imagesWillBeAddedTo": "Ảnh đã tải lên sẽ được thêm vào tài nguyên của bảng {{boardName}}.",
|
||||
"baseModelChangedCleared_other": "Dọn sạch hoặc tắt {{count}} model phụ không tương thích",
|
||||
"baseModelChangedCleared_other": "Cập nhật, dọn sạch hoặc tắt {{count}} model phụ không tương thích",
|
||||
"canceled": "Quá Trình Xử Lý Đã Huỷ",
|
||||
"baseModelChanged": "Model Cơ Sở Đã Đổi",
|
||||
"addedToUncategorized": "Thêm vào tài nguyên của bảng $t(boards.uncategorized)",
|
||||
"linkCopied": "Đường Liên Kết Đã Được Sao Chép",
|
||||
"outOfMemoryError": "Lỗi Vượt Quá Bộ Nhớ",
|
||||
"layerSavedToAssets": "Lưu Layer Vào Khu Tài Nguyên",
|
||||
"modelAddedSimple": "Đã Thêm Model Vào Hàng Đợi",
|
||||
"parametersSet": "Tham Số Đã Được Gợi Lại",
|
||||
"parameterNotSetDesc": "Không thể gợi lại {{parameter}}",
|
||||
@@ -2398,21 +2433,15 @@
|
||||
"chatGPT4oIncompatibleGenerationMode": "ChatGPT 4o chỉ hỗ trợ Từ Ngữ Sang Hình Ảnh và Hình Ảnh Sang Hình Ảnh. Hãy dùng model khác cho các tác vụ Inpaint và Outpaint.",
|
||||
"imagenIncompatibleGenerationMode": "Google {{model}} chỉ hỗ trợ Từ Ngữ Sang Hình Ảnh. Dùng các model khác cho Hình Ảnh Sang Hình Ảnh, Inpaint và Outpaint.",
|
||||
"fluxKontextIncompatibleGenerationMode": "FLUX Kontext không hỗ trợ tạo sinh từ hình ảnh từ canvas. Thử sử dụng Ảnh Mẫu và tắt các Layer Dạng Raster.",
|
||||
"noRasterLayers": "Không Tìm Thấy Layer Dạng Raster",
|
||||
"noRasterLayersDesc": "Tạo ít nhất một layer dạng raster để xuất file PSD",
|
||||
"noActiveRasterLayers": "Không Có Layer Dạng Raster Hoạt Động",
|
||||
"noActiveRasterLayersDesc": "Khởi động ít nhất một layer dạng raster để xuất file PSD",
|
||||
"noVisibleRasterLayers": "Không Có Layer Dạng Raster Hiển Thị",
|
||||
"noVisibleRasterLayersDesc": "Khởi động ít nhất một layer dạng raster để xuất file PSD",
|
||||
"invalidCanvasDimensions": "Kích Thước Canvas Không Phù Hợp",
|
||||
"canvasTooLarge": "Canvas Quá Lớn",
|
||||
"canvasTooLargeDesc": "Kích thước canvas vượt mức tối đa cho phép để xuất file PSD. Giảm cả chiều dài và chiều rộng chủa canvas và thử lại.",
|
||||
"failedToProcessLayers": "Thất Bại Khi Xử Lý Layer",
|
||||
"psdExportSuccess": "Xuất File PSD Hoàn Tất",
|
||||
"psdExportSuccessDesc": "Thành công xuất {{count}} layer sang file PSD",
|
||||
"problemExportingPSD": "Có Vấn Đề Khi Xuất File PSD",
|
||||
"canvasManagerNotAvailable": "Trình Quản Lý Canvas Không Có Sẵn",
|
||||
"noValidLayerAdapters": "Không có Layer Adaper Phù Hợp",
|
||||
"promptGenerationStarted": "Trình tạo sinh lệnh khởi động",
|
||||
"uploadAndPromptGenerationFailed": "Thất bại khi tải lên ảnh để tạo sinh lệnh",
|
||||
"promptExpansionFailed": "Có vấn đề xảy ra. Hãy thử mở rộng lệnh lại.",
|
||||
@@ -2420,6 +2449,20 @@
|
||||
"maskInvertFailed": "Thất Bại Khi Đảo Ngược Lớp Phủ",
|
||||
"noVisibleMasks": "Không Có Lớp Phủ Đang Hiển Thị",
|
||||
"noVisibleMasksDesc": "Tạo hoặc bật ít nhất một lớp phủ inpaint để đảo ngược",
|
||||
"imageNotLoadedDesc": "Không thể tìm thấy ảnh",
|
||||
"imageSaved": "Ảnh Đã Lưu",
|
||||
"imageSavingFailed": "Lưu Ảnh Thất Bại",
|
||||
"invalidUpload": "Dữ Liệu Tải Lên Không Hợp Lệ",
|
||||
"layerSavedToAssets": "Lưu Layer Vào Khu Tài Nguyên",
|
||||
"noRasterLayers": "Không Tìm Thấy Layer Dạng Raster",
|
||||
"noRasterLayersDesc": "Tạo ít nhất một layer dạng raster để xuất file PSD",
|
||||
"noActiveRasterLayers": "Không Có Layer Dạng Raster Hoạt Động",
|
||||
"noActiveRasterLayersDesc": "Bật ít nhất một layer dạng raster để xuất file PSD",
|
||||
"failedToProcessLayers": "Thất Bại Khi Xử Lý Layer",
|
||||
"noValidLayerAdapters": "Không có Layer Adaper Phù Hợp",
|
||||
"setControlImage": "Đặt làm ảnh điều khiển được",
|
||||
"setNodeField": "Đặt làm vùng node",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "Cần tối đa {{count}} ảnh PNG, JPEG, hoặc WEBP.",
|
||||
"noInpaintMaskSelected": "Không Có Lớp Phủ Inpant Được Chọn",
|
||||
"noInpaintMaskSelectedDesc": "Chọn một lớp phủ inpaint để đảo ngược",
|
||||
"invalidBbox": "Hộp Giới Hạn Không Hợp Lệ",
|
||||
@@ -2436,7 +2479,8 @@
|
||||
"queue": "Queue (Hàng Đợi)",
|
||||
"workflows": "Workflow (Luồng Làm Việc)",
|
||||
"workflowsTab": "$t(common.tab) $t(ui.tabs.workflows)",
|
||||
"generate": "Tạo Sinh"
|
||||
"generate": "Tạo Sinh",
|
||||
"video": "Video"
|
||||
},
|
||||
"launchpad": {
|
||||
"workflowsTitle": "Đi sâu hơn với Workflow.",
|
||||
@@ -2514,13 +2558,23 @@
|
||||
"generate": {
|
||||
"canvasCalloutTitle": "Đang tìm cách để điều khiển, chỉnh sửa, và làm lại ảnh?",
|
||||
"canvasCalloutLink": "Vào Canvas cho nhiều tính năng hơn."
|
||||
},
|
||||
"videoTitle": "Tạo sinh video từ lệnh chữ.",
|
||||
"video": {
|
||||
"startingFrameCalloutTitle": "Thêm Khung Hình Bắt Đầu",
|
||||
"startingFrameCalloutDesc": "Thêm ảnh nhằm điều khiển khung hình đầu của video."
|
||||
},
|
||||
"addStartingFrame": {
|
||||
"title": "Thêm Khung Hình Bắt Đầu",
|
||||
"description": "Thêm ảnh nhằm điều khiển khung hình đầu của video."
|
||||
}
|
||||
},
|
||||
"panels": {
|
||||
"launchpad": "Launchpad",
|
||||
"workflowEditor": "Trình Biên Tập Workflow",
|
||||
"imageViewer": "Trình Xem Ảnh",
|
||||
"canvas": "Canvas"
|
||||
"imageViewer": "Trình Xem",
|
||||
"canvas": "Canvas",
|
||||
"video": "Video"
|
||||
}
|
||||
},
|
||||
"workflows": {
|
||||
@@ -2535,28 +2589,20 @@
|
||||
"saveWorkflowAs": "Lưu Workflow Như",
|
||||
"downloadWorkflow": "Lưu Vào Tệp",
|
||||
"noWorkflows": "Không Có Workflow",
|
||||
"problemLoading": "Có Vấn Đề Khi Tải Workflow",
|
||||
"clearWorkflowSearchFilter": "Xoá Workflow Khỏi Bộ Lọc Tìm Kiếm",
|
||||
"defaultWorkflows": "Workflow Mặc Định",
|
||||
"userWorkflows": "Workflow Của Người Dùng",
|
||||
"projectWorkflows": "Dự Án Workflow",
|
||||
"savingWorkflow": "Đang Lưu Workflow...",
|
||||
"ascending": "Tăng Dần",
|
||||
"loading": "Đang Tải Workflow",
|
||||
"chooseWorkflowFromLibrary": "Chọn Workflow Từ Thư Viện",
|
||||
"workflows": "Workflow",
|
||||
"copyShareLinkForWorkflow": "Sao Chép Liên Kết Chia Sẻ Cho Workflow",
|
||||
"openWorkflow": "Mở Workflow",
|
||||
"name": "Tên",
|
||||
"unnamedWorkflow": "Workflow Vô Danh",
|
||||
"saveWorkflow": "Lưu Workflow",
|
||||
"problemSavingWorkflow": "Có Vấn Đề Khi Lưu Workflow",
|
||||
"noDescription": "Không có mô tả",
|
||||
"updated": "Đã Cập Nhật",
|
||||
"uploadWorkflow": "Tải Từ Tệp",
|
||||
"autoLayout": "Bố Trí Tự Động",
|
||||
"loadWorkflow": "$t(common.load) Workflow",
|
||||
"searchWorkflows": "Tìm Workflow",
|
||||
"newWorkflowCreated": "Workflow Mới Được Tạo",
|
||||
"workflowCleared": "Đã Dọn Dẹp Workflow",
|
||||
"loadFromGraph": "Tải Workflow Từ Đồ Thị",
|
||||
@@ -2567,7 +2613,6 @@
|
||||
"opened": "Đã Mở",
|
||||
"deleteWorkflow": "Xoá Workflow",
|
||||
"workflowEditorMenu": "Menu Biên Tập Workflow",
|
||||
"openLibrary": "Mở Thư Viện",
|
||||
"builder": {
|
||||
"resetAllNodeFields": "Tải Lại Các Vùng Node",
|
||||
"builder": "Trình Tạo Vùng Nhập",
|
||||
@@ -2583,7 +2628,6 @@
|
||||
"multiLine": "Nhiều Dòng",
|
||||
"slider": "Thanh Trượt",
|
||||
"both": "Cả Hai",
|
||||
"emptyRootPlaceholderViewMode": "Chọn Chỉnh Sửa để bắt đầu tạo nên một vùng nhập cho workflow này.",
|
||||
"emptyRootPlaceholderEditMode": "Kéo thành phần vùng nhập hoặc vùng node vào đây để bắt đầu.",
|
||||
"containerPlaceholder": "Hộp Chứa Trống",
|
||||
"headingPlaceholder": "Đầu Dòng Trống",
|
||||
@@ -2592,7 +2636,6 @@
|
||||
"deleteAllElements": "Xóa Tất Cả Thành Phần",
|
||||
"nodeField": "Vùng Node",
|
||||
"nodeFieldTooltip": "Để thêm vùng node, bấm vào dấu cộng nhỏ trên vùng trong Trình Biên Tập Workflow, hoặc kéo vùng theo tên của nó vào vùng nhập.",
|
||||
"workflowBuilderAlphaWarning": "Trình tạo vùng nhập đang trong giai đoạn alpha. Nó có thể xuất hiện những thay đổi đột ngột trước khi chính thức được phát hành.",
|
||||
"container": "Hộp Chứa",
|
||||
"heading": "Đầu Dòng",
|
||||
"text": "Văn Bản",
|
||||
@@ -2638,25 +2681,36 @@
|
||||
"errorWorkflowHasUnpublishableNodes": "Workflow có lô node, node sản sinh, hoặc node tách metadata",
|
||||
"removeFromForm": "Xóa Khỏi Vùng Nhập",
|
||||
"showShuffle": "Hiện Xáo Trộn",
|
||||
"shuffle": "Xáo Trộn"
|
||||
"shuffle": "Xáo Trộn",
|
||||
"emptyRootPlaceholderViewMode": "Chọn Chỉnh Sửa để bắt đầu tạo nên một vùng nhập cho workflow này.",
|
||||
"workflowBuilderAlphaWarning": "Trình tạo vùng nhập đang trong giai đoạn alpha. Nó có thể xuất hiện những thay đổi đột ngột trước khi chính thức được phát hành."
|
||||
},
|
||||
"yourWorkflows": "Workflow Của Bạn",
|
||||
"browseWorkflows": "Khám Phá Workflow",
|
||||
"workflowThumbnail": "Ảnh Minh Họa Workflow",
|
||||
"saveChanges": "Lưu Thay Đổi",
|
||||
"allLoaded": "Đã Tải Tất Cả Workflow",
|
||||
"shared": "Nhóm",
|
||||
"searchPlaceholder": "Tìm theo tên, mô tả, hoặc nhãn",
|
||||
"filterByTags": "Lọc Theo Nhãn",
|
||||
"recentlyOpened": "Mở Gần Đây",
|
||||
"private": "Cá Nhân",
|
||||
"loadMore": "Tải Thêm",
|
||||
"view": "Xem",
|
||||
"deselectAll": "Huỷ Chọn Tất Cả",
|
||||
"noRecentWorkflows": "Không Có Workflows Gần Đây",
|
||||
"recommended": "Có Thể Bạn Sẽ Cần",
|
||||
"emptyStringPlaceholder": "<xâu ký tự trống>",
|
||||
"published": "Đã Đăng"
|
||||
"published": "Đã Đăng",
|
||||
"defaultWorkflows": "Workflow Mặc Định",
|
||||
"userWorkflows": "Workflow Của Người Dùng",
|
||||
"projectWorkflows": "Dự Án Workflow",
|
||||
"allLoaded": "Đã Tải Tất Cả Workflow",
|
||||
"filterByTags": "Lọc Theo Nhãn",
|
||||
"noRecentWorkflows": "Không Có Workflows Gần Đây",
|
||||
"openWorkflow": "Mở Workflow",
|
||||
"problemLoading": "Có Vấn Đề Khi Tải Workflow",
|
||||
"noDescription": "Không có mô tả",
|
||||
"searchWorkflows": "Tìm Workflow",
|
||||
"clearWorkflowSearchFilter": "Xoá Workflow Khỏi Bộ Lọc Tìm Kiếm",
|
||||
"openLibrary": "Mở Thư Viện"
|
||||
},
|
||||
"upscaling": {
|
||||
"missingUpscaleInitialImage": "Thiếu ảnh dùng để upscale",
|
||||
@@ -2693,11 +2747,11 @@
|
||||
"whatsNewInInvoke": "Có Gì Mới Ở Invoke",
|
||||
"readReleaseNotes": "Đọc Ghi Chú Phát Hành",
|
||||
"watchRecentReleaseVideos": "Xem Video Phát Hành Mới Nhất",
|
||||
"watchUiUpdatesOverview": "Xem Tổng Quan Về Những Cập Nhật Cho Giao Diện Người Dùng",
|
||||
"items": [
|
||||
"Misc QoL: Bật/Tắt hiển thị hộp giới hạn, đánh dấu node bị lỗi, chặn lỗi thêm node vào vùng nhập nhiều lần, khả năng đọc lại metadata của CLIP Skip",
|
||||
"Giảm lượng tiêu thụ VRAM cho các ảnh mẫu Kontext và mã hóa VAE"
|
||||
]
|
||||
"Canvas: Chia tách màu nổi và màu nền - bật/tắt với 'x', khởi động lại về dạng đen trắng với 'd'",
|
||||
"LoRA: Đặt khối lượng mặc định cho LoRA trong Trình Quản Lý Model"
|
||||
],
|
||||
"watchUiUpdatesOverview": "Xem Tổng Quan Về Những Cập Nhật Cho Giao Diện Người Dùng"
|
||||
},
|
||||
"upsell": {
|
||||
"professional": "Chuyên Nghiệp",
|
||||
@@ -2725,5 +2779,12 @@
|
||||
"clearSucceeded": "Cache Model Đã Được Dọn",
|
||||
"clearFailed": "Có Vấn Đề Khi Dọn Cache Model",
|
||||
"clear": "Dọn Cache Model"
|
||||
},
|
||||
"lora": {
|
||||
"weight": "Trọng Lượng"
|
||||
},
|
||||
"video": {
|
||||
"noVideoSelected": "Không có video được chọn",
|
||||
"selectFromGallery": "Chọn một video trong thư viện để xem"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -25,7 +25,6 @@
|
||||
"batch": "批次管理器",
|
||||
"communityLabel": "社区",
|
||||
"modelManager": "模型管理器",
|
||||
"imageFailedToLoad": "无法加载图像",
|
||||
"learnMore": "了解更多",
|
||||
"advanced": "高级",
|
||||
"t2iAdapter": "T2I Adapter",
|
||||
@@ -51,23 +50,19 @@
|
||||
"somethingWentWrong": "出了点问题",
|
||||
"copyError": "$t(gallery.copy) 错误",
|
||||
"input": "输入",
|
||||
"notInstalled": "非 $t(common.installed)",
|
||||
"delete": "删除",
|
||||
"updated": "已上传",
|
||||
"save": "保存",
|
||||
"created": "已创建",
|
||||
"prevPage": "上一页",
|
||||
"unknownError": "未知错误",
|
||||
"direction": "指向",
|
||||
"orderBy": "排序方式:",
|
||||
"nextPage": "下一页",
|
||||
"saveAs": "保存为",
|
||||
"ai": "ai",
|
||||
"or": "或",
|
||||
"aboutDesc": "使用 Invoke 工作?来看看:",
|
||||
"add": "添加",
|
||||
"copy": "复制",
|
||||
"localSystem": "本地系统",
|
||||
"aboutHeading": "掌握你的创造力",
|
||||
"enabled": "已启用",
|
||||
"disabled": "已禁用",
|
||||
@@ -78,7 +73,6 @@
|
||||
"selected": "选中的",
|
||||
"green": "绿",
|
||||
"blue": "蓝",
|
||||
"goTo": "前往",
|
||||
"dontShowMeThese": "请勿显示这些内容",
|
||||
"beta": "测试版",
|
||||
"toResolve": "解决",
|
||||
@@ -104,13 +98,11 @@
|
||||
"galleryImageSize": "预览大小",
|
||||
"gallerySettings": "预览设置",
|
||||
"autoSwitchNewImages": "自动切换到新图像",
|
||||
"noImagesInGallery": "无图像可用于显示",
|
||||
"deleteImage_other": "删除{{count}}张图片",
|
||||
"deleteImagePermanent": "删除的图片无法被恢复。",
|
||||
"autoAssignBoardOnClick": "点击后自动分配面板",
|
||||
"featuresWillReset": "如果您删除该图像,这些功能会立即被重置。",
|
||||
"loading": "加载中",
|
||||
"unableToLoad": "无法加载图库",
|
||||
"currentlyInUse": "该图像目前在以下功能中使用:",
|
||||
"copy": "复制",
|
||||
"download": "下载",
|
||||
@@ -125,7 +117,6 @@
|
||||
"starImage": "收藏图像",
|
||||
"alwaysShowImageSizeBadge": "始终显示图像尺寸",
|
||||
"selectForCompare": "选择以比较",
|
||||
"selectAnImageToCompare": "选择一个图像进行比较",
|
||||
"slider": "滑块",
|
||||
"sideBySide": "并排",
|
||||
"bulkDownloadFailed": "下载失败",
|
||||
@@ -148,7 +139,6 @@
|
||||
"newestFirst": "最新在前",
|
||||
"compareHelp4": "按 <Kbd>Z</Kbd>或 <Kbd>Esc</Kbd> 键退出。",
|
||||
"searchImages": "按元数据搜索",
|
||||
"jump": "跳过",
|
||||
"compareHelp2": "按 <Kbd>M</Kbd> 键切换不同的比较模式。",
|
||||
"displayBoardSearch": "板块搜索",
|
||||
"displaySearch": "图像搜索",
|
||||
@@ -161,8 +151,6 @@
|
||||
"gallery": "画廊",
|
||||
"move": "移动",
|
||||
"imagesTab": "您在Invoke中创建和保存的图片。",
|
||||
"openViewer": "打开查看器",
|
||||
"closeViewer": "关闭查看器",
|
||||
"assetsTab": "您已上传用于项目的文件。"
|
||||
},
|
||||
"hotkeys": {
|
||||
@@ -573,9 +561,7 @@
|
||||
"huggingFacePlaceholder": "所有者或模型名称",
|
||||
"huggingFaceRepoID": "HuggingFace仓库ID",
|
||||
"loraTriggerPhrases": "LoRA 触发词",
|
||||
"ipAdapters": "IP适配器",
|
||||
"spandrelImageToImage": "图生图(Spandrel)",
|
||||
"starterModelsInModelManager": "您可以在模型管理器中找到初始模型",
|
||||
"noDefaultSettings": "此模型没有配置默认设置。请访问模型管理器添加默认设置。",
|
||||
"clipEmbed": "CLIP 嵌入",
|
||||
"defaultSettingsOutOfSync": "某些设置与模型的默认值不匹配:",
|
||||
@@ -626,12 +612,10 @@
|
||||
"scaledHeight": "缩放长度",
|
||||
"infillMethod": "填充方法",
|
||||
"tileSize": "方格尺寸",
|
||||
"downloadImage": "下载图像",
|
||||
"usePrompt": "使用提示",
|
||||
"useSeed": "使用种子",
|
||||
"useAll": "使用所有参数",
|
||||
"info": "信息",
|
||||
"showOptionsPanel": "显示侧栏浮窗 (O 或 T)",
|
||||
"seamlessYAxis": "无缝平铺 Y 轴",
|
||||
"seamlessXAxis": "无缝平铺 X 轴",
|
||||
"denoisingStrength": "去噪强度",
|
||||
@@ -657,15 +641,11 @@
|
||||
"addingImagesTo": "添加图像到",
|
||||
"noPrompts": "没有已生成的提示词",
|
||||
"canvasIsFiltering": "画布正在过滤",
|
||||
"fluxModelIncompatibleScaledBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),缩放后的边界框高度为 {{height}}",
|
||||
"noCLIPEmbedModelSelected": "未为FLUX生成选择CLIP嵌入模型",
|
||||
"noFLUXVAEModelSelected": "未为FLUX生成选择VAE模型",
|
||||
"canvasIsRasterizing": "画布正在栅格化",
|
||||
"canvasIsCompositing": "画布正在合成",
|
||||
"fluxModelIncompatibleBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),边界框宽度为 {{width}}",
|
||||
"fluxModelIncompatibleScaledBboxWidth": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),缩放后的边界框宽度为 {{width}}",
|
||||
"noT5EncoderModelSelected": "未为FLUX生成选择T5编码器模型",
|
||||
"fluxModelIncompatibleBboxHeight": "$t(parameters.invoke.fluxRequiresDimensionsToBeMultipleOf16),边界框高度为 {{height}}",
|
||||
"canvasIsTransforming": "画布正在变换"
|
||||
},
|
||||
"patchmatchDownScaleSize": "缩小",
|
||||
@@ -729,8 +709,6 @@
|
||||
"informationalPopoversDisabledDesc": "信息提示框已被禁用.请在设置中重新启用.",
|
||||
"enableModelDescriptions": "在下拉菜单中启用模型描述",
|
||||
"confirmOnNewSession": "新会话时确认",
|
||||
"modelDescriptionsDisabledDesc": "下拉菜单中的模型描述已被禁用。可在设置中启用。",
|
||||
"modelDescriptionsDisabled": "下拉菜单中的模型描述已禁用",
|
||||
"showDetailedInvocationProgress": "显示进度详情"
|
||||
},
|
||||
"toast": {
|
||||
@@ -746,14 +724,11 @@
|
||||
"problemCopyingImage": "无法复制图像",
|
||||
"modelAddedSimple": "模型已加入队列",
|
||||
"loadedWithWarnings": "已加载带有警告的工作流",
|
||||
"setControlImage": "设为控制图像",
|
||||
"setNodeField": "设为节点字段",
|
||||
"imageUploaded": "图像已上传",
|
||||
"addedToBoard": "添加到{{name}}的资产中",
|
||||
"workflowLoaded": "工作流已加载",
|
||||
"imageUploadFailed": "图像上传失败",
|
||||
"baseModelChangedCleared_other": "已清除或禁用{{count}}个不兼容的子模型",
|
||||
"invalidUpload": "无效的上传",
|
||||
"problemDeletingWorkflow": "删除工作流时出现问题",
|
||||
"workflowDeleted": "已删除工作流",
|
||||
"problemRetrievingWorkflow": "检索工作流时发生问题",
|
||||
@@ -773,21 +748,16 @@
|
||||
"modelImportCanceled": "模型导入已取消",
|
||||
"importFailed": "导入失败",
|
||||
"importSuccessful": "导入成功",
|
||||
"layerSavedToAssets": "图层已保存到资产",
|
||||
"sentToUpscale": "已发送到放大处理",
|
||||
"addedToUncategorized": "已添加到看板 $t(boards.uncategorized) 的资产中",
|
||||
"linkCopied": "链接已复制",
|
||||
"uploadFailedInvalidUploadDesc_withCount_other": "最多只能上传 {{count}} 张 PNG 或 JPEG 图像。",
|
||||
"problemSavingLayer": "无法保存图层",
|
||||
"unableToLoadImage": "无法加载图像",
|
||||
"imageNotLoadedDesc": "无法找到图像",
|
||||
"unableToLoadStylePreset": "无法加载样式预设",
|
||||
"stylePresetLoaded": "样式预设已加载",
|
||||
"problemCopyingLayer": "无法复制图层",
|
||||
"sentToCanvas": "已发送到画布",
|
||||
"unableToLoadImageMetadata": "无法加载图像元数据",
|
||||
"imageSaved": "图像已保存",
|
||||
"imageSavingFailed": "图像保存失败",
|
||||
"layerCopiedToClipboard": "图层已复制到剪贴板",
|
||||
"imagesWillBeAddedTo": "上传的图像将添加到看板 {{boardName}} 的资产中。"
|
||||
},
|
||||
@@ -815,11 +785,8 @@
|
||||
"fitViewportNodes": "自适应视图",
|
||||
"showMinimapnodes": "显示缩略图",
|
||||
"hideMinimapnodes": "隐藏缩略图",
|
||||
"showLegendNodes": "显示字段类型图例",
|
||||
"hideLegendNodes": "隐藏字段类型图例",
|
||||
"downloadWorkflow": "下载工作流 JSON",
|
||||
"workflowDescription": "简述",
|
||||
"versionUnknown": " 未知版本",
|
||||
"noNodeSelected": "无选中的节点",
|
||||
"addNode": "添加节点",
|
||||
"unableToValidateWorkflow": "无法验证工作流",
|
||||
@@ -829,9 +796,7 @@
|
||||
"workflowContact": "联系",
|
||||
"animatedEdges": "边缘动效",
|
||||
"nodeTemplate": "节点模板",
|
||||
"unableToLoadWorkflow": "无法加载工作流",
|
||||
"snapToGrid": "对齐网格",
|
||||
"noFieldsLinearview": "线性视图中未添加任何字段",
|
||||
"nodeSearch": "检索节点",
|
||||
"version": "版本",
|
||||
"validateConnections": "验证连接和节点图",
|
||||
@@ -846,8 +811,6 @@
|
||||
"fieldTypesMustMatch": "类型必须匹配",
|
||||
"workflow": "工作流",
|
||||
"animatedEdgesHelp": "为选中边缘和其连接的选中节点的边缘添加动画",
|
||||
"unknownTemplate": "未知模板",
|
||||
"removeLinearView": "从线性视图中移除",
|
||||
"workflowTags": "标签",
|
||||
"fullyContainNodesHelp": "节点必须完全位于选择框中才能被选中",
|
||||
"workflowValidation": "工作流验证错误",
|
||||
@@ -881,7 +844,6 @@
|
||||
"node": "节点",
|
||||
"collection": "合集",
|
||||
"string": "字符串",
|
||||
"mismatchedVersion": "无效的节点:类型为 {{type}} 的节点 {{node}} 版本不匹配(是否尝试更新?)",
|
||||
"cannotDuplicateConnection": "无法创建重复的连接",
|
||||
"enum": "Enum (枚举)",
|
||||
"float": "浮点",
|
||||
@@ -892,7 +854,6 @@
|
||||
"unableToUpdateNodes_other": "{{count}} 个节点无法完成更新",
|
||||
"inputFieldTypeParseError": "无法解析 {{node}} 的输入类型 {{field}}。({{message}})",
|
||||
"unsupportedArrayItemType": "不支持的数组类型 \"{{type}}\"",
|
||||
"addLinearView": "添加到线性视图",
|
||||
"targetNodeFieldDoesNotExist": "无效的边缘:{{node}} 的目标/输入区域 {{field}} 不存在",
|
||||
"unsupportedMismatchedUnion": "合集或标量类型与基类 {{firstType}} 和 {{secondType}} 不匹配",
|
||||
"allNodesUpdated": "已更新所有节点",
|
||||
@@ -912,7 +873,6 @@
|
||||
"collectionOrScalarFieldType": "{{name}} (单一项目或项目集合)",
|
||||
"nodeVersion": "节点版本",
|
||||
"deletedInvalidEdge": "已删除无效的边缘 {{source}} -> {{target}}",
|
||||
"unknownInput": "未知输入:{{name}}",
|
||||
"prototypeDesc": "此调用是一个原型 (prototype)。它可能会在本项目更新期间发生破坏性更改,并且随时可能被删除。",
|
||||
"betaDesc": "此调用尚处于测试阶段。在稳定之前,它可能会在项目更新期间发生破坏性更改。本项目计划长期支持这种调用。",
|
||||
"newWorkflow": "新建工作流",
|
||||
@@ -924,7 +884,6 @@
|
||||
"missingNode": "缺少调用节点",
|
||||
"missingInvocationTemplate": "缺少调用模版",
|
||||
"noFieldsViewMode": "此工作流程未选择任何要显示的字段.请查看完整工作流程以进行配置.",
|
||||
"reorderLinearView": "调整线性视图顺序",
|
||||
"viewMode": "在线性视图中使用",
|
||||
"showEdgeLabelsHelp": "在边缘上显示标签,指示连接的节点",
|
||||
"cannotMixAndMatchCollectionItemTypes": "集合项目类型不能混用",
|
||||
@@ -998,7 +957,6 @@
|
||||
"session": "会话",
|
||||
"enqueueing": "队列中的批次",
|
||||
"graphFailedToQueue": "节点图加入队列失败",
|
||||
"batchFieldValues": "批处理值",
|
||||
"time": "时间",
|
||||
"openQueue": "打开队列",
|
||||
"prompts_other": "提示词",
|
||||
@@ -1017,18 +975,14 @@
|
||||
"refinerStart": "Refiner 开始作用时机",
|
||||
"scheduler": "调度器",
|
||||
"cfgScale": "CFG 等级",
|
||||
"negStylePrompt": "负向样式提示词",
|
||||
"noModelsAvailable": "无可用模型",
|
||||
"negAestheticScore": "负向美学评分",
|
||||
"denoisingStrength": "去噪强度",
|
||||
"refinermodel": "Refiner 模型",
|
||||
"posAestheticScore": "正向美学评分",
|
||||
"concatPromptStyle": "链接提示词 & 样式",
|
||||
"loading": "加载中...",
|
||||
"steps": "步数",
|
||||
"posStylePrompt": "正向样式提示词",
|
||||
"refiner": "Refiner",
|
||||
"freePromptStyle": "手动输入样式提示词",
|
||||
"refinerSteps": "精炼步数"
|
||||
},
|
||||
"metadata": {
|
||||
@@ -1055,8 +1009,6 @@
|
||||
"vae": "VAE",
|
||||
"cfgRescaleMultiplier": "$t(parameters.cfgRescaleMultiplier)",
|
||||
"allPrompts": "所有提示",
|
||||
"parsingFailed": "解析失败",
|
||||
"recallParameter": "调用{{label}}",
|
||||
"imageDimensions": "图像尺寸",
|
||||
"parameterSet": "已设置参数{{parameter}}",
|
||||
"guidance": "指导",
|
||||
@@ -1067,11 +1019,9 @@
|
||||
"models": {
|
||||
"noMatchingModels": "无相匹配的模型",
|
||||
"loading": "加载中",
|
||||
"noMatchingLoRAs": "无相匹配的 LoRA",
|
||||
"noModelsAvailable": "无可用模型",
|
||||
"selectModel": "选择一个模型",
|
||||
"noRefinerModelsInstalled": "无已安装的 SDXL Refiner 模型",
|
||||
"noLoRAsInstalled": "无已安装的 LoRA",
|
||||
"addLora": "添加 LoRA",
|
||||
"lora": "LoRA",
|
||||
"defaultVAE": "默认 VAE",
|
||||
@@ -1100,10 +1050,8 @@
|
||||
"deletedBoardsCannotbeRestored": "删除的面板无法恢复。选择“仅删除面板”选项后,相关图片将会被移至未分类区域。",
|
||||
"movingImagesToBoard_other": "移动 {{count}} 张图像到面板:",
|
||||
"selectedForAutoAdd": "已选中自动添加",
|
||||
"hideBoards": "隐藏面板",
|
||||
"noBoards": "没有{{boardType}}类型的面板",
|
||||
"unarchiveBoard": "恢复面板",
|
||||
"viewBoards": "查看面板",
|
||||
"addPrivateBoard": "创建私密面板",
|
||||
"addSharedBoard": "创建共享面板",
|
||||
"boards": "面板",
|
||||
@@ -1572,8 +1520,6 @@
|
||||
"useCache": "使用缓存"
|
||||
},
|
||||
"hrf": {
|
||||
"enableHrf": "启用高分辨率修复",
|
||||
"upscaleMethod": "放大方法",
|
||||
"metadata": {
|
||||
"strength": "高分辨率修复强度",
|
||||
"enabled": "高分辨率修复已启用",
|
||||
@@ -1586,20 +1532,15 @@
|
||||
"workflowEditorMenu": "工作流编辑器菜单",
|
||||
"workflowName": "工作流名称",
|
||||
"saveWorkflow": "保存工作流",
|
||||
"openWorkflow": "打开工作流",
|
||||
"clearWorkflowSearchFilter": "清除工作流检索过滤器",
|
||||
"workflowLibrary": "工作流库",
|
||||
"downloadWorkflow": "保存到文件",
|
||||
"workflowSaved": "已保存工作流",
|
||||
"unnamedWorkflow": "未命名的工作流",
|
||||
"savingWorkflow": "保存工作流中...",
|
||||
"problemLoading": "加载工作流时出现问题",
|
||||
"loading": "加载工作流中",
|
||||
"searchWorkflows": "检索工作流",
|
||||
"problemSavingWorkflow": "保存工作流时出现问题",
|
||||
"deleteWorkflow": "删除工作流",
|
||||
"workflows": "工作流",
|
||||
"noDescription": "无描述",
|
||||
"uploadWorkflow": "从文件中加载",
|
||||
"newWorkflowCreated": "已创建新的工作流",
|
||||
"name": "名称",
|
||||
@@ -1619,9 +1560,6 @@
|
||||
"copyShareLinkForWorkflow": "复制工作流程的分享链接",
|
||||
"delete": "删除",
|
||||
"download": "下载",
|
||||
"defaultWorkflows": "默认工作流程",
|
||||
"userWorkflows": "用户工作流程",
|
||||
"projectWorkflows": "项目工作流程",
|
||||
"copyShareLink": "复制分享链接",
|
||||
"chooseWorkflowFromLibrary": "从库中选择工作流程",
|
||||
"deleteWorkflow2": "您确定要删除此工作流程吗?此操作无法撤销。"
|
||||
@@ -1659,7 +1597,6 @@
|
||||
"moveToBack": "移动到后面",
|
||||
"moveToFront": "移动到前面",
|
||||
"addLayer": "添加层",
|
||||
"deletePrompt": "删除提示词",
|
||||
"addPositivePrompt": "添加 $t(controlLayers.prompt)",
|
||||
"addNegativePrompt": "添加 $t(controlLayers.negativePrompt)",
|
||||
"rectangle": "矩形",
|
||||
@@ -1683,7 +1620,6 @@
|
||||
"maskFill": "遮罩填充",
|
||||
"newCanvasFromImage": "从图像创建新画布",
|
||||
"pullBboxIntoReferenceImageOk": "边界框已导入到参考图像",
|
||||
"globalReferenceImage_withCount_other": "全局参考图像",
|
||||
"addInpaintMask": "添加 $t(controlLayers.inpaintMask)",
|
||||
"referenceImage": "参考图像",
|
||||
"globalReferenceImage": "全局参考图像",
|
||||
@@ -1692,14 +1628,10 @@
|
||||
"copyRasterLayerTo": "复制 $t(controlLayers.rasterLayer) 到",
|
||||
"clearHistory": "清除历史记录",
|
||||
"inpaintMask": "修复遮罩",
|
||||
"regionalGuidance_withCount_visible": "区域引导({{count}} 个)",
|
||||
"inpaintMasks_withCount_hidden": "修复遮罩({{count}} 个已隐藏)",
|
||||
"enableAutoNegative": "启用自动负面提示",
|
||||
"disableAutoNegative": "禁用自动负面提示",
|
||||
"deleteReferenceImage": "删除参考图像",
|
||||
"sendToCanvas": "发送到画布",
|
||||
"controlLayers_withCount_visible": "控制图层({{count}} 个)",
|
||||
"rasterLayers_withCount_visible": "栅格图层({{count}} 个)",
|
||||
"convertRegionalGuidanceTo": "将 $t(controlLayers.regionalGuidance) 转换为",
|
||||
"newInpaintMask": "新建 $t(controlLayers.inpaintMask)",
|
||||
"regionIsEmpty": "选定区域为空",
|
||||
@@ -1711,14 +1643,12 @@
|
||||
"addRasterLayer": "添加 $t(controlLayers.rasterLayer)",
|
||||
"newRasterLayerOk": "已创建栅格层",
|
||||
"newRasterLayerError": "创建栅格层时出现问题",
|
||||
"inpaintMasks_withCount_visible": "修复遮罩({{count}} 个)",
|
||||
"convertRasterLayerTo": "将 $t(controlLayers.rasterLayer) 转换为",
|
||||
"copyControlLayerTo": "复制 $t(controlLayers.controlLayer) 到",
|
||||
"copyInpaintMaskTo": "复制 $t(controlLayers.inpaintMask) 到",
|
||||
"copyRegionalGuidanceTo": "复制 $t(controlLayers.regionalGuidance) 到",
|
||||
"newRasterLayer": "新建 $t(controlLayers.rasterLayer)",
|
||||
"newControlLayer": "新建 $t(controlLayers.controlLayer)",
|
||||
"newImg2ImgCanvasFromImage": "从图像创建新的图生图",
|
||||
"rasterLayer": "栅格层",
|
||||
"controlLayer": "控制层",
|
||||
"outputOnlyMaskedRegions": "仅输出生成的区域",
|
||||
@@ -1731,36 +1661,22 @@
|
||||
"bboxOverlay": "显示边界框覆盖层",
|
||||
"clipToBbox": "将Clip限制到边界框",
|
||||
"width": "宽度",
|
||||
"addGlobalReferenceImage": "添加 $t(controlLayers.globalReferenceImage)",
|
||||
"inpaintMask_withCount_other": "修复遮罩",
|
||||
"regionalGuidance_withCount_other": "区域引导",
|
||||
"newRegionalReferenceImageError": "创建局部参考图像时出现问题",
|
||||
"pullBboxIntoLayerError": "将边界框导入图层时出现问题",
|
||||
"pullBboxIntoLayerOk": "边界框已导入到图层",
|
||||
"sendToCanvasDesc": "按下“Invoke”按钮会将您的工作进度暂存到画布上。",
|
||||
"sendToGallery": "发送到图库",
|
||||
"sendToGalleryDesc": "按下“Invoke”键会生成并保存一张唯一的图像到您的图库中。",
|
||||
"rasterLayer_withCount_other": "栅格图层",
|
||||
"mergeDown": "向下合并",
|
||||
"clearCaches": "清除缓存",
|
||||
"recalculateRects": "重新计算矩形",
|
||||
"duplicate": "复制",
|
||||
"regionalGuidance_withCount_hidden": "区域引导({{count}} 个已隐藏)",
|
||||
"convertControlLayerTo": "将 $t(controlLayers.controlLayer) 转换为",
|
||||
"convertInpaintMaskTo": "将 $t(controlLayers.inpaintMask) 转换为",
|
||||
"viewProgressInViewer": "在 <Btn>图像查看器</Btn> 中查看进度和输出结果。",
|
||||
"viewProgressOnCanvas": "在 <Btn>画布</Btn> 上查看进度和暂存的输出内容。",
|
||||
"sendingToGallery": "将生成内容发送到图库",
|
||||
"copyToClipboard": "复制到剪贴板",
|
||||
"controlLayer_withCount_other": "控制图层",
|
||||
"sendingToCanvas": "在画布上准备生成",
|
||||
"addReferenceImage": "添加 $t(controlLayers.referenceImage)",
|
||||
"addRegionalGuidance": "添加 $t(controlLayers.regionalGuidance)",
|
||||
"controlLayers_withCount_hidden": "控制图层({{count}} 个已隐藏)",
|
||||
"rasterLayers_withCount_hidden": "栅格图层({{count}} 个已隐藏)",
|
||||
"globalReferenceImages_withCount_hidden": "全局参考图像({{count}} 个已隐藏)",
|
||||
"globalReferenceImages_withCount_visible": "全局参考图像({{count}} 个)",
|
||||
"layer_withCount_other": "图层({{count}} 个)",
|
||||
"enableTransparencyEffect": "启用透明效果",
|
||||
"disableTransparencyEffect": "禁用透明效果",
|
||||
"hidingType": "隐藏 {{type}}",
|
||||
|
||||
@@ -19,7 +19,6 @@
|
||||
"folder": "資料夾",
|
||||
"installed": "已安裝",
|
||||
"accept": "接受",
|
||||
"goTo": "前往",
|
||||
"input": "輸入",
|
||||
"random": "隨機",
|
||||
"selected": "已選擇",
|
||||
@@ -29,8 +28,7 @@
|
||||
"copy": "複製",
|
||||
"error": "錯誤",
|
||||
"file": "檔案",
|
||||
"format": "格式",
|
||||
"imageFailedToLoad": "無法載入圖片"
|
||||
"format": "格式"
|
||||
},
|
||||
"accessibility": {
|
||||
"invokeProgressBar": "Invoke 進度條",
|
||||
@@ -179,8 +177,7 @@
|
||||
"workflowAuthor": "作者",
|
||||
"version": "版本",
|
||||
"executionStateCompleted": "已完成",
|
||||
"edge": "邊緣",
|
||||
"versionUnknown": " 版本未知"
|
||||
"edge": "邊緣"
|
||||
},
|
||||
"sdxl": {
|
||||
"steps": "步數",
|
||||
|
||||
@@ -2,6 +2,7 @@ import { GlobalImageHotkeys } from 'app/components/GlobalImageHotkeys';
|
||||
import ChangeBoardModal from 'features/changeBoardModal/components/ChangeBoardModal';
|
||||
import { CanvasPasteModal } from 'features/controlLayers/components/CanvasPasteModal';
|
||||
import { CanvasManagerProviderGate } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
||||
import { CropImageModal } from 'features/cropper/components/CropImageModal';
|
||||
import { DeleteImageModal } from 'features/deleteImageModal/components/DeleteImageModal';
|
||||
import { DeleteVideoModal } from 'features/deleteVideoModal/components/DeleteVideoModal';
|
||||
import { FullscreenDropzone } from 'features/dnd/FullscreenDropzone';
|
||||
@@ -58,6 +59,7 @@ export const GlobalModalIsolator = memo(() => {
|
||||
<CanvasPasteModal />
|
||||
</CanvasManagerProviderGate>
|
||||
<LoadWorkflowFromGraphModal />
|
||||
<CropImageModal />
|
||||
</>
|
||||
);
|
||||
});
|
||||
|
||||
@@ -4,7 +4,6 @@ import { useAssertSingleton } from 'common/hooks/useAssertSingleton';
|
||||
import { withResultAsync } from 'common/util/result';
|
||||
import { canvasReset } from 'features/controlLayers/store/actions';
|
||||
import { rasterLayerAdded } from 'features/controlLayers/store/canvasSlice';
|
||||
import { paramsReset } from 'features/controlLayers/store/paramsSlice';
|
||||
import type { CanvasRasterLayerState } from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageObject } from 'features/controlLayers/store/util';
|
||||
import { sentImageToCanvas } from 'features/gallery/store/actions';
|
||||
@@ -164,7 +163,6 @@ export const useStudioInitAction = (action?: StudioInitAction) => {
|
||||
case 'generation':
|
||||
// Go to the generate tab, open the launchpad
|
||||
await navigationApi.focusPanel('generate', LAUNCHPAD_PANEL_ID);
|
||||
store.dispatch(paramsReset());
|
||||
break;
|
||||
case 'canvas':
|
||||
// Go to the canvas tab, open the launchpad
|
||||
|
||||
@@ -12,7 +12,13 @@ import {
|
||||
} from 'features/controlLayers/store/paramsSlice';
|
||||
import { refImageModelChanged, selectRefImagesSlice } from 'features/controlLayers/store/refImagesSlice';
|
||||
import { selectCanvasSlice } from 'features/controlLayers/store/selectors';
|
||||
import { getEntityIdentifier, isFLUXReduxConfig, isIPAdapterConfig } from 'features/controlLayers/store/types';
|
||||
import {
|
||||
getEntityIdentifier,
|
||||
isFLUXReduxConfig,
|
||||
isIPAdapterConfig,
|
||||
isRegionalGuidanceFLUXReduxConfig,
|
||||
isRegionalGuidanceIPAdapterConfig,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import { zModelIdentifierField } from 'features/nodes/types/common';
|
||||
import { modelSelected } from 'features/parameters/store/actions';
|
||||
import {
|
||||
@@ -252,7 +258,7 @@ const handleIPAdapterModels: ModelHandler = (models, state, dispatch, log) => {
|
||||
|
||||
selectCanvasSlice(state).regionalGuidance.entities.forEach((entity) => {
|
||||
entity.referenceImages.forEach(({ id: referenceImageId, config }) => {
|
||||
if (!isIPAdapterConfig(config)) {
|
||||
if (!isRegionalGuidanceIPAdapterConfig(config)) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -295,7 +301,7 @@ const handleFLUXReduxModels: ModelHandler = (models, state, dispatch, log) => {
|
||||
|
||||
selectCanvasSlice(state).regionalGuidance.entities.forEach((entity) => {
|
||||
entity.referenceImages.forEach(({ id: referenceImageId, config }) => {
|
||||
if (!isFLUXReduxConfig(config)) {
|
||||
if (!isRegionalGuidanceFLUXReduxConfig(config)) {
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
@@ -18,7 +18,8 @@ import { addModelsLoadedListener } from 'app/store/middleware/listenerMiddleware
|
||||
import { addSetDefaultSettingsListener } from 'app/store/middleware/listenerMiddleware/listeners/setDefaultSettings';
|
||||
import { addSocketConnectedEventListener } from 'app/store/middleware/listenerMiddleware/listeners/socketConnected';
|
||||
import { deepClone } from 'common/util/deepClone';
|
||||
import { keys, mergeWith, omit, pick } from 'es-toolkit/compat';
|
||||
import { merge } from 'es-toolkit';
|
||||
import { omit, pick } from 'es-toolkit/compat';
|
||||
import { changeBoardModalSliceConfig } from 'features/changeBoardModal/store/slice';
|
||||
import { canvasSettingsSliceConfig } from 'features/controlLayers/store/canvasSettingsSlice';
|
||||
import { canvasSliceConfig } from 'features/controlLayers/store/canvasSlice';
|
||||
@@ -133,16 +134,14 @@ const unserialize: UnserializeFunction = (data, key) => {
|
||||
const initialState = getInitialState();
|
||||
const parsed = JSON.parse(data);
|
||||
|
||||
// strip out old keys
|
||||
const stripped = pick(deepClone(parsed), keys(initialState));
|
||||
/*
|
||||
* Merge in initial state as default values, covering any missing keys. You might be tempted to use _.defaultsDeep,
|
||||
* but that merges arrays by index and partial objects by key. Using an identity function as the customizer results
|
||||
* in behaviour like defaultsDeep, but doesn't overwrite any values that are not undefined in the migrated state.
|
||||
*/
|
||||
const unPersistDenylisted = mergeWith(stripped, initialState, (objVal) => objVal);
|
||||
// run (additive) migrations
|
||||
const migrated = persistConfig.migrate(unPersistDenylisted);
|
||||
// We need to inject non-persisted values from initial state into the rehydrated state. These values always are
|
||||
// required to be in the state, but won't be in the persisted data. Build an object that consists of only these
|
||||
// values, then merge it with the rehydrated state.
|
||||
const nonPersistedSubsetOfState = pick(initialState, persistConfig.persistDenylist ?? []);
|
||||
const stateToMigrate = merge(deepClone(parsed), nonPersistedSubsetOfState);
|
||||
|
||||
// Run migrations to bring old state up to date with the current version.
|
||||
const migrated = persistConfig.migrate(stateToMigrate);
|
||||
|
||||
log.debug(
|
||||
{
|
||||
|
||||
@@ -1,4 +1,13 @@
|
||||
import { Box, Flex, Popover, PopoverBody, PopoverContent, PopoverTrigger, Tooltip } from '@invoke-ai/ui-library';
|
||||
import {
|
||||
Box,
|
||||
Flex,
|
||||
Popover,
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
Portal,
|
||||
Tooltip,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import RgbColorPicker from 'common/components/ColorPicker/RgbColorPicker';
|
||||
import { rgbColorToString } from 'common/util/colorCodeTransformers';
|
||||
@@ -62,14 +71,16 @@ export const EntityListSelectedEntityActionBarFill = memo(() => {
|
||||
</Tooltip>
|
||||
</Flex>
|
||||
</PopoverTrigger>
|
||||
<PopoverContent>
|
||||
<PopoverBody minH={64}>
|
||||
<Flex flexDir="column" gap={4}>
|
||||
<RgbColorPicker color={fill.color} onChange={onChangeFillColor} withNumberInput withSwatches />
|
||||
<MaskFillStyle style={fill.style} onChange={onChangeFillStyle} />
|
||||
</Flex>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
<Portal>
|
||||
<PopoverContent>
|
||||
<PopoverBody minH={64}>
|
||||
<Flex flexDir="column" gap={4}>
|
||||
<RgbColorPicker color={fill.color} onChange={onChangeFillColor} withNumberInput withSwatches />
|
||||
<MaskFillStyle style={fill.style} onChange={onChangeFillStyle} />
|
||||
</Flex>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Portal>
|
||||
</Popover>
|
||||
);
|
||||
});
|
||||
|
||||
@@ -12,6 +12,7 @@ import {
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
Portal,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
@@ -165,22 +166,24 @@ export const EntityListSelectedEntityActionBarOpacity = memo(() => {
|
||||
</NumberInput>
|
||||
</PopoverAnchor>
|
||||
</FormControl>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={localOpacity}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
isDisabled={selectedEntityIdentifier === null}
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
<Portal>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={localOpacity}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
isDisabled={selectedEntityIdentifier === null}
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Portal>
|
||||
</Popover>
|
||||
);
|
||||
});
|
||||
|
||||
@@ -4,6 +4,7 @@ import { CanvasEntityHeader } from 'features/controlLayers/components/common/Can
|
||||
import { CanvasEntityHeaderCommonActions } from 'features/controlLayers/components/common/CanvasEntityHeaderCommonActions';
|
||||
import { CanvasEntityPreviewImage } from 'features/controlLayers/components/common/CanvasEntityPreviewImage';
|
||||
import { CanvasEntityEditableTitle } from 'features/controlLayers/components/common/CanvasEntityTitleEdit';
|
||||
import { RasterLayerAdjustmentsPanel } from 'features/controlLayers/components/RasterLayer/RasterLayerAdjustmentsPanel';
|
||||
import { CanvasEntityStateGate } from 'features/controlLayers/contexts/CanvasEntityStateGate';
|
||||
import { RasterLayerAdapterGate } from 'features/controlLayers/contexts/EntityAdapterContext';
|
||||
import { EntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||
@@ -39,6 +40,7 @@ export const RasterLayer = memo(({ id }: Props) => {
|
||||
<Spacer />
|
||||
<CanvasEntityHeaderCommonActions />
|
||||
</CanvasEntityHeader>
|
||||
<RasterLayerAdjustmentsPanel />
|
||||
<DndDropTarget
|
||||
dndTarget={replaceCanvasEntityObjectsWithImageDndTarget}
|
||||
dndTargetData={dndTargetData}
|
||||
|
||||
@@ -0,0 +1,167 @@
|
||||
import { Button, ButtonGroup, Flex, IconButton, Switch, Text } from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { RasterLayerCurvesAdjustmentsEditor } from 'features/controlLayers/components/RasterLayer/RasterLayerCurvesAdjustmentsEditor';
|
||||
import { RasterLayerSimpleAdjustmentsEditor } from 'features/controlLayers/components/RasterLayer/RasterLayerSimpleAdjustmentsEditor';
|
||||
import { useCanvasManager } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
||||
import { useEntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||
import {
|
||||
rasterLayerAdjustmentsCancel,
|
||||
rasterLayerAdjustmentsCollapsedToggled,
|
||||
rasterLayerAdjustmentsEnabledToggled,
|
||||
rasterLayerAdjustmentsModeChanged,
|
||||
rasterLayerAdjustmentsReset,
|
||||
rasterLayerAdjustmentsSet,
|
||||
} from 'features/controlLayers/store/canvasSlice';
|
||||
import { selectCanvasSlice, selectEntity } from 'features/controlLayers/store/selectors';
|
||||
import React, { memo, useCallback, useMemo } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { PiArrowCounterClockwiseBold, PiCaretDownBold, PiCheckBold, PiTrashBold } from 'react-icons/pi';
|
||||
|
||||
export const RasterLayerAdjustmentsPanel = memo(() => {
|
||||
const { t } = useTranslation();
|
||||
const dispatch = useAppDispatch();
|
||||
const entityIdentifier = useEntityIdentifierContext<'raster_layer'>();
|
||||
const canvasManager = useCanvasManager();
|
||||
|
||||
const selectHasAdjustments = useMemo(() => {
|
||||
return createSelector(selectCanvasSlice, (canvas) => Boolean(selectEntity(canvas, entityIdentifier)?.adjustments));
|
||||
}, [entityIdentifier]);
|
||||
|
||||
const hasAdjustments = useAppSelector(selectHasAdjustments);
|
||||
|
||||
const selectMode = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) => selectEntity(canvas, entityIdentifier)?.adjustments?.mode ?? 'simple'
|
||||
);
|
||||
}, [entityIdentifier]);
|
||||
const mode = useAppSelector(selectMode);
|
||||
|
||||
const selectEnabled = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) => selectEntity(canvas, entityIdentifier)?.adjustments?.enabled ?? false
|
||||
);
|
||||
}, [entityIdentifier]);
|
||||
const enabled = useAppSelector(selectEnabled);
|
||||
|
||||
const selectCollapsed = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) => selectEntity(canvas, entityIdentifier)?.adjustments?.collapsed ?? false
|
||||
);
|
||||
}, [entityIdentifier]);
|
||||
const collapsed = useAppSelector(selectCollapsed);
|
||||
|
||||
const onToggleEnabled = useCallback(() => {
|
||||
dispatch(rasterLayerAdjustmentsEnabledToggled({ entityIdentifier }));
|
||||
}, [dispatch, entityIdentifier]);
|
||||
|
||||
const onReset = useCallback(() => {
|
||||
// Reset values to defaults but keep adjustments present; preserve enabled/collapsed/mode
|
||||
dispatch(rasterLayerAdjustmentsReset({ entityIdentifier }));
|
||||
}, [dispatch, entityIdentifier]);
|
||||
|
||||
const onCancel = useCallback(() => {
|
||||
// Clear out adjustments entirely
|
||||
dispatch(rasterLayerAdjustmentsCancel({ entityIdentifier }));
|
||||
}, [dispatch, entityIdentifier]);
|
||||
|
||||
const onToggleCollapsed = useCallback(() => {
|
||||
dispatch(rasterLayerAdjustmentsCollapsedToggled({ entityIdentifier }));
|
||||
}, [dispatch, entityIdentifier]);
|
||||
|
||||
const onClickModeSimple = useCallback(
|
||||
() => dispatch(rasterLayerAdjustmentsModeChanged({ entityIdentifier, mode: 'simple' })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
|
||||
const onClickModeCurves = useCallback(
|
||||
() => dispatch(rasterLayerAdjustmentsModeChanged({ entityIdentifier, mode: 'curves' })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
|
||||
const onFinish = useCallback(async () => {
|
||||
// Bake current visual into layer pixels, then clear adjustments
|
||||
const adapter = canvasManager.getAdapter(entityIdentifier);
|
||||
if (!adapter || adapter.type !== 'raster_layer_adapter') {
|
||||
return;
|
||||
}
|
||||
const rect = adapter.transformer.getRelativeRect();
|
||||
try {
|
||||
await adapter.renderer.rasterize({ rect, replaceObjects: true });
|
||||
// Clear adjustments after baking
|
||||
dispatch(rasterLayerAdjustmentsSet({ entityIdentifier, adjustments: null }));
|
||||
} catch {
|
||||
// no-op; leave state unchanged on failure
|
||||
}
|
||||
}, [canvasManager, entityIdentifier, dispatch]);
|
||||
|
||||
// Hide the panel entirely until adjustments are added via context menu
|
||||
if (!hasAdjustments) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<>
|
||||
<Flex px={2} pb={2} alignItems="center" gap={2}>
|
||||
<IconButton
|
||||
aria-label={collapsed ? t('controlLayers.adjustments.expand') : t('controlLayers.adjustments.collapse')}
|
||||
size="sm"
|
||||
variant="ghost"
|
||||
onClick={onToggleCollapsed}
|
||||
icon={
|
||||
<PiCaretDownBold
|
||||
style={{ transform: collapsed ? 'rotate(-90deg)' : 'rotate(0deg)', transition: 'transform 0.2s' }}
|
||||
/>
|
||||
}
|
||||
/>
|
||||
<Text fontWeight={600} flex={1}>
|
||||
Adjustments
|
||||
</Text>
|
||||
<ButtonGroup size="sm" isAttached variant="outline">
|
||||
<Button onClick={onClickModeSimple} colorScheme={mode === 'simple' ? 'invokeBlue' : undefined}>
|
||||
{t('controlLayers.adjustments.simple')}
|
||||
</Button>
|
||||
<Button onClick={onClickModeCurves} colorScheme={mode === 'curves' ? 'invokeBlue' : undefined}>
|
||||
{t('controlLayers.adjustments.curves')}
|
||||
</Button>
|
||||
</ButtonGroup>
|
||||
<Switch isChecked={enabled} onChange={onToggleEnabled} />
|
||||
<IconButton
|
||||
aria-label={t('controlLayers.adjustments.cancel')}
|
||||
size="md"
|
||||
onClick={onCancel}
|
||||
isDisabled={!hasAdjustments}
|
||||
colorScheme="red"
|
||||
icon={<PiTrashBold />}
|
||||
variant="ghost"
|
||||
/>
|
||||
<IconButton
|
||||
aria-label={t('controlLayers.adjustments.reset')}
|
||||
size="md"
|
||||
onClick={onReset}
|
||||
isDisabled={!hasAdjustments}
|
||||
icon={<PiArrowCounterClockwiseBold />}
|
||||
variant="ghost"
|
||||
/>
|
||||
<IconButton
|
||||
aria-label={t('controlLayers.adjustments.finish')}
|
||||
size="md"
|
||||
onClick={onFinish}
|
||||
isDisabled={!hasAdjustments}
|
||||
colorScheme="green"
|
||||
icon={<PiCheckBold />}
|
||||
variant="ghost"
|
||||
/>
|
||||
</Flex>
|
||||
|
||||
{!collapsed && mode === 'simple' && <RasterLayerSimpleAdjustmentsEditor />}
|
||||
|
||||
{!collapsed && mode === 'curves' && <RasterLayerCurvesAdjustmentsEditor />}
|
||||
</>
|
||||
);
|
||||
});
|
||||
|
||||
RasterLayerAdjustmentsPanel.displayName = 'RasterLayerAdjustmentsPanel';
|
||||
@@ -0,0 +1,179 @@
|
||||
import { Box, Flex } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { useEntityAdapterContext } from 'features/controlLayers/contexts/EntityAdapterContext';
|
||||
import { useEntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||
import { rasterLayerAdjustmentsCurvesUpdated } from 'features/controlLayers/store/canvasSlice';
|
||||
import { selectCanvasSlice, selectEntity } from 'features/controlLayers/store/selectors';
|
||||
import type { ChannelName, ChannelPoints, CurvesAdjustmentsConfig } from 'features/controlLayers/store/types';
|
||||
import { memo, useCallback, useEffect, useMemo, useState } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
import { RasterLayerCurvesAdjustmentsGraph } from './RasterLayerCurvesAdjustmentsGraph';
|
||||
|
||||
const DEFAULT_POINTS: ChannelPoints = [
|
||||
[0, 0],
|
||||
[255, 255],
|
||||
];
|
||||
|
||||
const DEFAULT_CURVES: CurvesAdjustmentsConfig = {
|
||||
master: DEFAULT_POINTS,
|
||||
r: DEFAULT_POINTS,
|
||||
g: DEFAULT_POINTS,
|
||||
b: DEFAULT_POINTS,
|
||||
};
|
||||
|
||||
type ChannelHistograms = Record<ChannelName, number[] | null>;
|
||||
|
||||
const calculateHistogramsFromImageData = (imageData: ImageData): ChannelHistograms | null => {
|
||||
try {
|
||||
const data = imageData.data;
|
||||
const len = data.length / 4;
|
||||
const master = new Array<number>(256).fill(0);
|
||||
const r = new Array<number>(256).fill(0);
|
||||
const g = new Array<number>(256).fill(0);
|
||||
const b = new Array<number>(256).fill(0);
|
||||
// sample every 4th pixel to lighten work
|
||||
for (let i = 0; i < len; i += 4) {
|
||||
const idx = i * 4;
|
||||
const rv = data[idx] as number;
|
||||
const gv = data[idx + 1] as number;
|
||||
const bv = data[idx + 2] as number;
|
||||
const m = Math.round(0.2126 * rv + 0.7152 * gv + 0.0722 * bv);
|
||||
if (m >= 0 && m < 256) {
|
||||
master[m] = (master[m] ?? 0) + 1;
|
||||
}
|
||||
if (rv >= 0 && rv < 256) {
|
||||
r[rv] = (r[rv] ?? 0) + 1;
|
||||
}
|
||||
if (gv >= 0 && gv < 256) {
|
||||
g[gv] = (g[gv] ?? 0) + 1;
|
||||
}
|
||||
if (bv >= 0 && bv < 256) {
|
||||
b[bv] = (b[bv] ?? 0) + 1;
|
||||
}
|
||||
}
|
||||
return {
|
||||
master,
|
||||
r,
|
||||
g,
|
||||
b,
|
||||
};
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
export const RasterLayerCurvesAdjustmentsEditor = memo(() => {
|
||||
const dispatch = useAppDispatch();
|
||||
const entityIdentifier = useEntityIdentifierContext<'raster_layer'>();
|
||||
const adapter = useEntityAdapterContext<'raster_layer'>('raster_layer');
|
||||
const { t } = useTranslation();
|
||||
const selectCurves = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) => selectEntity(canvas, entityIdentifier)?.adjustments?.curves ?? DEFAULT_CURVES
|
||||
);
|
||||
}, [entityIdentifier]);
|
||||
const curves = useAppSelector(selectCurves);
|
||||
|
||||
const selectIsDisabled = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) => selectEntity(canvas, entityIdentifier)?.adjustments?.enabled !== true
|
||||
);
|
||||
}, [entityIdentifier]);
|
||||
const isDisabled = useAppSelector(selectIsDisabled);
|
||||
// The canvas cache for the layer serves as a proxy for when the layer changes and can be used to trigger histo recalc
|
||||
const canvasCache = useStore(adapter.$canvasCache);
|
||||
|
||||
const [histMaster, setHistMaster] = useState<number[] | null>(null);
|
||||
const [histR, setHistR] = useState<number[] | null>(null);
|
||||
const [histG, setHistG] = useState<number[] | null>(null);
|
||||
const [histB, setHistB] = useState<number[] | null>(null);
|
||||
|
||||
const recalcHistogram = useCallback(() => {
|
||||
try {
|
||||
const rect = adapter.transformer.getRelativeRect();
|
||||
if (rect.width === 0 || rect.height === 0) {
|
||||
setHistMaster(Array(256).fill(0));
|
||||
setHistR(Array(256).fill(0));
|
||||
setHistG(Array(256).fill(0));
|
||||
setHistB(Array(256).fill(0));
|
||||
return;
|
||||
}
|
||||
const imageData = adapter.renderer.getImageData({ rect });
|
||||
const h = calculateHistogramsFromImageData(imageData);
|
||||
if (h) {
|
||||
setHistMaster(h.master);
|
||||
setHistR(h.r);
|
||||
setHistG(h.g);
|
||||
setHistB(h.b);
|
||||
}
|
||||
} catch {
|
||||
// ignore
|
||||
}
|
||||
}, [adapter]);
|
||||
|
||||
useEffect(() => {
|
||||
recalcHistogram();
|
||||
}, [canvasCache, recalcHistogram]);
|
||||
|
||||
const onChangePoints = useCallback(
|
||||
(channel: ChannelName, pts: ChannelPoints) => {
|
||||
dispatch(rasterLayerAdjustmentsCurvesUpdated({ entityIdentifier, channel, points: pts }));
|
||||
},
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
|
||||
// Memoize per-channel change handlers to avoid inline lambdas in JSX
|
||||
const onChangeMaster = useCallback((pts: ChannelPoints) => onChangePoints('master', pts), [onChangePoints]);
|
||||
const onChangeR = useCallback((pts: ChannelPoints) => onChangePoints('r', pts), [onChangePoints]);
|
||||
const onChangeG = useCallback((pts: ChannelPoints) => onChangePoints('g', pts), [onChangePoints]);
|
||||
const onChangeB = useCallback((pts: ChannelPoints) => onChangePoints('b', pts), [onChangePoints]);
|
||||
|
||||
return (
|
||||
<Flex
|
||||
direction="column"
|
||||
gap={2}
|
||||
px={3}
|
||||
pb={3}
|
||||
opacity={isDisabled ? 0.3 : 1}
|
||||
pointerEvents={isDisabled ? 'none' : 'auto'}
|
||||
>
|
||||
<Box display="grid" gridTemplateColumns="repeat(2, minmax(0, 1fr))" gap={4}>
|
||||
<RasterLayerCurvesAdjustmentsGraph
|
||||
title={t('controlLayers.adjustments.master')}
|
||||
channel="master"
|
||||
points={curves.master}
|
||||
histogram={histMaster}
|
||||
onChange={onChangeMaster}
|
||||
/>
|
||||
<RasterLayerCurvesAdjustmentsGraph
|
||||
title={t('common.red')}
|
||||
channel="r"
|
||||
points={curves.r}
|
||||
histogram={histR}
|
||||
onChange={onChangeR}
|
||||
/>
|
||||
<RasterLayerCurvesAdjustmentsGraph
|
||||
title={t('common.green')}
|
||||
channel="g"
|
||||
points={curves.g}
|
||||
histogram={histG}
|
||||
onChange={onChangeG}
|
||||
/>
|
||||
<RasterLayerCurvesAdjustmentsGraph
|
||||
title={t('common.blue')}
|
||||
channel="b"
|
||||
points={curves.b}
|
||||
histogram={histB}
|
||||
onChange={onChangeB}
|
||||
/>
|
||||
</Box>
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
RasterLayerCurvesAdjustmentsEditor.displayName = 'RasterLayerCurvesEditor';
|
||||
@@ -0,0 +1,432 @@
|
||||
import { Flex, IconButton, Text } from '@invoke-ai/ui-library';
|
||||
import type { ChannelName, ChannelPoints } from 'features/controlLayers/store/types';
|
||||
import React, { memo, useCallback, useEffect, useRef, useState } from 'react';
|
||||
import { PiArrowCounterClockwiseBold } from 'react-icons/pi';
|
||||
|
||||
const DEFAULT_POINTS: ChannelPoints = [
|
||||
[0, 0],
|
||||
[255, 255],
|
||||
];
|
||||
|
||||
const channelColor: Record<ChannelName, string> = {
|
||||
master: '#888',
|
||||
r: '#e53e3e',
|
||||
g: '#38a169',
|
||||
b: '#3182ce',
|
||||
};
|
||||
|
||||
const clamp = (v: number, min: number, max: number) => (v < min ? min : v > max ? max : v);
|
||||
|
||||
const sortPoints = (pts: ChannelPoints) =>
|
||||
[...pts]
|
||||
.sort((a, b) => {
|
||||
const xDiff = a[0] - b[0];
|
||||
if (xDiff) {
|
||||
return xDiff;
|
||||
}
|
||||
if (a[0] === 0 || a[0] === 255) {
|
||||
return a[1] - b[1];
|
||||
}
|
||||
return 0;
|
||||
})
|
||||
// Finally, clamp to valid range and round to integers
|
||||
.map(([x, y]) => [clamp(Math.round(x), 0, 255), clamp(Math.round(y), 0, 255)] satisfies [number, number]);
|
||||
|
||||
// Base canvas logical coordinate system (used for aspect ratio & initial sizing)
|
||||
const CANVAS_WIDTH = 256;
|
||||
const CANVAS_HEIGHT = 160;
|
||||
const MARGIN_LEFT = 8;
|
||||
const MARGIN_RIGHT = 8;
|
||||
const MARGIN_TOP = 8;
|
||||
const MARGIN_BOTTOM = 10;
|
||||
|
||||
const CANVAS_STYLE: React.CSSProperties = {
|
||||
width: '100%',
|
||||
// Maintain aspect ratio while allowing responsive width. Height is set automatically via aspect-ratio.
|
||||
aspectRatio: `${CANVAS_WIDTH} / ${CANVAS_HEIGHT}`,
|
||||
height: 'auto',
|
||||
touchAction: 'none',
|
||||
borderRadius: 4,
|
||||
background: '#111',
|
||||
display: 'block',
|
||||
};
|
||||
|
||||
type CurveGraphProps = {
|
||||
title: string;
|
||||
channel: ChannelName;
|
||||
points: ChannelPoints | undefined;
|
||||
histogram: number[] | null;
|
||||
onChange: (pts: ChannelPoints) => void;
|
||||
};
|
||||
|
||||
const drawHistogram = (
|
||||
c: HTMLCanvasElement,
|
||||
channel: ChannelName,
|
||||
histogram: number[] | null,
|
||||
points: ChannelPoints
|
||||
) => {
|
||||
// Use device pixel ratio for crisp rendering on HiDPI displays.
|
||||
const dpr = window.devicePixelRatio || 1;
|
||||
const cssWidth = c.clientWidth || CANVAS_WIDTH; // CSS pixels
|
||||
const cssHeight = (cssWidth * CANVAS_HEIGHT) / CANVAS_WIDTH; // maintain aspect ratio
|
||||
|
||||
// Ensure the backing store matches current display size * dpr (only if changed).
|
||||
const targetWidth = Math.round(cssWidth * dpr);
|
||||
const targetHeight = Math.round(cssHeight * dpr);
|
||||
if (c.width !== targetWidth || c.height !== targetHeight) {
|
||||
c.width = targetWidth;
|
||||
c.height = targetHeight;
|
||||
}
|
||||
// Guarantee the CSS height stays synced (width is 100%).
|
||||
if (c.style.height !== `${cssHeight}px`) {
|
||||
c.style.height = `${cssHeight}px`;
|
||||
}
|
||||
|
||||
const ctx = c.getContext('2d');
|
||||
if (!ctx) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Reset transform then scale for dpr so we can draw in CSS pixel coordinates.
|
||||
ctx.setTransform(1, 0, 0, 1, 0, 0);
|
||||
ctx.scale(dpr, dpr);
|
||||
|
||||
// Dynamic inner geometry (CSS pixel space)
|
||||
const innerWidth = cssWidth - MARGIN_LEFT - MARGIN_RIGHT;
|
||||
const innerHeight = cssHeight - MARGIN_TOP - MARGIN_BOTTOM;
|
||||
|
||||
const valueToCanvasX = (x: number) => MARGIN_LEFT + (clamp(x, 0, 255) / 255) * innerWidth;
|
||||
const valueToCanvasY = (y: number) => MARGIN_TOP + innerHeight - (clamp(y, 0, 255) / 255) * innerHeight;
|
||||
|
||||
// Clear & background
|
||||
ctx.clearRect(0, 0, cssWidth, cssHeight);
|
||||
ctx.fillStyle = '#111';
|
||||
ctx.fillRect(0, 0, cssWidth, cssHeight);
|
||||
|
||||
// Grid
|
||||
ctx.strokeStyle = '#2a2a2a';
|
||||
ctx.lineWidth = 1;
|
||||
for (let i = 0; i <= 4; i++) {
|
||||
const y = MARGIN_TOP + (i * innerHeight) / 4;
|
||||
ctx.beginPath();
|
||||
ctx.moveTo(MARGIN_LEFT + 0.5, y + 0.5);
|
||||
ctx.lineTo(MARGIN_LEFT + innerWidth - 0.5, y + 0.5);
|
||||
ctx.stroke();
|
||||
}
|
||||
for (let i = 0; i <= 4; i++) {
|
||||
const x = MARGIN_LEFT + (i * innerWidth) / 4;
|
||||
ctx.beginPath();
|
||||
ctx.moveTo(x + 0.5, MARGIN_TOP + 0.5);
|
||||
ctx.lineTo(x + 0.5, MARGIN_TOP + innerHeight - 0.5);
|
||||
ctx.stroke();
|
||||
}
|
||||
|
||||
// Histogram
|
||||
if (histogram) {
|
||||
const logHist = histogram.map((v) => Math.log10((v ?? 0) + 1));
|
||||
const max = Math.max(1e-6, ...logHist);
|
||||
ctx.fillStyle = '#5557';
|
||||
|
||||
// If there's enough horizontal room, draw each of the 256 bins with exact (possibly fractional) width so they tessellate.
|
||||
// Otherwise, aggregate multiple bins into per-pixel columns to avoid aliasing.
|
||||
if (innerWidth >= 256) {
|
||||
for (let i = 0; i < 256; i++) {
|
||||
const v = logHist[i] ?? 0;
|
||||
const h = (v / max) * (innerHeight - 2);
|
||||
// Exact fractional coordinates for seamless coverage (no gaps as width grows)
|
||||
const x0 = MARGIN_LEFT + (i / 256) * innerWidth;
|
||||
const x1 = MARGIN_LEFT + ((i + 1) / 256) * innerWidth;
|
||||
const w = x1 - x0;
|
||||
if (w <= 0) {
|
||||
continue;
|
||||
} // safety
|
||||
const y = MARGIN_TOP + innerHeight - h;
|
||||
ctx.fillRect(x0, y, w, h);
|
||||
}
|
||||
} else {
|
||||
// Aggregate bins per CSS pixel column (similar to previous anti-moire approach)
|
||||
const columns = Math.max(1, Math.round(innerWidth));
|
||||
const binsPerCol = 256 / columns;
|
||||
for (let col = 0; col < columns; col++) {
|
||||
const startBin = Math.floor(col * binsPerCol);
|
||||
const endBin = Math.min(255, Math.floor((col + 1) * binsPerCol - 1));
|
||||
let acc = 0;
|
||||
let count = 0;
|
||||
for (let b = startBin; b <= endBin; b++) {
|
||||
acc += logHist[b] ?? 0;
|
||||
count++;
|
||||
}
|
||||
const v = count > 0 ? acc / count : 0;
|
||||
const h = (v / max) * (innerHeight - 2);
|
||||
const x = MARGIN_LEFT + col;
|
||||
const y = MARGIN_TOP + innerHeight - h;
|
||||
ctx.fillRect(x, y, 1, h);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Curve
|
||||
const pts = sortPoints(points);
|
||||
ctx.strokeStyle = channelColor[channel];
|
||||
ctx.lineWidth = 2;
|
||||
ctx.beginPath();
|
||||
for (let i = 0; i < pts.length; i++) {
|
||||
const [x, y] = pts[i]!;
|
||||
const cx = valueToCanvasX(x);
|
||||
const cy = valueToCanvasY(y);
|
||||
if (i === 0) {
|
||||
ctx.moveTo(cx, cy);
|
||||
} else {
|
||||
ctx.lineTo(cx, cy);
|
||||
}
|
||||
}
|
||||
ctx.stroke();
|
||||
|
||||
// Control points
|
||||
for (let i = 0; i < pts.length; i++) {
|
||||
const [x, y] = pts[i]!;
|
||||
const cx = valueToCanvasX(x);
|
||||
const cy = valueToCanvasY(y);
|
||||
ctx.fillStyle = '#000';
|
||||
ctx.beginPath();
|
||||
ctx.arc(cx, cy, 3.5, 0, Math.PI * 2);
|
||||
ctx.fill();
|
||||
ctx.strokeStyle = channelColor[channel];
|
||||
ctx.lineWidth = 1.5;
|
||||
ctx.stroke();
|
||||
}
|
||||
};
|
||||
|
||||
const getNearestPointIndex = (c: HTMLCanvasElement, points: ChannelPoints, mx: number, my: number) => {
|
||||
const cssWidth = c.clientWidth || CANVAS_WIDTH;
|
||||
const cssHeight = c.clientHeight || CANVAS_HEIGHT;
|
||||
const innerWidth = cssWidth - MARGIN_LEFT - MARGIN_RIGHT;
|
||||
const innerHeight = cssHeight - MARGIN_TOP - MARGIN_BOTTOM;
|
||||
const canvasToValueX = (cx: number) => clamp(Math.round(((cx - MARGIN_LEFT) / innerWidth) * 255), 0, 255);
|
||||
const canvasToValueY = (cy: number) => clamp(Math.round(255 - ((cy - MARGIN_TOP) / innerHeight) * 255), 0, 255);
|
||||
const xVal = canvasToValueX(mx);
|
||||
const yVal = canvasToValueY(my);
|
||||
let best = -1;
|
||||
let bestDist = 9999;
|
||||
for (let i = 0; i < points.length; i++) {
|
||||
const [px, py] = points[i]!;
|
||||
const dx = px - xVal;
|
||||
const dy = py - yVal;
|
||||
const d = dx * dx + dy * dy;
|
||||
if (d < bestDist) {
|
||||
best = i;
|
||||
bestDist = d;
|
||||
}
|
||||
}
|
||||
if (best !== -1 && bestDist <= 20 * 20) {
|
||||
return best;
|
||||
}
|
||||
return -1;
|
||||
};
|
||||
|
||||
const canvasXToValueX = (c: HTMLCanvasElement, cx: number): number => {
|
||||
const cssWidth = c.clientWidth || CANVAS_WIDTH;
|
||||
const innerWidth = cssWidth - MARGIN_LEFT - MARGIN_RIGHT;
|
||||
return clamp(Math.round(((cx - MARGIN_LEFT) / innerWidth) * 255), 0, 255);
|
||||
};
|
||||
|
||||
const canvasYToValueY = (c: HTMLCanvasElement, cy: number) => {
|
||||
const cssHeight = c.clientHeight || CANVAS_HEIGHT;
|
||||
const innerHeight = cssHeight - MARGIN_TOP - MARGIN_BOTTOM;
|
||||
return clamp(Math.round(255 - ((cy - MARGIN_TOP) / innerHeight) * 255), 0, 255);
|
||||
};
|
||||
|
||||
export const RasterLayerCurvesAdjustmentsGraph = memo((props: CurveGraphProps) => {
|
||||
const { title, channel, points, histogram, onChange } = props;
|
||||
const canvasRef = useRef<HTMLCanvasElement | null>(null);
|
||||
const [localPoints, setLocalPoints] = useState<ChannelPoints>(sortPoints(points ?? DEFAULT_POINTS));
|
||||
const [dragIndex, setDragIndex] = useState<number | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
setLocalPoints(sortPoints(points ?? DEFAULT_POINTS));
|
||||
}, [points]);
|
||||
|
||||
useEffect(() => {
|
||||
const c = canvasRef.current;
|
||||
if (!c) {
|
||||
return;
|
||||
}
|
||||
drawHistogram(c, channel, histogram, localPoints);
|
||||
}, [channel, histogram, localPoints]);
|
||||
|
||||
const handlePointerDown = useCallback(
|
||||
(e: React.PointerEvent<HTMLCanvasElement>) => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
const c = canvasRef.current;
|
||||
if (!c) {
|
||||
return;
|
||||
}
|
||||
// Capture the pointer so we still get pointerup even if released outside the canvas.
|
||||
try {
|
||||
c.setPointerCapture(e.pointerId);
|
||||
} catch {
|
||||
/* ignore */
|
||||
}
|
||||
const rect = c.getBoundingClientRect();
|
||||
const mx = e.clientX - rect.left; // CSS pixel coordinates
|
||||
const my = e.clientY - rect.top;
|
||||
const idx = getNearestPointIndex(c, localPoints, mx, my);
|
||||
if (idx !== -1 && idx !== 0 && idx !== localPoints.length - 1) {
|
||||
setDragIndex(idx);
|
||||
return;
|
||||
}
|
||||
const xVal = canvasXToValueX(c, mx);
|
||||
const yVal = canvasYToValueY(c, my);
|
||||
const next = sortPoints([...localPoints, [xVal, yVal]]);
|
||||
setLocalPoints(next);
|
||||
setDragIndex(next.findIndex(([x, y]) => x === xVal && y === yVal));
|
||||
},
|
||||
[localPoints]
|
||||
);
|
||||
|
||||
const handlePointerMove = useCallback(
|
||||
(e: React.PointerEvent<HTMLCanvasElement>) => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
if (dragIndex === null) {
|
||||
return;
|
||||
}
|
||||
const c = canvasRef.current;
|
||||
if (!c) {
|
||||
return;
|
||||
}
|
||||
const rect = c.getBoundingClientRect();
|
||||
const mx = e.clientX - rect.left;
|
||||
const my = e.clientY - rect.top;
|
||||
const mxVal = canvasXToValueX(c, mx);
|
||||
const myVal = canvasYToValueY(c, my);
|
||||
setLocalPoints((prev) => {
|
||||
// Endpoints are immutable; safety check.
|
||||
if (dragIndex === 0 || dragIndex === prev.length - 1) {
|
||||
return prev;
|
||||
}
|
||||
const leftX = prev[dragIndex - 1]![0];
|
||||
const rightX = prev[dragIndex + 1]![0];
|
||||
// Constrain to strictly between neighbors so ordering is preserved & no crossing.
|
||||
const minX = Math.min(254, leftX);
|
||||
const maxX = Math.max(1, rightX);
|
||||
const clampedX = clamp(mxVal, minX, maxX);
|
||||
// If neighbors are adjacent (minX > maxX after adjustments), effectively lock X.
|
||||
const finalX = minX > maxX ? leftX + 1 - 1 /* keep existing */ : clampedX;
|
||||
const next = [...prev];
|
||||
next[dragIndex] = [finalX, myVal];
|
||||
return next; // already ordered due to constraints
|
||||
});
|
||||
},
|
||||
[dragIndex]
|
||||
);
|
||||
|
||||
const commit = useCallback(
|
||||
(pts: ChannelPoints) => {
|
||||
onChange(sortPoints(pts));
|
||||
},
|
||||
[onChange]
|
||||
);
|
||||
|
||||
const handlePointerUp = useCallback(
|
||||
(e: React.PointerEvent<HTMLCanvasElement>) => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
const c = canvasRef.current;
|
||||
if (c) {
|
||||
try {
|
||||
c.releasePointerCapture(e.pointerId);
|
||||
} catch {
|
||||
/* ignore */
|
||||
}
|
||||
}
|
||||
setDragIndex(null);
|
||||
commit(localPoints);
|
||||
},
|
||||
[commit, localPoints]
|
||||
);
|
||||
|
||||
const handlePointerCancel = useCallback(
|
||||
(e: React.PointerEvent<HTMLCanvasElement>) => {
|
||||
const c = canvasRef.current;
|
||||
if (c) {
|
||||
try {
|
||||
c.releasePointerCapture(e.pointerId);
|
||||
} catch {
|
||||
/* ignore */
|
||||
}
|
||||
}
|
||||
setDragIndex(null);
|
||||
commit(localPoints);
|
||||
},
|
||||
[commit, localPoints]
|
||||
);
|
||||
|
||||
const handleDoubleClick = useCallback(
|
||||
(e: React.MouseEvent<HTMLCanvasElement>) => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
const c = canvasRef.current;
|
||||
if (!c) {
|
||||
return;
|
||||
}
|
||||
const rect = c.getBoundingClientRect();
|
||||
const mx = e.clientX - rect.left;
|
||||
const my = e.clientY - rect.top;
|
||||
const idx = getNearestPointIndex(c, localPoints, mx, my);
|
||||
if (idx > 0 && idx < localPoints.length - 1) {
|
||||
const next = localPoints.filter((_, i) => i !== idx);
|
||||
setLocalPoints(next);
|
||||
commit(next);
|
||||
}
|
||||
},
|
||||
[commit, localPoints]
|
||||
);
|
||||
|
||||
// Observe size changes to redraw (responsive behavior)
|
||||
useEffect(() => {
|
||||
const c = canvasRef.current;
|
||||
if (!c) {
|
||||
return;
|
||||
}
|
||||
const ro = new ResizeObserver(() => {
|
||||
drawHistogram(c, channel, histogram, localPoints);
|
||||
});
|
||||
ro.observe(c);
|
||||
return () => ro.disconnect();
|
||||
}, [channel, histogram, localPoints]);
|
||||
|
||||
const resetPoints = useCallback(() => {
|
||||
setLocalPoints(sortPoints(DEFAULT_POINTS));
|
||||
commit(DEFAULT_POINTS);
|
||||
}, [commit]);
|
||||
|
||||
return (
|
||||
<Flex flexDir="column" gap={2}>
|
||||
<Flex justifyContent="space-between">
|
||||
<Text fontSize="sm" color={channelColor[channel]} fontWeight="semibold">
|
||||
{title}
|
||||
</Text>
|
||||
<IconButton
|
||||
icon={<PiArrowCounterClockwiseBold />}
|
||||
aria-label="Reset"
|
||||
size="sm"
|
||||
variant="link"
|
||||
onClick={resetPoints}
|
||||
/>
|
||||
</Flex>
|
||||
<canvas
|
||||
ref={canvasRef}
|
||||
onPointerDown={handlePointerDown}
|
||||
onPointerMove={handlePointerMove}
|
||||
onPointerUp={handlePointerUp}
|
||||
onPointerCancel={handlePointerCancel}
|
||||
onDoubleClick={handleDoubleClick}
|
||||
style={CANVAS_STYLE}
|
||||
/>
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
RasterLayerCurvesAdjustmentsGraph.displayName = 'RasterLayerCurvesAdjustmentsGraph';
|
||||
@@ -9,6 +9,7 @@ import { CanvasEntityMenuItemsMergeDown } from 'features/controlLayers/component
|
||||
import { CanvasEntityMenuItemsSave } from 'features/controlLayers/components/common/CanvasEntityMenuItemsSave';
|
||||
import { CanvasEntityMenuItemsSelectObject } from 'features/controlLayers/components/common/CanvasEntityMenuItemsSelectObject';
|
||||
import { CanvasEntityMenuItemsTransform } from 'features/controlLayers/components/common/CanvasEntityMenuItemsTransform';
|
||||
import { RasterLayerMenuItemsAdjustments } from 'features/controlLayers/components/RasterLayer/RasterLayerMenuItemsAdjustments';
|
||||
import { RasterLayerMenuItemsConvertToSubMenu } from 'features/controlLayers/components/RasterLayer/RasterLayerMenuItemsConvertToSubMenu';
|
||||
import { RasterLayerMenuItemsCopyToSubMenu } from 'features/controlLayers/components/RasterLayer/RasterLayerMenuItemsCopyToSubMenu';
|
||||
import { memo } from 'react';
|
||||
@@ -21,10 +22,10 @@ export const RasterLayerMenuItems = memo(() => {
|
||||
<CanvasEntityMenuItemsDuplicate />
|
||||
<CanvasEntityMenuItemsDelete asIcon />
|
||||
</IconMenuItemGroup>
|
||||
<MenuDivider />
|
||||
<CanvasEntityMenuItemsTransform />
|
||||
<CanvasEntityMenuItemsFilter />
|
||||
<CanvasEntityMenuItemsSelectObject />
|
||||
<RasterLayerMenuItemsAdjustments />
|
||||
<MenuDivider />
|
||||
<CanvasEntityMenuItemsMergeDown />
|
||||
<RasterLayerMenuItemsCopyToSubMenu />
|
||||
|
||||
@@ -0,0 +1,39 @@
|
||||
import { MenuItem } from '@invoke-ai/ui-library';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { useEntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||
import { rasterLayerAdjustmentsCancel, rasterLayerAdjustmentsSet } from 'features/controlLayers/store/canvasSlice';
|
||||
import type { CanvasRasterLayerState } from 'features/controlLayers/store/types';
|
||||
import { makeDefaultRasterLayerAdjustments } from 'features/controlLayers/store/util';
|
||||
import { memo, useCallback } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { PiSlidersHorizontalBold } from 'react-icons/pi';
|
||||
|
||||
export const RasterLayerMenuItemsAdjustments = memo(() => {
|
||||
const dispatch = useAppDispatch();
|
||||
const entityIdentifier = useEntityIdentifierContext<'raster_layer'>();
|
||||
const { t } = useTranslation();
|
||||
const layer = useAppSelector((s) =>
|
||||
s.canvas.present.rasterLayers.entities.find((e: CanvasRasterLayerState) => e.id === entityIdentifier.id)
|
||||
);
|
||||
const hasAdjustments = Boolean(layer?.adjustments);
|
||||
const onToggleAdjustmentsPresence = useCallback(() => {
|
||||
if (hasAdjustments) {
|
||||
dispatch(rasterLayerAdjustmentsCancel({ entityIdentifier }));
|
||||
} else {
|
||||
dispatch(
|
||||
rasterLayerAdjustmentsSet({
|
||||
entityIdentifier,
|
||||
adjustments: makeDefaultRasterLayerAdjustments('simple'),
|
||||
})
|
||||
);
|
||||
}
|
||||
}, [dispatch, entityIdentifier, hasAdjustments]);
|
||||
|
||||
return (
|
||||
<MenuItem onClick={onToggleAdjustmentsPresence} icon={<PiSlidersHorizontalBold />}>
|
||||
{hasAdjustments ? t('controlLayers.removeAdjustments') : t('controlLayers.addAdjustments')}
|
||||
</MenuItem>
|
||||
);
|
||||
});
|
||||
|
||||
RasterLayerMenuItemsAdjustments.displayName = 'RasterLayerMenuItemsAdjustments';
|
||||
@@ -0,0 +1,118 @@
|
||||
import { CompositeNumberInput, CompositeSlider, Flex, FormControl, FormLabel } from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { useEntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||
import { rasterLayerAdjustmentsSimpleUpdated } from 'features/controlLayers/store/canvasSlice';
|
||||
import { selectCanvasSlice, selectEntity } from 'features/controlLayers/store/selectors';
|
||||
import type { SimpleAdjustmentsConfig } from 'features/controlLayers/store/types';
|
||||
import React, { memo, useCallback, useMemo } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
type AdjustmentSliderRowProps = {
|
||||
label: string;
|
||||
name: keyof SimpleAdjustmentsConfig;
|
||||
onChange: (v: number) => void;
|
||||
min?: number;
|
||||
max?: number;
|
||||
step?: number;
|
||||
};
|
||||
|
||||
const AdjustmentSliderRow = ({ label, name, onChange, min = -1, max = 1, step = 0.01 }: AdjustmentSliderRowProps) => {
|
||||
const entityIdentifier = useEntityIdentifierContext<'raster_layer'>();
|
||||
const selectValue = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) =>
|
||||
selectEntity(canvas, entityIdentifier)?.adjustments?.simple?.[name] ?? DEFAULT_SIMPLE_ADJUSTMENTS[name]
|
||||
);
|
||||
}, [entityIdentifier, name]);
|
||||
const value = useAppSelector(selectValue);
|
||||
|
||||
return (
|
||||
<FormControl orientation="horizontal" mb={1} w="full">
|
||||
<FormLabel m={0} minW="90px">
|
||||
{label}
|
||||
</FormLabel>
|
||||
<CompositeSlider value={value} onChange={onChange} defaultValue={0} min={min} max={max} step={step} marks />
|
||||
<CompositeNumberInput value={value} onChange={onChange} defaultValue={0} min={min} max={max} step={step} />
|
||||
</FormControl>
|
||||
);
|
||||
};
|
||||
|
||||
const DEFAULT_SIMPLE_ADJUSTMENTS = {
|
||||
brightness: 0,
|
||||
contrast: 0,
|
||||
saturation: 0,
|
||||
temperature: 0,
|
||||
tint: 0,
|
||||
sharpness: 0,
|
||||
};
|
||||
|
||||
export const RasterLayerSimpleAdjustmentsEditor = memo(() => {
|
||||
const dispatch = useAppDispatch();
|
||||
const entityIdentifier = useEntityIdentifierContext<'raster_layer'>();
|
||||
const { t } = useTranslation();
|
||||
const selectIsDisabled = useMemo(() => {
|
||||
return createSelector(
|
||||
selectCanvasSlice,
|
||||
(canvas) => selectEntity(canvas, entityIdentifier)?.adjustments?.enabled !== true
|
||||
);
|
||||
}, [entityIdentifier]);
|
||||
const isDisabled = useAppSelector(selectIsDisabled);
|
||||
|
||||
const onBrightness = useCallback(
|
||||
(v: number) => dispatch(rasterLayerAdjustmentsSimpleUpdated({ entityIdentifier, simple: { brightness: v } })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
const onContrast = useCallback(
|
||||
(v: number) => dispatch(rasterLayerAdjustmentsSimpleUpdated({ entityIdentifier, simple: { contrast: v } })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
const onSaturation = useCallback(
|
||||
(v: number) => dispatch(rasterLayerAdjustmentsSimpleUpdated({ entityIdentifier, simple: { saturation: v } })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
const onTemperature = useCallback(
|
||||
(v: number) => dispatch(rasterLayerAdjustmentsSimpleUpdated({ entityIdentifier, simple: { temperature: v } })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
const onTint = useCallback(
|
||||
(v: number) => dispatch(rasterLayerAdjustmentsSimpleUpdated({ entityIdentifier, simple: { tint: v } })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
const onSharpness = useCallback(
|
||||
(v: number) => dispatch(rasterLayerAdjustmentsSimpleUpdated({ entityIdentifier, simple: { sharpness: v } })),
|
||||
[dispatch, entityIdentifier]
|
||||
);
|
||||
|
||||
return (
|
||||
<Flex px={3} pb={2} direction="column" opacity={isDisabled ? 0.3 : 1} pointerEvents={isDisabled ? 'none' : 'auto'}>
|
||||
<AdjustmentSliderRow
|
||||
label={t('controlLayers.adjustments.brightness')}
|
||||
name="brightness"
|
||||
onChange={onBrightness}
|
||||
/>
|
||||
<AdjustmentSliderRow label={t('controlLayers.adjustments.contrast')} name="contrast" onChange={onContrast} />
|
||||
<AdjustmentSliderRow
|
||||
label={t('controlLayers.adjustments.saturation')}
|
||||
name="saturation"
|
||||
onChange={onSaturation}
|
||||
/>
|
||||
<AdjustmentSliderRow
|
||||
label={t('controlLayers.adjustments.temperature')}
|
||||
name="temperature"
|
||||
onChange={onTemperature}
|
||||
/>
|
||||
<AdjustmentSliderRow label={t('controlLayers.adjustments.tint')} name="tint" onChange={onTint} />
|
||||
<AdjustmentSliderRow
|
||||
label={t('controlLayers.adjustments.sharpness')}
|
||||
name="sharpness"
|
||||
onChange={onSharpness}
|
||||
min={0}
|
||||
max={1}
|
||||
/>
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
RasterLayerSimpleAdjustmentsEditor.displayName = 'RasterLayerSimpleAdjustmentsEditor';
|
||||
@@ -1,12 +1,16 @@
|
||||
import { Flex } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { objectEquals } from '@observ33r/object-equals';
|
||||
import { skipToken } from '@reduxjs/toolkit/query';
|
||||
import { useAppSelector, useAppStore } from 'app/store/storeHooks';
|
||||
import { UploadImageIconButton } from 'common/hooks/useImageUploadButton';
|
||||
import { bboxSizeOptimized, bboxSizeRecalled } from 'features/controlLayers/store/canvasSlice';
|
||||
import { useCanvasIsStaging } from 'features/controlLayers/store/canvasStagingAreaSlice';
|
||||
import { sizeOptimized, sizeRecalled } from 'features/controlLayers/store/paramsSlice';
|
||||
import type { ImageWithDims } from 'features/controlLayers/store/types';
|
||||
import type { CroppableImageWithDims } from 'features/controlLayers/store/types';
|
||||
import { imageDTOToCroppableImage, imageDTOToImageWithDims } from 'features/controlLayers/store/util';
|
||||
import { Editor } from 'features/cropper/lib/editor';
|
||||
import { cropImageModalApi } from 'features/cropper/store';
|
||||
import type { setGlobalReferenceImageDndTarget, setRegionalGuidanceReferenceImageDndTarget } from 'features/dnd/dnd';
|
||||
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
||||
import { DndImage } from 'features/dnd/DndImage';
|
||||
@@ -14,14 +18,14 @@ import { DndImageIcon } from 'features/dnd/DndImageIcon';
|
||||
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
||||
import { memo, useCallback, useEffect } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { PiArrowCounterClockwiseBold, PiRulerBold } from 'react-icons/pi';
|
||||
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
||||
import { PiArrowCounterClockwiseBold, PiCropBold, PiRulerBold } from 'react-icons/pi';
|
||||
import { useGetImageDTOQuery, useUploadImageMutation } from 'services/api/endpoints/images';
|
||||
import type { ImageDTO } from 'services/api/types';
|
||||
import { $isConnected } from 'services/events/stores';
|
||||
|
||||
type Props<T extends typeof setGlobalReferenceImageDndTarget | typeof setRegionalGuidanceReferenceImageDndTarget> = {
|
||||
image: ImageWithDims | null;
|
||||
onChangeImage: (imageDTO: ImageDTO | null) => void;
|
||||
image: CroppableImageWithDims | null;
|
||||
onChangeImage: (croppableImage: CroppableImageWithDims | null) => void;
|
||||
dndTarget: T;
|
||||
dndTargetData: ReturnType<T['getData']>;
|
||||
};
|
||||
@@ -38,20 +42,28 @@ export const RefImageImage = memo(
|
||||
const isConnected = useStore($isConnected);
|
||||
const tab = useAppSelector(selectActiveTab);
|
||||
const isStaging = useCanvasIsStaging();
|
||||
const { currentData: imageDTO, isError } = useGetImageDTOQuery(image?.image_name ?? skipToken);
|
||||
const imageWithDims = image?.crop?.image ?? image?.original.image ?? null;
|
||||
const croppedImageDTOReq = useGetImageDTOQuery(image?.crop?.image?.image_name ?? skipToken);
|
||||
const originalImageDTOReq = useGetImageDTOQuery(image?.original.image.image_name ?? skipToken);
|
||||
const [uploadImage] = useUploadImageMutation();
|
||||
|
||||
const originalImageDTO = originalImageDTOReq.currentData;
|
||||
const croppedImageDTO = croppedImageDTOReq.currentData;
|
||||
const imageDTO = croppedImageDTO ?? originalImageDTO;
|
||||
|
||||
const handleResetControlImage = useCallback(() => {
|
||||
onChangeImage(null);
|
||||
}, [onChangeImage]);
|
||||
|
||||
useEffect(() => {
|
||||
if (isConnected && isError) {
|
||||
if ((isConnected && croppedImageDTOReq.isError) || originalImageDTOReq.isError) {
|
||||
handleResetControlImage();
|
||||
}
|
||||
}, [handleResetControlImage, isError, isConnected]);
|
||||
}, [handleResetControlImage, isConnected, croppedImageDTOReq.isError, originalImageDTOReq.isError]);
|
||||
|
||||
const onUpload = useCallback(
|
||||
(imageDTO: ImageDTO) => {
|
||||
onChangeImage(imageDTO);
|
||||
onChangeImage(imageDTOToCroppableImage(imageDTO));
|
||||
},
|
||||
[onChangeImage]
|
||||
);
|
||||
@@ -70,13 +82,67 @@ export const RefImageImage = memo(
|
||||
}
|
||||
}, [imageDTO, isStaging, store, tab]);
|
||||
|
||||
const edit = useCallback(() => {
|
||||
if (!originalImageDTO) {
|
||||
return;
|
||||
}
|
||||
|
||||
// We will create a new editor instance each time the user wants to edit
|
||||
const editor = new Editor();
|
||||
|
||||
// When the user applies the crop, we will upload the cropped image and store the applied crop box so if the user
|
||||
// re-opens the editor they see the same crop
|
||||
const onApplyCrop = async () => {
|
||||
const box = editor.getCropBox();
|
||||
if (objectEquals(box, image?.crop?.box)) {
|
||||
// If the box hasn't changed, don't do anything
|
||||
return;
|
||||
}
|
||||
if (!box || objectEquals(box, { x: 0, y: 0, width: originalImageDTO.width, height: originalImageDTO.height })) {
|
||||
// There is a crop applied but it is the whole iamge - revert to original image
|
||||
onChangeImage(imageDTOToCroppableImage(originalImageDTO));
|
||||
return;
|
||||
}
|
||||
const blob = await editor.exportImage('blob');
|
||||
const file = new File([blob], 'image.png', { type: 'image/png' });
|
||||
|
||||
const newCroppedImageDTO = await uploadImage({
|
||||
file,
|
||||
is_intermediate: true,
|
||||
image_category: 'user',
|
||||
}).unwrap();
|
||||
|
||||
onChangeImage(
|
||||
imageDTOToCroppableImage(originalImageDTO, {
|
||||
image: imageDTOToImageWithDims(newCroppedImageDTO),
|
||||
box,
|
||||
ratio: editor.getCropAspectRatio(),
|
||||
})
|
||||
);
|
||||
};
|
||||
|
||||
const onReady = async () => {
|
||||
const initial = image?.crop ? { cropBox: image.crop.box, aspectRatio: image.crop.ratio } : undefined;
|
||||
// Load the image into the editor and open the modal once it's ready
|
||||
await editor.loadImage(originalImageDTO.image_url, initial);
|
||||
};
|
||||
|
||||
cropImageModalApi.open({ editor, onApplyCrop, onReady });
|
||||
}, [image?.crop, onChangeImage, originalImageDTO, uploadImage]);
|
||||
|
||||
return (
|
||||
<Flex position="relative" w="full" h="full" alignItems="center" data-error={!imageDTO && !image?.image_name}>
|
||||
<Flex
|
||||
position="relative"
|
||||
w="full"
|
||||
h="full"
|
||||
alignItems="center"
|
||||
data-error={!imageDTO && !imageWithDims?.image_name}
|
||||
>
|
||||
{!imageDTO && (
|
||||
<UploadImageIconButton
|
||||
w="full"
|
||||
h="full"
|
||||
isError={!imageDTO && !image?.image_name}
|
||||
isError={!imageDTO && !imageWithDims?.image_name}
|
||||
onUpload={onUpload}
|
||||
fontSize={36}
|
||||
/>
|
||||
@@ -99,6 +165,15 @@ export const RefImageImage = memo(
|
||||
isDisabled={!imageDTO || (tab === 'canvas' && isStaging)}
|
||||
/>
|
||||
</Flex>
|
||||
|
||||
<Flex position="absolute" flexDir="column" top={2} insetInlineStart={2} gap={1}>
|
||||
<DndImageIcon
|
||||
onClick={edit}
|
||||
icon={<PiCropBold size={16} />}
|
||||
tooltip={t('common.crop')}
|
||||
isDisabled={!imageDTO}
|
||||
/>
|
||||
</Flex>
|
||||
</>
|
||||
)}
|
||||
<DndDropTarget dndTarget={dndTarget} dndTargetData={dndTargetData} label={t('gallery.drop')} />
|
||||
|
||||
@@ -13,7 +13,7 @@ import {
|
||||
selectRefImageEntityIds,
|
||||
selectSelectedRefEntityId,
|
||||
} from 'features/controlLayers/store/refImagesSlice';
|
||||
import { imageDTOToImageWithDims } from 'features/controlLayers/store/util';
|
||||
import { imageDTOToCroppableImage } from 'features/controlLayers/store/util';
|
||||
import { addGlobalReferenceImageDndTarget } from 'features/dnd/dnd';
|
||||
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
||||
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
||||
@@ -92,7 +92,7 @@ const AddRefImageDropTargetAndButton = memo(() => {
|
||||
({
|
||||
onUpload: (imageDTO: ImageDTO) => {
|
||||
const config = getDefaultRefImageConfig(getState);
|
||||
config.image = imageDTOToImageWithDims(imageDTO);
|
||||
config.image = imageDTOToCroppableImage(imageDTO);
|
||||
dispatch(refImageAdded({ overrides: { config } }));
|
||||
},
|
||||
allowMultiple: false,
|
||||
|
||||
@@ -1,6 +1,5 @@
|
||||
import type { SystemStyleObject } from '@invoke-ai/ui-library';
|
||||
import { Flex, Icon, IconButton, Image, Skeleton, Text, Tooltip } from '@invoke-ai/ui-library';
|
||||
import { skipToken } from '@reduxjs/toolkit/query';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { round } from 'es-toolkit/compat';
|
||||
import { useRefImageEntity } from 'features/controlLayers/components/RefImage/useRefImageEntity';
|
||||
@@ -15,7 +14,7 @@ import { isIPAdapterConfig } from 'features/controlLayers/store/types';
|
||||
import { getGlobalReferenceImageWarnings } from 'features/controlLayers/store/validators';
|
||||
import { memo, useCallback, useEffect, useMemo, useState } from 'react';
|
||||
import { PiExclamationMarkBold, PiEyeSlashBold, PiImageBold } from 'react-icons/pi';
|
||||
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
||||
import { useImageDTOFromCroppableImage } from 'services/api/endpoints/images';
|
||||
|
||||
import { RefImageWarningTooltipContent } from './RefImageWarningTooltipContent';
|
||||
|
||||
@@ -72,7 +71,8 @@ export const RefImagePreview = memo(() => {
|
||||
const selectedEntityId = useAppSelector(selectSelectedRefEntityId);
|
||||
const isPanelOpen = useAppSelector(selectIsRefImagePanelOpen);
|
||||
const [showWeightDisplay, setShowWeightDisplay] = useState(false);
|
||||
const { data: imageDTO } = useGetImageDTOQuery(entity.config.image?.image_name ?? skipToken);
|
||||
|
||||
const imageDTO = useImageDTOFromCroppableImage(entity.config.image);
|
||||
|
||||
const sx = useMemo(() => {
|
||||
if (!isIPAdapterConfig(entity.config)) {
|
||||
@@ -145,7 +145,7 @@ export const RefImagePreview = memo(() => {
|
||||
overflow="hidden"
|
||||
>
|
||||
<Image
|
||||
src={imageDTO?.thumbnail_url}
|
||||
src={imageDTO?.image_url}
|
||||
objectFit="contain"
|
||||
aspectRatio="1/1"
|
||||
height={imageDTO?.height}
|
||||
|
||||
@@ -30,6 +30,7 @@ import {
|
||||
} from 'features/controlLayers/store/refImagesSlice';
|
||||
import type {
|
||||
CLIPVisionModelV2,
|
||||
CroppableImageWithDims,
|
||||
FLUXReduxImageInfluence as FLUXReduxImageInfluenceType,
|
||||
IPMethodV2,
|
||||
} from 'features/controlLayers/store/types';
|
||||
@@ -42,7 +43,6 @@ import type {
|
||||
ChatGPT4oModelConfig,
|
||||
FLUXKontextModelConfig,
|
||||
FLUXReduxModelConfig,
|
||||
ImageDTO,
|
||||
IPAdapterModelConfig,
|
||||
} from 'services/api/types';
|
||||
|
||||
@@ -104,15 +104,19 @@ const RefImageSettingsContent = memo(() => {
|
||||
);
|
||||
|
||||
const onChangeImage = useCallback(
|
||||
(imageDTO: ImageDTO | null) => {
|
||||
dispatch(refImageImageChanged({ id, imageDTO }));
|
||||
(croppableImage: CroppableImageWithDims | null) => {
|
||||
dispatch(refImageImageChanged({ id, croppableImage }));
|
||||
},
|
||||
[dispatch, id]
|
||||
);
|
||||
|
||||
const dndTargetData = useMemo<SetGlobalReferenceImageDndTargetData>(
|
||||
() => setGlobalReferenceImageDndTarget.getData({ id }, config.image?.image_name),
|
||||
[id, config.image?.image_name]
|
||||
() =>
|
||||
setGlobalReferenceImageDndTarget.getData(
|
||||
{ id },
|
||||
config.image?.crop?.image.image_name ?? config.image?.original.image.image_name
|
||||
),
|
||||
[id, config.image?.crop?.image.image_name, config.image?.original.image.image_name]
|
||||
);
|
||||
|
||||
const isFLUX = useAppSelector(selectIsFLUX);
|
||||
|
||||
@@ -6,7 +6,6 @@ import { FLUXReduxImageInfluence } from 'features/controlLayers/components/commo
|
||||
import { IPAdapterCLIPVisionModel } from 'features/controlLayers/components/common/IPAdapterCLIPVisionModel';
|
||||
import { Weight } from 'features/controlLayers/components/common/Weight';
|
||||
import { IPAdapterMethod } from 'features/controlLayers/components/RefImage/IPAdapterMethod';
|
||||
import { RefImageImage } from 'features/controlLayers/components/RefImage/RefImageImage';
|
||||
import { RegionalGuidanceIPAdapterSettingsEmptyState } from 'features/controlLayers/components/RegionalGuidance/RegionalGuidanceIPAdapterSettingsEmptyState';
|
||||
import { RegionalReferenceImageModel } from 'features/controlLayers/components/RegionalGuidance/RegionalReferenceImageModel';
|
||||
import { useEntityIdentifierContext } from 'features/controlLayers/contexts/EntityIdentifierContext';
|
||||
@@ -37,6 +36,8 @@ import { PiBoundingBoxBold, PiXBold } from 'react-icons/pi';
|
||||
import type { FLUXReduxModelConfig, ImageDTO, IPAdapterModelConfig } from 'services/api/types';
|
||||
import { assert } from 'tsafe';
|
||||
|
||||
import { RegionalGuidanceRefImageImage } from './RegionalGuidanceRefImageImage';
|
||||
|
||||
type Props = {
|
||||
referenceImageId: string;
|
||||
};
|
||||
@@ -114,7 +115,7 @@ const RegionalGuidanceIPAdapterSettingsContent = memo(({ referenceImageId }: Pro
|
||||
{ entityIdentifier, referenceImageId },
|
||||
config.image?.image_name
|
||||
),
|
||||
[entityIdentifier, config.image?.image_name, referenceImageId]
|
||||
[entityIdentifier, config.image, referenceImageId]
|
||||
);
|
||||
|
||||
const pullBboxIntoIPAdapter = usePullBboxIntoRegionalGuidanceReferenceImage(entityIdentifier, referenceImageId);
|
||||
@@ -170,7 +171,7 @@ const RegionalGuidanceIPAdapterSettingsContent = memo(({ referenceImageId }: Pro
|
||||
</Flex>
|
||||
)}
|
||||
<Flex alignItems="center" justifyContent="center" h={32} w={32} aspectRatio="1/1" flexGrow={1}>
|
||||
<RefImageImage
|
||||
<RegionalGuidanceRefImageImage
|
||||
image={config.image}
|
||||
onChangeImage={onChangeImage}
|
||||
dndTarget={setRegionalGuidanceReferenceImageDndTarget}
|
||||
|
||||
@@ -0,0 +1,103 @@
|
||||
import { Flex } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { skipToken } from '@reduxjs/toolkit/query';
|
||||
import { useAppSelector, useAppStore } from 'app/store/storeHooks';
|
||||
import { UploadImageIconButton } from 'common/hooks/useImageUploadButton';
|
||||
import { bboxSizeOptimized, bboxSizeRecalled } from 'features/controlLayers/store/canvasSlice';
|
||||
import { useCanvasIsStaging } from 'features/controlLayers/store/canvasStagingAreaSlice';
|
||||
import { sizeOptimized, sizeRecalled } from 'features/controlLayers/store/paramsSlice';
|
||||
import type { ImageWithDims } from 'features/controlLayers/store/types';
|
||||
import type { setRegionalGuidanceReferenceImageDndTarget } from 'features/dnd/dnd';
|
||||
import { DndDropTarget } from 'features/dnd/DndDropTarget';
|
||||
import { DndImage } from 'features/dnd/DndImage';
|
||||
import { DndImageIcon } from 'features/dnd/DndImageIcon';
|
||||
import { selectActiveTab } from 'features/ui/store/uiSelectors';
|
||||
import { memo, useCallback, useEffect } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { PiArrowCounterClockwiseBold, PiRulerBold } from 'react-icons/pi';
|
||||
import { useGetImageDTOQuery } from 'services/api/endpoints/images';
|
||||
import type { ImageDTO } from 'services/api/types';
|
||||
import { $isConnected } from 'services/events/stores';
|
||||
|
||||
type Props = {
|
||||
image: ImageWithDims | null;
|
||||
onChangeImage: (imageDTO: ImageDTO | null) => void;
|
||||
dndTarget: typeof setRegionalGuidanceReferenceImageDndTarget;
|
||||
dndTargetData: ReturnType<(typeof setRegionalGuidanceReferenceImageDndTarget)['getData']>;
|
||||
};
|
||||
|
||||
export const RegionalGuidanceRefImageImage = memo(({ image, onChangeImage, dndTarget, dndTargetData }: Props) => {
|
||||
const { t } = useTranslation();
|
||||
const store = useAppStore();
|
||||
const isConnected = useStore($isConnected);
|
||||
const tab = useAppSelector(selectActiveTab);
|
||||
const isStaging = useCanvasIsStaging();
|
||||
const { currentData: imageDTO, isError } = useGetImageDTOQuery(image?.image_name ?? skipToken);
|
||||
const handleResetControlImage = useCallback(() => {
|
||||
onChangeImage(null);
|
||||
}, [onChangeImage]);
|
||||
|
||||
useEffect(() => {
|
||||
if (isConnected && isError) {
|
||||
handleResetControlImage();
|
||||
}
|
||||
}, [handleResetControlImage, isError, isConnected]);
|
||||
|
||||
const onUpload = useCallback(
|
||||
(imageDTO: ImageDTO) => {
|
||||
onChangeImage(imageDTO);
|
||||
},
|
||||
[onChangeImage]
|
||||
);
|
||||
|
||||
const recallSizeAndOptimize = useCallback(() => {
|
||||
if (!imageDTO || (tab === 'canvas' && isStaging)) {
|
||||
return;
|
||||
}
|
||||
const { width, height } = imageDTO;
|
||||
if (tab === 'canvas') {
|
||||
store.dispatch(bboxSizeRecalled({ width, height }));
|
||||
store.dispatch(bboxSizeOptimized());
|
||||
} else if (tab === 'generate') {
|
||||
store.dispatch(sizeRecalled({ width, height }));
|
||||
store.dispatch(sizeOptimized());
|
||||
}
|
||||
}, [imageDTO, isStaging, store, tab]);
|
||||
|
||||
return (
|
||||
<Flex position="relative" w="full" h="full" alignItems="center" data-error={!imageDTO && !image?.image_name}>
|
||||
{!imageDTO && (
|
||||
<UploadImageIconButton
|
||||
w="full"
|
||||
h="full"
|
||||
isError={!imageDTO && !image?.image_name}
|
||||
onUpload={onUpload}
|
||||
fontSize={36}
|
||||
/>
|
||||
)}
|
||||
{imageDTO && (
|
||||
<>
|
||||
<DndImage imageDTO={imageDTO} borderRadius="base" borderWidth={1} borderStyle="solid" w="full" />
|
||||
<Flex position="absolute" flexDir="column" top={2} insetInlineEnd={2} gap={1}>
|
||||
<DndImageIcon
|
||||
onClick={handleResetControlImage}
|
||||
icon={<PiArrowCounterClockwiseBold size={16} />}
|
||||
tooltip={t('common.reset')}
|
||||
/>
|
||||
</Flex>
|
||||
<Flex position="absolute" flexDir="column" bottom={2} insetInlineEnd={2} gap={1}>
|
||||
<DndImageIcon
|
||||
onClick={recallSizeAndOptimize}
|
||||
icon={<PiRulerBold size={16} />}
|
||||
tooltip={t('parameters.useSize')}
|
||||
isDisabled={!imageDTO || (tab === 'canvas' && isStaging)}
|
||||
/>
|
||||
</Flex>
|
||||
</>
|
||||
)}
|
||||
<DndDropTarget dndTarget={dndTarget} dndTargetData={dndTargetData} label={t('gallery.drop')} />
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
RegionalGuidanceRefImageImage.displayName = 'RegionalGuidanceRefImageImage';
|
||||
@@ -1,36 +1,22 @@
|
||||
import {
|
||||
Button,
|
||||
ButtonGroup,
|
||||
Flex,
|
||||
Heading,
|
||||
Icon,
|
||||
ListItem,
|
||||
Menu,
|
||||
MenuButton,
|
||||
MenuItem,
|
||||
MenuList,
|
||||
Spacer,
|
||||
Spinner,
|
||||
Text,
|
||||
Tooltip,
|
||||
UnorderedList,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { Flex, Heading, Spacer } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { useAppSelector } from 'app/store/storeHooks';
|
||||
import { useFocusRegion, useIsRegionFocused } from 'common/hooks/focus';
|
||||
import { CanvasAutoProcessSwitch } from 'features/controlLayers/components/CanvasAutoProcessSwitch';
|
||||
import { CanvasOperationIsolatedLayerPreviewSwitch } from 'features/controlLayers/components/CanvasOperationIsolatedLayerPreviewSwitch';
|
||||
import { SelectObjectActionButtons } from 'features/controlLayers/components/SelectObject/SelectObjectActionButtons';
|
||||
import { SelectObjectInfoTooltip } from 'features/controlLayers/components/SelectObject/SelectObjectInfoTooltip';
|
||||
import { SelectObjectInputTypeButtons } from 'features/controlLayers/components/SelectObject/SelectObjectInputTypeButtons';
|
||||
import { SelectObjectInvert } from 'features/controlLayers/components/SelectObject/SelectObjectInvert';
|
||||
import { SelectObjectPointType } from 'features/controlLayers/components/SelectObject/SelectObjectPointType';
|
||||
import { useCanvasManager } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
||||
import { selectAutoProcess } from 'features/controlLayers/store/canvasSettingsSlice';
|
||||
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||
import type { PropsWithChildren } from 'react';
|
||||
import { memo, useCallback, useRef } from 'react';
|
||||
import { Trans, useTranslation } from 'react-i18next';
|
||||
import { PiCaretDownBold, PiInfoBold } from 'react-icons/pi';
|
||||
import { memo, useRef } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
import { SelectObjectModel } from './SelectObjectModel';
|
||||
import { SelectObjectPrompt } from './SelectObjectPrompt';
|
||||
|
||||
const SelectObjectContent = memo(
|
||||
({ adapter }: { adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer }) => {
|
||||
@@ -39,25 +25,7 @@ const SelectObjectContent = memo(
|
||||
useFocusRegion('canvas', ref, { focusOnMount: true });
|
||||
const isCanvasFocused = useIsRegionFocused('canvas');
|
||||
const isProcessing = useStore(adapter.segmentAnything.$isProcessing);
|
||||
const hasPoints = useStore(adapter.segmentAnything.$hasPoints);
|
||||
const hasImageState = useStore(adapter.segmentAnything.$hasImageState);
|
||||
const autoProcess = useAppSelector(selectAutoProcess);
|
||||
|
||||
const saveAsInpaintMask = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('inpaint_mask');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const saveAsRegionalGuidance = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('regional_guidance');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const saveAsRasterLayer = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('raster_layer');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const saveAsControlLayer = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('control_layer');
|
||||
}, [adapter.segmentAnything]);
|
||||
const inputType = useStore(adapter.segmentAnything.$inputType);
|
||||
|
||||
useRegisteredHotkeys({
|
||||
id: 'applySegmentAnything',
|
||||
@@ -94,11 +62,7 @@ const SelectObjectContent = memo(
|
||||
<Heading size="md" color="base.300" userSelect="none">
|
||||
{t('controlLayers.selectObject.selectObject')}
|
||||
</Heading>
|
||||
<Tooltip label={<SelectObjectHelpTooltipContent />}>
|
||||
<Flex alignItems="center">
|
||||
<Icon as={PiInfoBold} color="base.500" />
|
||||
</Flex>
|
||||
</Tooltip>
|
||||
<SelectObjectInfoTooltip />
|
||||
</Flex>
|
||||
<Spacer />
|
||||
<CanvasAutoProcessSwitch />
|
||||
@@ -106,71 +70,14 @@ const SelectObjectContent = memo(
|
||||
</Flex>
|
||||
|
||||
<Flex w="full" justifyContent="space-between" py={2}>
|
||||
<SelectObjectPointType adapter={adapter} />
|
||||
<SelectObjectInputTypeButtons adapter={adapter} />
|
||||
<SelectObjectInvert adapter={adapter} />
|
||||
</Flex>
|
||||
|
||||
<ButtonGroup isAttached={false} size="sm" w="full">
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.processImmediate}
|
||||
loadingText={t('controlLayers.selectObject.process')}
|
||||
variant="ghost"
|
||||
isDisabled={isProcessing || !hasPoints || (autoProcess && hasImageState)}
|
||||
>
|
||||
{t('controlLayers.selectObject.process')}
|
||||
{isProcessing && <Spinner ms={3} boxSize={5} color="base.600" />}
|
||||
</Button>
|
||||
<Spacer />
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.reset}
|
||||
isDisabled={isProcessing || !hasPoints}
|
||||
loadingText={t('controlLayers.selectObject.reset')}
|
||||
variant="ghost"
|
||||
>
|
||||
{t('controlLayers.selectObject.reset')}
|
||||
</Button>
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.apply}
|
||||
loadingText={t('controlLayers.selectObject.apply')}
|
||||
variant="ghost"
|
||||
isDisabled={isProcessing || !hasImageState}
|
||||
>
|
||||
{t('controlLayers.selectObject.apply')}
|
||||
</Button>
|
||||
<Menu>
|
||||
<MenuButton
|
||||
as={Button}
|
||||
loadingText={t('controlLayers.selectObject.saveAs')}
|
||||
variant="ghost"
|
||||
isDisabled={isProcessing || !hasImageState}
|
||||
rightIcon={<PiCaretDownBold />}
|
||||
>
|
||||
{t('controlLayers.selectObject.saveAs')}
|
||||
</MenuButton>
|
||||
<MenuList>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsInpaintMask}>
|
||||
{t('controlLayers.newInpaintMask')}
|
||||
</MenuItem>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsRegionalGuidance}>
|
||||
{t('controlLayers.newRegionalGuidance')}
|
||||
</MenuItem>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsControlLayer}>
|
||||
{t('controlLayers.newControlLayer')}
|
||||
</MenuItem>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsRasterLayer}>
|
||||
{t('controlLayers.newRasterLayer')}
|
||||
</MenuItem>
|
||||
</MenuList>
|
||||
</Menu>
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.cancel}
|
||||
isDisabled={isProcessing}
|
||||
loadingText={t('common.cancel')}
|
||||
variant="ghost"
|
||||
>
|
||||
{t('controlLayers.selectObject.cancel')}
|
||||
</Button>
|
||||
</ButtonGroup>
|
||||
{inputType === 'visual' && <SelectObjectPointType adapter={adapter} />}
|
||||
{inputType === 'prompt' && <SelectObjectPrompt adapter={adapter} />}
|
||||
<SelectObjectModel adapter={adapter} />
|
||||
<SelectObjectActionButtons adapter={adapter} />
|
||||
</Flex>
|
||||
);
|
||||
}
|
||||
@@ -190,34 +97,3 @@ export const SelectObject = memo(() => {
|
||||
});
|
||||
|
||||
SelectObject.displayName = 'SelectObject';
|
||||
|
||||
const Bold = (props: PropsWithChildren) => (
|
||||
<Text as="span" fontWeight="semibold">
|
||||
{props.children}
|
||||
</Text>
|
||||
);
|
||||
|
||||
const SelectObjectHelpTooltipContent = memo(() => {
|
||||
const { t } = useTranslation();
|
||||
|
||||
return (
|
||||
<Flex gap={3} flexDir="column">
|
||||
<Text>
|
||||
<Trans i18nKey="controlLayers.selectObject.help1" components={{ Bold: <Bold /> }} />
|
||||
</Text>
|
||||
<Text>
|
||||
<Trans i18nKey="controlLayers.selectObject.help2" components={{ Bold: <Bold /> }} />
|
||||
</Text>
|
||||
<Text>
|
||||
<Trans i18nKey="controlLayers.selectObject.help3" />
|
||||
</Text>
|
||||
<UnorderedList>
|
||||
<ListItem>{t('controlLayers.selectObject.clickToAdd')}</ListItem>
|
||||
<ListItem>{t('controlLayers.selectObject.dragToMove')}</ListItem>
|
||||
<ListItem>{t('controlLayers.selectObject.clickToRemove')}</ListItem>
|
||||
</UnorderedList>
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
SelectObjectHelpTooltipContent.displayName = 'SelectObjectHelpTooltipContent';
|
||||
|
||||
@@ -0,0 +1,64 @@
|
||||
import { Button, ButtonGroup, Spacer, Spinner } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { useAppSelector } from 'app/store/storeHooks';
|
||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
||||
import { selectAutoProcess } from 'features/controlLayers/store/canvasSettingsSlice';
|
||||
import { memo } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
import { SelectObjectSaveAsMenu } from './SelectObjectSaveAsMenu';
|
||||
|
||||
interface SelectObjectActionButtonsProps {
|
||||
adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer;
|
||||
}
|
||||
|
||||
export const SelectObjectActionButtons = memo(({ adapter }: SelectObjectActionButtonsProps) => {
|
||||
const { t } = useTranslation();
|
||||
const isProcessing = useStore(adapter.segmentAnything.$isProcessing);
|
||||
const hasInput = useStore(adapter.segmentAnything.$hasInputData);
|
||||
const hasImageState = useStore(adapter.segmentAnything.$hasImageState);
|
||||
const autoProcess = useAppSelector(selectAutoProcess);
|
||||
|
||||
return (
|
||||
<ButtonGroup isAttached={false} size="sm" w="full">
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.processImmediate}
|
||||
loadingText={t('controlLayers.selectObject.process')}
|
||||
variant="ghost"
|
||||
isDisabled={isProcessing || !hasInput || (autoProcess && hasImageState)}
|
||||
>
|
||||
{t('controlLayers.selectObject.process')}
|
||||
{isProcessing && <Spinner ms={3} boxSize={5} color="base.600" />}
|
||||
</Button>
|
||||
<Spacer />
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.reset}
|
||||
isDisabled={isProcessing || !hasInput}
|
||||
loadingText={t('controlLayers.selectObject.reset')}
|
||||
variant="ghost"
|
||||
>
|
||||
{t('controlLayers.selectObject.reset')}
|
||||
</Button>
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.apply}
|
||||
loadingText={t('controlLayers.selectObject.apply')}
|
||||
variant="ghost"
|
||||
isDisabled={isProcessing || !hasImageState}
|
||||
>
|
||||
{t('controlLayers.selectObject.apply')}
|
||||
</Button>
|
||||
<SelectObjectSaveAsMenu adapter={adapter} />
|
||||
<Button
|
||||
onClick={adapter.segmentAnything.cancel}
|
||||
isDisabled={isProcessing}
|
||||
loadingText={t('common.cancel')}
|
||||
variant="ghost"
|
||||
>
|
||||
{t('controlLayers.selectObject.cancel')}
|
||||
</Button>
|
||||
</ButtonGroup>
|
||||
);
|
||||
});
|
||||
|
||||
SelectObjectActionButtons.displayName = 'SelectObjectActionButtons';
|
||||
@@ -0,0 +1,59 @@
|
||||
import { Flex, Icon, ListItem, Text, Tooltip, UnorderedList } from '@invoke-ai/ui-library';
|
||||
import type { PropsWithChildren } from 'react';
|
||||
import { memo } from 'react';
|
||||
import { Trans } from 'react-i18next';
|
||||
import { PiInfoBold } from 'react-icons/pi';
|
||||
|
||||
const Bold = (props: PropsWithChildren) => (
|
||||
<Text as="span" fontWeight="semibold">
|
||||
{props.children}
|
||||
</Text>
|
||||
);
|
||||
|
||||
const components = { Bold: <Bold /> };
|
||||
|
||||
const SelectObjectHelpTooltipContent = memo(() => {
|
||||
return (
|
||||
<Flex gap={3} flexDir="column">
|
||||
<Text>
|
||||
<Trans i18nKey="controlLayers.selectObject.desc" components={components} />
|
||||
</Text>
|
||||
<UnorderedList>
|
||||
<ListItem>
|
||||
<Trans i18nKey="controlLayers.selectObject.visualMode1" components={components} />
|
||||
</ListItem>
|
||||
<ListItem>
|
||||
<Trans i18nKey="controlLayers.selectObject.visualMode2" components={components} />
|
||||
</ListItem>
|
||||
<ListItem>
|
||||
<Trans i18nKey="controlLayers.selectObject.visualMode3" components={components} />
|
||||
</ListItem>
|
||||
</UnorderedList>
|
||||
<Text>
|
||||
<Trans i18nKey="controlLayers.selectObject.promptModeDesc" components={components} />
|
||||
</Text>
|
||||
<UnorderedList>
|
||||
<ListItem>
|
||||
<Trans i18nKey="controlLayers.selectObject.promptMode1" components={components} />
|
||||
</ListItem>
|
||||
<ListItem>
|
||||
<Trans i18nKey="controlLayers.selectObject.promptMode2" components={components} />
|
||||
</ListItem>
|
||||
</UnorderedList>
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
SelectObjectHelpTooltipContent.displayName = 'SelectObjectHelpTooltipContent';
|
||||
|
||||
export const SelectObjectInfoTooltip = memo(() => {
|
||||
return (
|
||||
<Tooltip label={<SelectObjectHelpTooltipContent />} minW={420}>
|
||||
<Flex alignItems="center">
|
||||
<Icon as={PiInfoBold} color="base.500" />
|
||||
</Flex>
|
||||
</Tooltip>
|
||||
);
|
||||
});
|
||||
|
||||
SelectObjectInfoTooltip.displayName = 'SelectObjectInfoTooltip';
|
||||
@@ -0,0 +1,34 @@
|
||||
import { Button, ButtonGroup } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
||||
import { memo, useCallback } from 'react';
|
||||
|
||||
interface SelectObjectInputTypeButtonsProps {
|
||||
adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer;
|
||||
}
|
||||
|
||||
export const SelectObjectInputTypeButtons = memo(({ adapter }: SelectObjectInputTypeButtonsProps) => {
|
||||
const inputType = useStore(adapter.segmentAnything.$inputType);
|
||||
|
||||
const setInputToVisual = useCallback(() => {
|
||||
adapter.segmentAnything.setInputType('visual');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const setInputToPrompt = useCallback(() => {
|
||||
adapter.segmentAnything.setInputType('prompt');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
return (
|
||||
<ButtonGroup size="sm" variant="outline">
|
||||
<Button colorScheme={inputType === 'visual' ? 'invokeBlue' : undefined} onClick={setInputToVisual}>
|
||||
Visual
|
||||
</Button>
|
||||
<Button colorScheme={inputType === 'prompt' ? 'invokeBlue' : undefined} onClick={setInputToPrompt}>
|
||||
Prompt
|
||||
</Button>
|
||||
</ButtonGroup>
|
||||
);
|
||||
});
|
||||
|
||||
SelectObjectInputTypeButtons.displayName = 'SelectObjectInputTypeButtons';
|
||||
@@ -0,0 +1,40 @@
|
||||
import { Flex, FormControl, FormLabel, Radio, RadioGroup, Text } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
||||
import { zSAMModel } from 'features/controlLayers/store/types';
|
||||
import { memo, useCallback } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
export const SelectObjectModel = memo(
|
||||
({ adapter }: { adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer }) => {
|
||||
const { t } = useTranslation();
|
||||
const model = useStore(adapter.segmentAnything.$model);
|
||||
|
||||
const onChange = useCallback(
|
||||
(v: string) => {
|
||||
const model = zSAMModel.parse(v);
|
||||
adapter.segmentAnything.$model.set(model);
|
||||
},
|
||||
[adapter.segmentAnything.$model]
|
||||
);
|
||||
|
||||
return (
|
||||
<FormControl w="full">
|
||||
<FormLabel m={0}>{t('controlLayers.selectObject.model')}</FormLabel>
|
||||
<RadioGroup value={model} onChange={onChange} w="full" size="md">
|
||||
<Flex alignItems="center" w="full" gap={4} color="base.300">
|
||||
<Radio value="SAM1">
|
||||
<Text>{t('controlLayers.selectObject.segmentAnything1')}</Text>
|
||||
</Radio>
|
||||
<Radio value="SAM2">
|
||||
<Text>{t('controlLayers.selectObject.segmentAnything2')}</Text>
|
||||
</Radio>
|
||||
</Flex>
|
||||
</RadioGroup>
|
||||
</FormControl>
|
||||
);
|
||||
}
|
||||
);
|
||||
|
||||
SelectObjectModel.displayName = 'SelectObjectModel';
|
||||
@@ -24,7 +24,7 @@ export const SelectObjectPointType = memo(
|
||||
<FormControl w="min-content">
|
||||
<FormLabel m={0}>{t('controlLayers.selectObject.pointType')}</FormLabel>
|
||||
<RadioGroup value={pointType} onChange={onChange} w="full" size="md">
|
||||
<Flex alignItems="center" w="full" gap={4} fontWeight="semibold" color="base.300">
|
||||
<Flex alignItems="center" w="full" gap={4} color="base.300">
|
||||
<Radio value="foreground">
|
||||
<Text>{t('controlLayers.selectObject.include')}</Text>
|
||||
</Radio>
|
||||
|
||||
@@ -0,0 +1,34 @@
|
||||
import { FormControl, FormLabel, Input } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
||||
import type { ChangeEvent } from 'react';
|
||||
import { memo, useCallback } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
|
||||
export const SelectObjectPrompt = memo(
|
||||
({ adapter }: { adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer }) => {
|
||||
const { t } = useTranslation();
|
||||
const inputData = useStore(adapter.segmentAnything.$inputData);
|
||||
|
||||
const onChange = useCallback(
|
||||
(e: ChangeEvent<HTMLInputElement>) => {
|
||||
adapter.segmentAnything.$inputData.set({ type: 'prompt', prompt: e.target.value });
|
||||
},
|
||||
[adapter.segmentAnything.$inputData]
|
||||
);
|
||||
|
||||
if (inputData.type !== 'prompt') {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
<FormControl w="full">
|
||||
<FormLabel m={0}>{t('controlLayers.selectObject.prompt')}</FormLabel>
|
||||
<Input value={inputData.prompt} onChange={onChange} />
|
||||
</FormControl>
|
||||
);
|
||||
}
|
||||
);
|
||||
|
||||
SelectObjectPrompt.displayName = 'SelectObjectPrompt';
|
||||
@@ -0,0 +1,63 @@
|
||||
import { Button, Menu, MenuButton, MenuItem, MenuList } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import type { CanvasEntityAdapterControlLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterControlLayer';
|
||||
import type { CanvasEntityAdapterRasterLayer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterRasterLayer';
|
||||
import { memo, useCallback } from 'react';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import { PiCaretDownBold } from 'react-icons/pi';
|
||||
|
||||
interface SelectObjectSaveAsMenuProps {
|
||||
adapter: CanvasEntityAdapterRasterLayer | CanvasEntityAdapterControlLayer;
|
||||
}
|
||||
|
||||
export const SelectObjectSaveAsMenu = memo(({ adapter }: SelectObjectSaveAsMenuProps) => {
|
||||
const { t } = useTranslation();
|
||||
const isProcessing = useStore(adapter.segmentAnything.$isProcessing);
|
||||
const hasImageState = useStore(adapter.segmentAnything.$hasImageState);
|
||||
|
||||
const saveAsInpaintMask = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('inpaint_mask');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const saveAsRegionalGuidance = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('regional_guidance');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const saveAsRasterLayer = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('raster_layer');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
const saveAsControlLayer = useCallback(() => {
|
||||
adapter.segmentAnything.saveAs('control_layer');
|
||||
}, [adapter.segmentAnything]);
|
||||
|
||||
return (
|
||||
<Menu>
|
||||
<MenuButton
|
||||
as={Button}
|
||||
loadingText={t('controlLayers.selectObject.saveAs')}
|
||||
variant="ghost"
|
||||
isDisabled={isProcessing || !hasImageState}
|
||||
rightIcon={<PiCaretDownBold />}
|
||||
>
|
||||
{t('controlLayers.selectObject.saveAs')}
|
||||
</MenuButton>
|
||||
<MenuList>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsInpaintMask}>
|
||||
{t('controlLayers.newInpaintMask')}
|
||||
</MenuItem>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsRegionalGuidance}>
|
||||
{t('controlLayers.newRegionalGuidance')}
|
||||
</MenuItem>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsControlLayer}>
|
||||
{t('controlLayers.newControlLayer')}
|
||||
</MenuItem>
|
||||
<MenuItem isDisabled={isProcessing || !hasImageState} onClick={saveAsRasterLayer}>
|
||||
{t('controlLayers.newRasterLayer')}
|
||||
</MenuItem>
|
||||
</MenuList>
|
||||
</Menu>
|
||||
);
|
||||
});
|
||||
|
||||
SelectObjectSaveAsMenu.displayName = 'SelectObjectSaveAsMenu';
|
||||
@@ -8,6 +8,7 @@ import {
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
Portal,
|
||||
Text,
|
||||
useShiftModifier,
|
||||
} from '@invoke-ai/ui-library';
|
||||
@@ -45,62 +46,64 @@ export const CanvasSettingsPopover = memo(() => {
|
||||
alignSelf="stretch"
|
||||
/>
|
||||
</PopoverTrigger>
|
||||
<PopoverContent maxW="280px">
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<Flex direction="column" gap={2}>
|
||||
{/* Behavior Settings */}
|
||||
<Flex direction="column" gap={1}>
|
||||
<Flex align="center" gap={2}>
|
||||
<Icon as={PiPencilFill} boxSize={4} />
|
||||
<Text fontWeight="bold" fontSize="sm" color="base.100">
|
||||
{t('hotkeys.canvas.settings.behavior')}
|
||||
</Text>
|
||||
<Portal>
|
||||
<PopoverContent maxW="280px">
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<Flex direction="column" gap={2}>
|
||||
{/* Behavior Settings */}
|
||||
<Flex direction="column" gap={1}>
|
||||
<Flex align="center" gap={2}>
|
||||
<Icon as={PiPencilFill} boxSize={4} />
|
||||
<Text fontWeight="bold" fontSize="sm" color="base.100">
|
||||
{t('hotkeys.canvas.settings.behavior')}
|
||||
</Text>
|
||||
</Flex>
|
||||
<CanvasSettingsInvertScrollCheckbox />
|
||||
<CanvasSettingsPressureSensitivityCheckbox />
|
||||
<CanvasSettingsPreserveMaskCheckbox />
|
||||
<CanvasSettingsClipToBboxCheckbox />
|
||||
<CanvasSettingsOutputOnlyMaskedRegionsCheckbox />
|
||||
<CanvasSettingsSaveAllImagesToGalleryCheckbox />
|
||||
</Flex>
|
||||
<CanvasSettingsInvertScrollCheckbox />
|
||||
<CanvasSettingsPressureSensitivityCheckbox />
|
||||
<CanvasSettingsPreserveMaskCheckbox />
|
||||
<CanvasSettingsClipToBboxCheckbox />
|
||||
<CanvasSettingsOutputOnlyMaskedRegionsCheckbox />
|
||||
<CanvasSettingsSaveAllImagesToGalleryCheckbox />
|
||||
</Flex>
|
||||
|
||||
<Divider />
|
||||
<Divider />
|
||||
|
||||
{/* Display Settings */}
|
||||
<Flex direction="column" gap={1}>
|
||||
<Flex align="center" gap={2} color="base.200">
|
||||
<Icon as={PiEyeFill} boxSize={4} />
|
||||
<Text fontWeight="bold" fontSize="sm">
|
||||
{t('hotkeys.canvas.settings.display')}
|
||||
</Text>
|
||||
{/* Display Settings */}
|
||||
<Flex direction="column" gap={1}>
|
||||
<Flex align="center" gap={2} color="base.200">
|
||||
<Icon as={PiEyeFill} boxSize={4} />
|
||||
<Text fontWeight="bold" fontSize="sm">
|
||||
{t('hotkeys.canvas.settings.display')}
|
||||
</Text>
|
||||
</Flex>
|
||||
<CanvasSettingsShowProgressOnCanvas />
|
||||
<CanvasSettingsIsolatedStagingPreviewSwitch />
|
||||
<CanvasSettingsIsolatedLayerPreviewSwitch />
|
||||
<CanvasSettingsBboxOverlaySwitch />
|
||||
<CanvasSettingsShowHUDSwitch />
|
||||
</Flex>
|
||||
<CanvasSettingsShowProgressOnCanvas />
|
||||
<CanvasSettingsIsolatedStagingPreviewSwitch />
|
||||
<CanvasSettingsIsolatedLayerPreviewSwitch />
|
||||
<CanvasSettingsBboxOverlaySwitch />
|
||||
<CanvasSettingsShowHUDSwitch />
|
||||
</Flex>
|
||||
|
||||
<Divider />
|
||||
<Divider />
|
||||
|
||||
{/* Grid Settings */}
|
||||
<Flex direction="column" gap={1}>
|
||||
<Flex align="center" gap={2} color="base.200">
|
||||
<Icon as={PiSquaresFourFill} boxSize={4} />
|
||||
<Text fontWeight="bold" fontSize="sm">
|
||||
{t('hotkeys.canvas.settings.grid')}
|
||||
</Text>
|
||||
{/* Grid Settings */}
|
||||
<Flex direction="column" gap={1}>
|
||||
<Flex align="center" gap={2} color="base.200">
|
||||
<Icon as={PiSquaresFourFill} boxSize={4} />
|
||||
<Text fontWeight="bold" fontSize="sm">
|
||||
{t('hotkeys.canvas.settings.grid')}
|
||||
</Text>
|
||||
</Flex>
|
||||
<CanvasSettingsSnapToGridCheckbox />
|
||||
<CanvasSettingsDynamicGridSwitch />
|
||||
<CanvasSettingsRuleOfThirdsSwitch />
|
||||
</Flex>
|
||||
<CanvasSettingsSnapToGridCheckbox />
|
||||
<CanvasSettingsDynamicGridSwitch />
|
||||
<CanvasSettingsRuleOfThirdsSwitch />
|
||||
</Flex>
|
||||
|
||||
<DebugSettings />
|
||||
</Flex>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
<DebugSettings />
|
||||
</Flex>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Portal>
|
||||
</Popover>
|
||||
);
|
||||
});
|
||||
|
||||
@@ -707,10 +707,9 @@ describe('StagingAreaApi', () => {
|
||||
|
||||
// Should end up with the last set of items
|
||||
expect(api.$items.get()).toBe(items2);
|
||||
// The selectedItemId retains the old value (1) but $selectedItem will be null
|
||||
// because item 1 is no longer in the items list
|
||||
expect(api.$selectedItemId.get()).toBe(1);
|
||||
expect(api.$selectedItem.get()).toBe(null);
|
||||
// We expect the selection to have moved to the next existent item
|
||||
expect(api.$selectedItemId.get()).toBe(2);
|
||||
expect(api.$selectedItem.get()?.item.item_id).toBe(2);
|
||||
});
|
||||
|
||||
it('should handle multiple progress events for same item', () => {
|
||||
|
||||
@@ -361,6 +361,27 @@ export class StagingAreaApi {
|
||||
}
|
||||
}
|
||||
|
||||
const selectedItemId = this.$selectedItemId.get();
|
||||
if (selectedItemId !== null && !items.find(({ item_id }) => item_id === selectedItemId)) {
|
||||
// If the selected item no longer exists, select the next best item.
|
||||
// Prefer the next item in the list - must check oldItems to determine this
|
||||
const nextItemIndex = oldItems.findIndex(({ item_id }) => item_id === selectedItemId);
|
||||
if (nextItemIndex !== -1) {
|
||||
const nextItem = items[nextItemIndex] ?? items[nextItemIndex - 1];
|
||||
if (nextItem) {
|
||||
this.$selectedItemId.set(nextItem.item_id);
|
||||
}
|
||||
} else {
|
||||
// Next, if there is an in-progress item, select that.
|
||||
const inProgressItem = items.find(({ status }) => status === 'in_progress');
|
||||
if (inProgressItem) {
|
||||
this.$selectedItemId.set(inProgressItem.item_id);
|
||||
}
|
||||
// Finally just select the first item.
|
||||
this.$selectedItemId.set(items[0]?.item_id ?? null);
|
||||
}
|
||||
}
|
||||
|
||||
this.$items.set(items);
|
||||
};
|
||||
|
||||
|
||||
@@ -1,195 +0,0 @@
|
||||
import {
|
||||
CompositeSlider,
|
||||
FormControl,
|
||||
IconButton,
|
||||
NumberInput,
|
||||
NumberInputField,
|
||||
Popover,
|
||||
PopoverAnchor,
|
||||
PopoverArrow,
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { clamp } from 'es-toolkit/compat';
|
||||
import { useToolIsSelected } from 'features/controlLayers/components/Tool/hooks';
|
||||
import { selectCanvasSettingsSlice, settingsBrushWidthChanged } from 'features/controlLayers/store/canvasSettingsSlice';
|
||||
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||
import type { KeyboardEvent } from 'react';
|
||||
import { memo, useCallback, useEffect, useState } from 'react';
|
||||
import { PiCaretDownBold } from 'react-icons/pi';
|
||||
|
||||
const selectBrushWidth = createSelector(selectCanvasSettingsSlice, (settings) => settings.brushWidth);
|
||||
const formatPx = (v: number | string) => `${v} px`;
|
||||
|
||||
function mapSliderValueToRawValue(value: number) {
|
||||
if (value <= 40) {
|
||||
// 0 to 40 on the slider -> 1px to 50px
|
||||
return 1 + (49 * value) / 40;
|
||||
} else if (value <= 70) {
|
||||
// 40 to 70 on the slider -> 50px to 200px
|
||||
return 50 + (150 * (value - 40)) / 30;
|
||||
} else {
|
||||
// 70 to 100 on the slider -> 200px to 600px
|
||||
return 200 + (400 * (value - 70)) / 30;
|
||||
}
|
||||
}
|
||||
|
||||
function mapRawValueToSliderValue(value: number) {
|
||||
if (value <= 50) {
|
||||
// 1px to 50px -> 0 to 40 on the slider
|
||||
return ((value - 1) * 40) / 49;
|
||||
} else if (value <= 200) {
|
||||
// 50px to 200px -> 40 to 70 on the slider
|
||||
return 40 + ((value - 50) * 30) / 150;
|
||||
} else {
|
||||
// 200px to 600px -> 70 to 100 on the slider
|
||||
return 70 + ((value - 200) * 30) / 400;
|
||||
}
|
||||
}
|
||||
|
||||
function formatSliderValue(value: number) {
|
||||
return `${String(mapSliderValueToRawValue(value))} px`;
|
||||
}
|
||||
|
||||
const marks = [
|
||||
mapRawValueToSliderValue(1),
|
||||
mapRawValueToSliderValue(50),
|
||||
mapRawValueToSliderValue(200),
|
||||
mapRawValueToSliderValue(600),
|
||||
];
|
||||
|
||||
const sliderDefaultValue = mapRawValueToSliderValue(50);
|
||||
|
||||
export const ToolBrushWidth = memo(() => {
|
||||
const dispatch = useAppDispatch();
|
||||
const isSelected = useToolIsSelected('brush');
|
||||
const width = useAppSelector(selectBrushWidth);
|
||||
const [localValue, setLocalValue] = useState(width);
|
||||
const onChange = useCallback(
|
||||
(v: number) => {
|
||||
dispatch(settingsBrushWidthChanged(clamp(Math.round(v), 1, 600)));
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
const increment = useCallback(() => {
|
||||
let newWidth = Math.round(width * 1.15);
|
||||
if (newWidth === width) {
|
||||
newWidth += 1;
|
||||
}
|
||||
onChange(newWidth);
|
||||
}, [onChange, width]);
|
||||
|
||||
const decrement = useCallback(() => {
|
||||
let newWidth = Math.round(width * 0.85);
|
||||
if (newWidth === width) {
|
||||
newWidth -= 1;
|
||||
}
|
||||
onChange(newWidth);
|
||||
}, [onChange, width]);
|
||||
|
||||
const onChangeSlider = useCallback(
|
||||
(value: number) => {
|
||||
onChange(mapSliderValueToRawValue(value));
|
||||
},
|
||||
[onChange]
|
||||
);
|
||||
|
||||
const onBlur = useCallback(() => {
|
||||
if (isNaN(Number(localValue))) {
|
||||
onChange(50);
|
||||
setLocalValue(50);
|
||||
} else {
|
||||
onChange(localValue);
|
||||
}
|
||||
}, [localValue, onChange]);
|
||||
|
||||
const onChangeNumberInput = useCallback((valueAsString: string, valueAsNumber: number) => {
|
||||
setLocalValue(valueAsNumber);
|
||||
}, []);
|
||||
|
||||
const onKeyDown = useCallback(
|
||||
(e: KeyboardEvent<HTMLInputElement>) => {
|
||||
if (e.key === 'Enter') {
|
||||
onBlur();
|
||||
}
|
||||
},
|
||||
[onBlur]
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
setLocalValue(width);
|
||||
}, [width]);
|
||||
|
||||
useRegisteredHotkeys({
|
||||
id: 'decrementToolWidth',
|
||||
category: 'canvas',
|
||||
callback: decrement,
|
||||
options: { enabled: isSelected },
|
||||
dependencies: [decrement, isSelected],
|
||||
});
|
||||
useRegisteredHotkeys({
|
||||
id: 'incrementToolWidth',
|
||||
category: 'canvas',
|
||||
callback: increment,
|
||||
options: { enabled: isSelected },
|
||||
dependencies: [increment, isSelected],
|
||||
});
|
||||
|
||||
return (
|
||||
<Popover>
|
||||
<FormControl w="min-content" gap={2}>
|
||||
<PopoverAnchor>
|
||||
<NumberInput
|
||||
variant="outline"
|
||||
display="flex"
|
||||
alignItems="center"
|
||||
min={1}
|
||||
max={600}
|
||||
value={localValue}
|
||||
onChange={onChangeNumberInput}
|
||||
onBlur={onBlur}
|
||||
w="76px"
|
||||
format={formatPx}
|
||||
defaultValue={50}
|
||||
onKeyDown={onKeyDown}
|
||||
clampValueOnBlur={false}
|
||||
>
|
||||
<NumberInputField _focusVisible={{ zIndex: 0 }} title="" paddingInlineEnd={7} />
|
||||
<PopoverTrigger>
|
||||
<IconButton
|
||||
aria-label="open-slider"
|
||||
icon={<PiCaretDownBold />}
|
||||
size="sm"
|
||||
variant="link"
|
||||
position="absolute"
|
||||
insetInlineEnd={0}
|
||||
h="full"
|
||||
/>
|
||||
</PopoverTrigger>
|
||||
</NumberInput>
|
||||
</PopoverAnchor>
|
||||
</FormControl>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={mapRawValueToSliderValue(localValue)}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Popover>
|
||||
);
|
||||
});
|
||||
|
||||
ToolBrushWidth.displayName = 'ToolBrushWidth';
|
||||
@@ -1,198 +0,0 @@
|
||||
import {
|
||||
CompositeSlider,
|
||||
FormControl,
|
||||
IconButton,
|
||||
NumberInput,
|
||||
NumberInputField,
|
||||
Popover,
|
||||
PopoverAnchor,
|
||||
PopoverArrow,
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { clamp } from 'es-toolkit/compat';
|
||||
import { useToolIsSelected } from 'features/controlLayers/components/Tool/hooks';
|
||||
import {
|
||||
selectCanvasSettingsSlice,
|
||||
settingsEraserWidthChanged,
|
||||
} from 'features/controlLayers/store/canvasSettingsSlice';
|
||||
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||
import type { KeyboardEvent } from 'react';
|
||||
import { memo, useCallback, useEffect, useState } from 'react';
|
||||
import { PiCaretDownBold } from 'react-icons/pi';
|
||||
|
||||
const selectEraserWidth = createSelector(selectCanvasSettingsSlice, (settings) => settings.eraserWidth);
|
||||
const formatPx = (v: number | string) => `${v} px`;
|
||||
|
||||
function mapSliderValueToRawValue(value: number) {
|
||||
if (value <= 40) {
|
||||
// 0 to 40 on the slider -> 1px to 50px
|
||||
return 1 + (49 * value) / 40;
|
||||
} else if (value <= 70) {
|
||||
// 40 to 70 on the slider -> 50px to 200px
|
||||
return 50 + (150 * (value - 40)) / 30;
|
||||
} else {
|
||||
// 70 to 100 on the slider -> 200px to 600px
|
||||
return 200 + (400 * (value - 70)) / 30;
|
||||
}
|
||||
}
|
||||
|
||||
function mapRawValueToSliderValue(value: number) {
|
||||
if (value <= 50) {
|
||||
// 1px to 50px -> 0 to 40 on the slider
|
||||
return ((value - 1) * 40) / 49;
|
||||
} else if (value <= 200) {
|
||||
// 50px to 200px -> 40 to 70 on the slider
|
||||
return 40 + ((value - 50) * 30) / 150;
|
||||
} else {
|
||||
// 200px to 600px -> 70 to 100 on the slider
|
||||
return 70 + ((value - 200) * 30) / 400;
|
||||
}
|
||||
}
|
||||
|
||||
function formatSliderValue(value: number) {
|
||||
return `${String(mapSliderValueToRawValue(value))} px`;
|
||||
}
|
||||
|
||||
const marks = [
|
||||
mapRawValueToSliderValue(1),
|
||||
mapRawValueToSliderValue(50),
|
||||
mapRawValueToSliderValue(200),
|
||||
mapRawValueToSliderValue(600),
|
||||
];
|
||||
|
||||
const sliderDefaultValue = mapRawValueToSliderValue(50);
|
||||
|
||||
export const ToolEraserWidth = memo(() => {
|
||||
const dispatch = useAppDispatch();
|
||||
const isSelected = useToolIsSelected('eraser');
|
||||
const width = useAppSelector(selectEraserWidth);
|
||||
const [localValue, setLocalValue] = useState(width);
|
||||
const onChange = useCallback(
|
||||
(v: number) => {
|
||||
dispatch(settingsEraserWidthChanged(clamp(Math.round(v), 1, 600)));
|
||||
},
|
||||
[dispatch]
|
||||
);
|
||||
|
||||
const increment = useCallback(() => {
|
||||
let newWidth = Math.round(width * 1.15);
|
||||
if (newWidth === width) {
|
||||
newWidth += 1;
|
||||
}
|
||||
onChange(newWidth);
|
||||
}, [onChange, width]);
|
||||
|
||||
const decrement = useCallback(() => {
|
||||
let newWidth = Math.round(width * 0.85);
|
||||
if (newWidth === width) {
|
||||
newWidth -= 1;
|
||||
}
|
||||
onChange(newWidth);
|
||||
}, [onChange, width]);
|
||||
|
||||
const onChangeSlider = useCallback(
|
||||
(value: number) => {
|
||||
onChange(mapSliderValueToRawValue(value));
|
||||
},
|
||||
[onChange]
|
||||
);
|
||||
|
||||
const onBlur = useCallback(() => {
|
||||
if (isNaN(Number(localValue))) {
|
||||
onChange(50);
|
||||
setLocalValue(50);
|
||||
} else {
|
||||
onChange(localValue);
|
||||
}
|
||||
}, [localValue, onChange]);
|
||||
|
||||
const onChangeNumberInput = useCallback((valueAsString: string, valueAsNumber: number) => {
|
||||
setLocalValue(valueAsNumber);
|
||||
}, []);
|
||||
|
||||
const onKeyDown = useCallback(
|
||||
(e: KeyboardEvent<HTMLInputElement>) => {
|
||||
if (e.key === 'Enter') {
|
||||
onBlur();
|
||||
}
|
||||
},
|
||||
[onBlur]
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
setLocalValue(width);
|
||||
}, [width]);
|
||||
|
||||
useRegisteredHotkeys({
|
||||
id: 'decrementToolWidth',
|
||||
category: 'canvas',
|
||||
callback: decrement,
|
||||
options: { enabled: isSelected },
|
||||
dependencies: [decrement, isSelected],
|
||||
});
|
||||
useRegisteredHotkeys({
|
||||
id: 'incrementToolWidth',
|
||||
category: 'canvas',
|
||||
callback: increment,
|
||||
options: { enabled: isSelected },
|
||||
dependencies: [increment, isSelected],
|
||||
});
|
||||
|
||||
return (
|
||||
<Popover>
|
||||
<FormControl w="min-content" gap={2}>
|
||||
<PopoverAnchor>
|
||||
<NumberInput
|
||||
variant="outline"
|
||||
display="flex"
|
||||
alignItems="center"
|
||||
min={1}
|
||||
max={600}
|
||||
value={localValue}
|
||||
onChange={onChangeNumberInput}
|
||||
onBlur={onBlur}
|
||||
w="76px"
|
||||
format={formatPx}
|
||||
defaultValue={50}
|
||||
onKeyDown={onKeyDown}
|
||||
clampValueOnBlur={false}
|
||||
>
|
||||
<NumberInputField _focusVisible={{ zIndex: 0 }} title="" paddingInlineEnd={7} />
|
||||
<PopoverTrigger>
|
||||
<IconButton
|
||||
aria-label="open-slider"
|
||||
icon={<PiCaretDownBold />}
|
||||
size="sm"
|
||||
variant="link"
|
||||
position="absolute"
|
||||
insetInlineEnd={0}
|
||||
h="full"
|
||||
/>
|
||||
</PopoverTrigger>
|
||||
</NumberInput>
|
||||
</PopoverAnchor>
|
||||
</FormControl>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={mapRawValueToSliderValue(localValue)}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Popover>
|
||||
);
|
||||
});
|
||||
|
||||
ToolEraserWidth.displayName = 'ToolEraserWidth';
|
||||
@@ -6,6 +6,7 @@ import {
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
Portal,
|
||||
Tooltip,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
@@ -71,7 +72,7 @@ export const ToolFillColorPicker = memo(() => {
|
||||
return (
|
||||
<Popover isLazy>
|
||||
<PopoverTrigger>
|
||||
<Flex role="button" aria-label={t('controlLayers.fill.fillColor')} tabIndex={-1} w={8} h={8}>
|
||||
<Flex role="button" aria-label={t('controlLayers.fill.fillColor')} tabIndex={-1} minW={8} w={8} h={8}>
|
||||
<Tooltip label={tooltip}>
|
||||
<Flex alignItems="center" justifyContent="center" position="relative" w="full" h="full">
|
||||
<Box
|
||||
@@ -102,12 +103,14 @@ export const ToolFillColorPicker = memo(() => {
|
||||
</Tooltip>
|
||||
</Flex>
|
||||
</PopoverTrigger>
|
||||
<PopoverContent>
|
||||
<PopoverArrow />
|
||||
<PopoverBody minH={64}>
|
||||
<RgbaColorPicker color={activeColor} onChange={onColorChange} withNumberInput withSwatches />
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
<Portal>
|
||||
<PopoverContent>
|
||||
<PopoverArrow />
|
||||
<PopoverBody minH={64}>
|
||||
<RgbaColorPicker color={activeColor} onChange={onColorChange} withNumberInput withSwatches />
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Portal>
|
||||
</Popover>
|
||||
);
|
||||
});
|
||||
|
||||
@@ -1,19 +0,0 @@
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { ToolBrushWidth } from 'features/controlLayers/components/Tool/ToolBrushWidth';
|
||||
import { ToolEraserWidth } from 'features/controlLayers/components/Tool/ToolEraserWidth';
|
||||
import { useCanvasManager } from 'features/controlLayers/contexts/CanvasManagerProviderGate';
|
||||
import { memo } from 'react';
|
||||
|
||||
export const ToolSettings = memo(() => {
|
||||
const canvasManager = useCanvasManager();
|
||||
const tool = useStore(canvasManager.tool.$tool);
|
||||
if (tool === 'brush') {
|
||||
return <ToolBrushWidth />;
|
||||
}
|
||||
if (tool === 'eraser') {
|
||||
return <ToolEraserWidth />;
|
||||
}
|
||||
return null;
|
||||
});
|
||||
|
||||
ToolSettings.displayName = 'ToolSettings';
|
||||
@@ -0,0 +1,335 @@
|
||||
import {
|
||||
CompositeNumberInput,
|
||||
CompositeSlider,
|
||||
Flex,
|
||||
FormControl,
|
||||
IconButton,
|
||||
NumberInput,
|
||||
NumberInputField,
|
||||
Popover,
|
||||
PopoverAnchor,
|
||||
PopoverArrow,
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
Portal,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { createSelector } from '@reduxjs/toolkit';
|
||||
import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { clamp } from 'es-toolkit/compat';
|
||||
import {
|
||||
selectCanvasSettingsSlice,
|
||||
settingsBrushWidthChanged,
|
||||
settingsEraserWidthChanged,
|
||||
} from 'features/controlLayers/store/canvasSettingsSlice';
|
||||
import { useRegisteredHotkeys } from 'features/system/components/HotkeysModal/useHotkeyData';
|
||||
import type { KeyboardEvent } from 'react';
|
||||
import { memo, useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||
import { PiCaretDownBold } from 'react-icons/pi';
|
||||
|
||||
import { useToolIsSelected } from './hooks';
|
||||
|
||||
const formatPx = (v: number | string) => `${v} px`;
|
||||
|
||||
function mapSliderValueToRawValue(value: number) {
|
||||
if (value <= 40) {
|
||||
// 0 to 40 on the slider -> 1px to 50px
|
||||
return 1 + (49 * value) / 40;
|
||||
} else if (value <= 70) {
|
||||
// 40 to 70 on the slider -> 50px to 200px
|
||||
return 50 + (150 * (value - 40)) / 30;
|
||||
} else {
|
||||
// 70 to 100 on the slider -> 200px to 600px
|
||||
return 200 + (400 * (value - 70)) / 30;
|
||||
}
|
||||
}
|
||||
|
||||
function mapRawValueToSliderValue(value: number) {
|
||||
if (value <= 50) {
|
||||
// 1px to 50px -> 0 to 40 on the slider
|
||||
return ((value - 1) * 40) / 49;
|
||||
} else if (value <= 200) {
|
||||
// 50px to 200px -> 40 to 70 on the slider
|
||||
return 40 + ((value - 50) * 30) / 150;
|
||||
} else {
|
||||
// 200px to 600px -> 70 to 100 on the slider
|
||||
return 70 + ((value - 200) * 30) / 400;
|
||||
}
|
||||
}
|
||||
|
||||
function formatSliderValue(value: number) {
|
||||
return `${String(mapSliderValueToRawValue(value))} px`;
|
||||
}
|
||||
|
||||
const marks = [
|
||||
mapRawValueToSliderValue(1),
|
||||
mapRawValueToSliderValue(50),
|
||||
mapRawValueToSliderValue(200),
|
||||
mapRawValueToSliderValue(600),
|
||||
];
|
||||
|
||||
const sliderDefaultValue = mapRawValueToSliderValue(50);
|
||||
|
||||
const SLIDER_VS_DROPDOWN_CONTAINER_WIDTH_THRESHOLD = 280;
|
||||
|
||||
interface ToolWidthPickerComponentProps {
|
||||
localValue: number;
|
||||
onChangeSlider: (value: number) => void;
|
||||
onChangeInput: (value: number) => void;
|
||||
onBlur: () => void;
|
||||
onKeyDown: (value: KeyboardEvent<HTMLInputElement>) => void;
|
||||
}
|
||||
|
||||
const DropDownToolWidthPickerComponent = memo(
|
||||
({ localValue, onChangeSlider, onChangeInput, onKeyDown, onBlur }: ToolWidthPickerComponentProps) => {
|
||||
const onChangeNumberInput = useCallback(
|
||||
(valueAsString: string, valueAsNumber: number) => {
|
||||
onChangeInput(valueAsNumber);
|
||||
},
|
||||
[onChangeInput]
|
||||
);
|
||||
|
||||
return (
|
||||
<Popover>
|
||||
<FormControl w="min-content" gap={2} overflow="hidden">
|
||||
<PopoverAnchor>
|
||||
<NumberInput
|
||||
variant="outline"
|
||||
display="flex"
|
||||
alignItems="center"
|
||||
min={1}
|
||||
max={600}
|
||||
value={localValue}
|
||||
onChange={onChangeNumberInput}
|
||||
onBlur={onBlur}
|
||||
w={76}
|
||||
format={formatPx}
|
||||
defaultValue={50}
|
||||
onKeyDown={onKeyDown}
|
||||
clampValueOnBlur={false}
|
||||
>
|
||||
<NumberInputField _focusVisible={{ zIndex: 0 }} title="" paddingInlineEnd={7} />
|
||||
<PopoverTrigger>
|
||||
<IconButton
|
||||
aria-label="open-slider"
|
||||
icon={<PiCaretDownBold />}
|
||||
size="sm"
|
||||
variant="link"
|
||||
position="absolute"
|
||||
insetInlineEnd={0}
|
||||
h="full"
|
||||
/>
|
||||
</PopoverTrigger>
|
||||
</NumberInput>
|
||||
</PopoverAnchor>
|
||||
</FormControl>
|
||||
<Portal>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={mapRawValueToSliderValue(localValue)}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Portal>
|
||||
</Popover>
|
||||
);
|
||||
}
|
||||
);
|
||||
DropDownToolWidthPickerComponent.displayName = 'DropDownToolWidthPickerComponent';
|
||||
|
||||
const SliderToolWidthPickerComponent = memo(
|
||||
({ localValue, onChangeSlider, onChangeInput, onKeyDown, onBlur }: ToolWidthPickerComponentProps) => {
|
||||
return (
|
||||
<Flex w={SLIDER_VS_DROPDOWN_CONTAINER_WIDTH_THRESHOLD} gap={4}>
|
||||
<CompositeSlider
|
||||
w={200}
|
||||
h="unset"
|
||||
min={0}
|
||||
max={100}
|
||||
value={mapRawValueToSliderValue(localValue)}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
/>
|
||||
<CompositeNumberInput
|
||||
w={28}
|
||||
variant="outline"
|
||||
min={1}
|
||||
max={600}
|
||||
value={localValue}
|
||||
onChange={onChangeInput}
|
||||
onBlur={onBlur}
|
||||
onKeyDown={onKeyDown}
|
||||
format={formatPx}
|
||||
defaultValue={50}
|
||||
/>
|
||||
</Flex>
|
||||
);
|
||||
}
|
||||
);
|
||||
SliderToolWidthPickerComponent.displayName = 'SliderToolWidthPickerComponent';
|
||||
|
||||
const selectBrushWidth = createSelector(selectCanvasSettingsSlice, (settings) => settings.brushWidth);
|
||||
const selectEraserWidth = createSelector(selectCanvasSettingsSlice, (settings) => settings.eraserWidth);
|
||||
|
||||
export const ToolWidthPicker = memo(() => {
|
||||
const ref = useRef<HTMLDivElement>(null);
|
||||
const dispatch = useAppDispatch();
|
||||
const isBrushSelected = useToolIsSelected('brush');
|
||||
const isEraserSelected = useToolIsSelected('eraser');
|
||||
const isToolSelected = useMemo(() => {
|
||||
return isBrushSelected || isEraserSelected;
|
||||
}, [isBrushSelected, isEraserSelected]);
|
||||
const brushWidth = useAppSelector(selectBrushWidth);
|
||||
const eraserWidth = useAppSelector(selectEraserWidth);
|
||||
const width = useMemo(() => {
|
||||
if (isBrushSelected) {
|
||||
return brushWidth;
|
||||
}
|
||||
if (isEraserSelected) {
|
||||
return eraserWidth;
|
||||
}
|
||||
return 0;
|
||||
}, [isBrushSelected, isEraserSelected, brushWidth, eraserWidth]);
|
||||
const [localValue, setLocalValue] = useState(width);
|
||||
const [componentType, setComponentType] = useState<'slider' | 'dropdown' | null>(null);
|
||||
|
||||
useEffect(() => {
|
||||
const el = ref.current;
|
||||
if (!el) {
|
||||
return;
|
||||
}
|
||||
const observer = new ResizeObserver((entries) => {
|
||||
for (let entry of entries) {
|
||||
if (entry.contentRect.width > SLIDER_VS_DROPDOWN_CONTAINER_WIDTH_THRESHOLD) {
|
||||
setComponentType('slider');
|
||||
} else {
|
||||
setComponentType('dropdown');
|
||||
}
|
||||
}
|
||||
});
|
||||
observer.observe(el);
|
||||
|
||||
return () => {
|
||||
observer.disconnect();
|
||||
};
|
||||
}, []);
|
||||
|
||||
const onValueChange = useCallback(
|
||||
(value: number) => {
|
||||
if (isBrushSelected) {
|
||||
dispatch(settingsBrushWidthChanged(value));
|
||||
} else if (isEraserSelected) {
|
||||
dispatch(settingsEraserWidthChanged(value));
|
||||
}
|
||||
},
|
||||
[isBrushSelected, isEraserSelected, dispatch]
|
||||
);
|
||||
|
||||
const onChange = useCallback(
|
||||
(value: number) => {
|
||||
onValueChange(clamp(Math.round(value), 1, 600));
|
||||
},
|
||||
[onValueChange]
|
||||
);
|
||||
|
||||
const increment = useCallback(() => {
|
||||
let newWidth = Math.round(width * 1.15);
|
||||
if (newWidth === width) {
|
||||
newWidth += 1;
|
||||
}
|
||||
onChange(newWidth);
|
||||
}, [onChange, width]);
|
||||
|
||||
const decrement = useCallback(() => {
|
||||
let newWidth = Math.round(width * 0.85);
|
||||
if (newWidth === width) {
|
||||
newWidth -= 1;
|
||||
}
|
||||
onChange(newWidth);
|
||||
}, [onChange, width]);
|
||||
|
||||
const onChangeSlider = useCallback(
|
||||
(value: number) => {
|
||||
onChange(mapSliderValueToRawValue(value));
|
||||
},
|
||||
[onChange]
|
||||
);
|
||||
|
||||
const onChangeInput = useCallback((value: number) => {
|
||||
setLocalValue(value);
|
||||
}, []);
|
||||
|
||||
const onBlur = useCallback(() => {
|
||||
if (isNaN(Number(localValue))) {
|
||||
onChange(50);
|
||||
setLocalValue(50);
|
||||
} else {
|
||||
onChange(localValue);
|
||||
}
|
||||
}, [localValue, onChange]);
|
||||
|
||||
const onKeyDown = useCallback(
|
||||
(e: KeyboardEvent<HTMLInputElement>) => {
|
||||
if (e.key === 'Enter') {
|
||||
onBlur();
|
||||
}
|
||||
},
|
||||
[onBlur]
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
setLocalValue(width);
|
||||
}, [width]);
|
||||
|
||||
useRegisteredHotkeys({
|
||||
id: 'decrementToolWidth',
|
||||
category: 'canvas',
|
||||
callback: decrement,
|
||||
options: { enabled: isToolSelected },
|
||||
dependencies: [decrement, isToolSelected],
|
||||
});
|
||||
useRegisteredHotkeys({
|
||||
id: 'incrementToolWidth',
|
||||
category: 'canvas',
|
||||
callback: increment,
|
||||
options: { enabled: isToolSelected },
|
||||
dependencies: [increment, isToolSelected],
|
||||
});
|
||||
|
||||
return (
|
||||
<Flex ref={ref} alignItems="center" h="full" flexGrow={1} flexShrink={1} justifyContent="flex-start" px={4}>
|
||||
{componentType === 'slider' && (
|
||||
<SliderToolWidthPickerComponent
|
||||
localValue={localValue}
|
||||
onChangeSlider={onChangeSlider}
|
||||
onChangeInput={onChangeInput}
|
||||
onBlur={onBlur}
|
||||
onKeyDown={onKeyDown}
|
||||
/>
|
||||
)}
|
||||
{componentType === 'dropdown' && (
|
||||
<DropDownToolWidthPickerComponent
|
||||
localValue={localValue}
|
||||
onChangeSlider={onChangeSlider}
|
||||
onChangeInput={onChangeInput}
|
||||
onBlur={onBlur}
|
||||
onKeyDown={onKeyDown}
|
||||
/>
|
||||
)}
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
ToolWidthPicker.displayName = 'ToolWidthPicker';
|
||||
@@ -1,7 +1,8 @@
|
||||
import { Divider, Flex } from '@invoke-ai/ui-library';
|
||||
import { CanvasSettingsPopover } from 'features/controlLayers/components/Settings/CanvasSettingsPopover';
|
||||
import { useToolIsSelected } from 'features/controlLayers/components/Tool/hooks';
|
||||
import { ToolFillColorPicker } from 'features/controlLayers/components/Tool/ToolFillColorPicker';
|
||||
import { ToolSettings } from 'features/controlLayers/components/Tool/ToolSettings';
|
||||
import { ToolWidthPicker } from 'features/controlLayers/components/Tool/ToolWidthPicker';
|
||||
import { CanvasToolbarFitBboxToLayersButton } from 'features/controlLayers/components/Toolbar/CanvasToolbarFitBboxToLayersButton';
|
||||
import { CanvasToolbarFitBboxToMasksButton } from 'features/controlLayers/components/Toolbar/CanvasToolbarFitBboxToMasksButton';
|
||||
import { CanvasToolbarNewSessionMenuButton } from 'features/controlLayers/components/Toolbar/CanvasToolbarNewSessionMenuButton';
|
||||
@@ -20,9 +21,15 @@ import { useCanvasToggleNonRasterLayersHotkey } from 'features/controlLayers/hoo
|
||||
import { useCanvasTransformHotkey } from 'features/controlLayers/hooks/useCanvasTransformHotkey';
|
||||
import { useCanvasUndoRedoHotkeys } from 'features/controlLayers/hooks/useCanvasUndoRedoHotkeys';
|
||||
import { useNextPrevEntityHotkeys } from 'features/controlLayers/hooks/useNextPrevEntity';
|
||||
import { memo } from 'react';
|
||||
import { memo, useMemo } from 'react';
|
||||
|
||||
export const CanvasToolbar = memo(() => {
|
||||
const isBrushSelected = useToolIsSelected('brush');
|
||||
const isEraserSelected = useToolIsSelected('eraser');
|
||||
const showToolWithPicker = useMemo(() => {
|
||||
return isBrushSelected || isEraserSelected;
|
||||
}, [isBrushSelected, isEraserSelected]);
|
||||
|
||||
useCanvasResetLayerHotkey();
|
||||
useCanvasDeleteLayerHotkey();
|
||||
useCanvasUndoRedoHotkeys();
|
||||
@@ -36,9 +43,11 @@ export const CanvasToolbar = memo(() => {
|
||||
|
||||
return (
|
||||
<Flex w="full" gap={2} alignItems="center" px={2}>
|
||||
<ToolFillColorPicker />
|
||||
<ToolSettings />
|
||||
<Flex alignItems="center" h="full" flexGrow={1} justifyContent="flex-end">
|
||||
<Flex alignItems="center" h="full" flexGrow={1}>
|
||||
<ToolFillColorPicker />
|
||||
{showToolWithPicker && <ToolWidthPicker />}
|
||||
</Flex>
|
||||
<Flex alignItems="center" h="full">
|
||||
<CanvasToolbarScale />
|
||||
<CanvasToolbarResetViewButton />
|
||||
<CanvasToolbarFitBboxToLayersButton />
|
||||
|
||||
@@ -12,6 +12,7 @@ import {
|
||||
PopoverBody,
|
||||
PopoverContent,
|
||||
PopoverTrigger,
|
||||
Portal,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { round } from 'es-toolkit/compat';
|
||||
@@ -153,21 +154,23 @@ export const CanvasToolbarScale = memo(() => {
|
||||
</PopoverTrigger>
|
||||
</NumberInput>
|
||||
</PopoverAnchor>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={mapRawValueToSliderValue(localScale)}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
<Portal>
|
||||
<PopoverContent w={200} pt={0} pb={2} px={4}>
|
||||
<PopoverArrow />
|
||||
<PopoverBody>
|
||||
<CompositeSlider
|
||||
min={0}
|
||||
max={100}
|
||||
value={mapRawValueToSliderValue(localScale)}
|
||||
onChange={onChangeSlider}
|
||||
defaultValue={sliderDefaultValue}
|
||||
marks={marks}
|
||||
formatValue={formatSliderValue}
|
||||
alwaysShowMarks
|
||||
/>
|
||||
</PopoverBody>
|
||||
</PopoverContent>
|
||||
</Portal>
|
||||
</Popover>
|
||||
<ZoomInButton />
|
||||
</Flex>
|
||||
|
||||
@@ -30,6 +30,7 @@ import type {
|
||||
FluxKontextReferenceImageConfig,
|
||||
Gemini2_5ReferenceImageConfig,
|
||||
IPAdapterConfig,
|
||||
RegionalGuidanceIPAdapterConfig,
|
||||
T2IAdapterConfig,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import {
|
||||
@@ -38,6 +39,7 @@ import {
|
||||
initialFluxKontextReferenceImage,
|
||||
initialGemini2_5ReferenceImage,
|
||||
initialIPAdapter,
|
||||
initialRegionalGuidanceIPAdapter,
|
||||
initialT2IAdapter,
|
||||
} from 'features/controlLayers/store/util';
|
||||
import { zModelIdentifierField } from 'features/nodes/types/common';
|
||||
@@ -125,7 +127,7 @@ export const getDefaultRefImageConfig = (
|
||||
return config;
|
||||
};
|
||||
|
||||
export const getDefaultRegionalGuidanceRefImageConfig = (getState: AppGetState): IPAdapterConfig => {
|
||||
export const getDefaultRegionalGuidanceRefImageConfig = (getState: AppGetState): RegionalGuidanceIPAdapterConfig => {
|
||||
// Regional guidance ref images do not support ChatGPT-4o, so we always return the IP Adapter config.
|
||||
const state = getState();
|
||||
|
||||
@@ -138,7 +140,7 @@ export const getDefaultRegionalGuidanceRefImageConfig = (getState: AppGetState):
|
||||
const modelConfig = ipAdapterModelConfigs.find((m) => m.base === base);
|
||||
|
||||
// Clone the initial IP Adapter config and set the model if available.
|
||||
const config = deepClone(initialIPAdapter);
|
||||
const config = deepClone(initialRegionalGuidanceIPAdapter);
|
||||
|
||||
if (modelConfig) {
|
||||
config.model = zModelIdentifierField.parse(modelConfig);
|
||||
|
||||
@@ -32,7 +32,12 @@ import type {
|
||||
RefImageState,
|
||||
RegionalGuidanceRefImageState,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageObject, imageDTOToImageWithDims, initialControlNet } from 'features/controlLayers/store/util';
|
||||
import {
|
||||
imageDTOToCroppableImage,
|
||||
imageDTOToImageObject,
|
||||
imageDTOToImageWithDims,
|
||||
initialControlNet,
|
||||
} from 'features/controlLayers/store/util';
|
||||
import { selectAutoAddBoardId } from 'features/gallery/store/gallerySelectors';
|
||||
import type { BoardId } from 'features/gallery/store/types';
|
||||
import { Graph } from 'features/nodes/util/graph/generation/Graph';
|
||||
@@ -209,7 +214,7 @@ export const useNewGlobalReferenceImageFromBbox = () => {
|
||||
const overrides: Partial<RefImageState> = {
|
||||
config: {
|
||||
...getDefaultRefImageConfig(getState),
|
||||
image: imageDTOToImageWithDims(imageDTO),
|
||||
image: imageDTOToCroppableImage(imageDTO),
|
||||
},
|
||||
};
|
||||
dispatch(refImageAdded({ overrides }));
|
||||
@@ -312,7 +317,7 @@ export const usePullBboxIntoGlobalReferenceImage = (id: string) => {
|
||||
|
||||
const arg = useMemo<UseSaveCanvasArg>(() => {
|
||||
const onSave = (imageDTO: ImageDTO, _: Rect) => {
|
||||
dispatch(refImageImageChanged({ id, imageDTO }));
|
||||
dispatch(refImageImageChanged({ id, croppableImage: imageDTOToCroppableImage(imageDTO) }));
|
||||
};
|
||||
|
||||
return {
|
||||
|
||||
@@ -475,7 +475,7 @@ export abstract class CanvasEntityAdapterBase<T extends CanvasEntityState, U ext
|
||||
* to hide this entity.
|
||||
*/
|
||||
const filteringAdapter = this.manager.stateApi.$filteringAdapter.get();
|
||||
if (filteringAdapter && filteringAdapter !== this) {
|
||||
if (filteringAdapter && filteringAdapter.id !== this.id) {
|
||||
this.setVisibility(false);
|
||||
return;
|
||||
}
|
||||
@@ -492,7 +492,7 @@ export abstract class CanvasEntityAdapterBase<T extends CanvasEntityState, U ext
|
||||
}
|
||||
|
||||
const segmentingAdapter = this.manager.stateApi.$segmentingAdapter.get();
|
||||
if (segmentingAdapter && segmentingAdapter !== this) {
|
||||
if (segmentingAdapter && segmentingAdapter.id !== this.id) {
|
||||
this.setVisibility(false);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -72,7 +72,7 @@ export class CanvasEntityAdapterControlLayer extends CanvasEntityAdapterBase<
|
||||
this.log.trace({ rect }, 'Getting canvas');
|
||||
// The opacity may have been changed in response to user selecting a different entity category, so we must restore
|
||||
// the original opacity before rendering the canvas
|
||||
const attrs: GroupConfig = { opacity: this.state.opacity, filters: [] };
|
||||
const attrs: GroupConfig = { opacity: this.state.opacity };
|
||||
const canvas = this.renderer.getCanvas({ rect, attrs });
|
||||
return canvas;
|
||||
};
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import { omit } from 'es-toolkit/compat';
|
||||
import { omit, throttle } from 'es-toolkit/compat';
|
||||
import { CanvasEntityAdapterBase } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityAdapterBase';
|
||||
import { CanvasEntityBufferObjectRenderer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityBufferObjectRenderer';
|
||||
import { CanvasEntityFilterer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityFilterer';
|
||||
@@ -6,6 +6,7 @@ import { CanvasEntityObjectRenderer } from 'features/controlLayers/konva/CanvasE
|
||||
import { CanvasEntityTransformer } from 'features/controlLayers/konva/CanvasEntity/CanvasEntityTransformer';
|
||||
import type { CanvasManager } from 'features/controlLayers/konva/CanvasManager';
|
||||
import { CanvasSegmentAnythingModule } from 'features/controlLayers/konva/CanvasSegmentAnythingModule';
|
||||
import { AdjustmentsCurvesFilter, AdjustmentsSimpleFilter, buildCurveLUT } from 'features/controlLayers/konva/filters';
|
||||
import type { CanvasEntityIdentifier, CanvasRasterLayerState, Rect } from 'features/controlLayers/store/types';
|
||||
import type { GroupConfig } from 'konva/lib/Group';
|
||||
import type { JsonObject } from 'type-fest';
|
||||
@@ -59,13 +60,18 @@ export class CanvasEntityAdapterRasterLayer extends CanvasEntityAdapterBase<
|
||||
if (!prevState || this.state.opacity !== prevState.opacity) {
|
||||
this.syncOpacity();
|
||||
}
|
||||
|
||||
// Apply per-layer adjustments as a Konva filter
|
||||
if (!prevState || this.haveAdjustmentsChanged(prevState, this.state)) {
|
||||
this.syncAdjustmentsFilter();
|
||||
}
|
||||
};
|
||||
|
||||
getCanvas = (rect?: Rect): HTMLCanvasElement => {
|
||||
this.log.trace({ rect }, 'Getting canvas');
|
||||
// The opacity may have been changed in response to user selecting a different entity category, so we must restore
|
||||
// the original opacity before rendering the canvas
|
||||
const attrs: GroupConfig = { opacity: this.state.opacity, filters: [] };
|
||||
const attrs: GroupConfig = { opacity: this.state.opacity };
|
||||
const canvas = this.renderer.getCanvas({ rect, attrs });
|
||||
return canvas;
|
||||
};
|
||||
@@ -74,4 +80,79 @@ export class CanvasEntityAdapterRasterLayer extends CanvasEntityAdapterBase<
|
||||
const keysToOmit: (keyof CanvasRasterLayerState)[] = ['name', 'isLocked'];
|
||||
return omit(this.state, keysToOmit);
|
||||
};
|
||||
|
||||
private syncAdjustmentsFilter = () => {
|
||||
const a = this.state.adjustments;
|
||||
const apply = !!a && a.enabled;
|
||||
// The filter operates on the renderer's object group; we can set filters at the group level via renderer
|
||||
const group = this.renderer.konva.objectGroup;
|
||||
if (apply) {
|
||||
const filters = group.filters() ?? [];
|
||||
let nextFilters = filters.filter((f) => f !== AdjustmentsSimpleFilter && f !== AdjustmentsCurvesFilter);
|
||||
if (a.mode === 'simple') {
|
||||
group.setAttr('adjustmentsSimple', a.simple);
|
||||
group.setAttr('adjustmentsCurves', null);
|
||||
nextFilters = [...nextFilters, AdjustmentsSimpleFilter];
|
||||
} else {
|
||||
// Build LUTs and set curves attr
|
||||
const master = buildCurveLUT(a.curves.master);
|
||||
const r = buildCurveLUT(a.curves.r);
|
||||
const g = buildCurveLUT(a.curves.g);
|
||||
const b = buildCurveLUT(a.curves.b);
|
||||
group.setAttr('adjustmentsCurves', { master, r, g, b });
|
||||
group.setAttr('adjustmentsSimple', null);
|
||||
nextFilters = [...nextFilters, AdjustmentsCurvesFilter];
|
||||
}
|
||||
group.filters(nextFilters);
|
||||
this._throttledCacheRefresh();
|
||||
} else {
|
||||
// Remove our filter if present
|
||||
const filters = (group.filters() ?? []).filter(
|
||||
(f) => f !== AdjustmentsSimpleFilter && f !== AdjustmentsCurvesFilter
|
||||
);
|
||||
group.filters(filters);
|
||||
group.setAttr('adjustmentsSimple', null);
|
||||
group.setAttr('adjustmentsCurves', null);
|
||||
this._throttledCacheRefresh();
|
||||
}
|
||||
};
|
||||
|
||||
private _throttledCacheRefresh = throttle(() => this.renderer.syncKonvaCache(true), 50);
|
||||
|
||||
private haveAdjustmentsChanged = (prevState: CanvasRasterLayerState, currState: CanvasRasterLayerState): boolean => {
|
||||
const pa = prevState.adjustments;
|
||||
const ca = currState.adjustments;
|
||||
if (pa === ca) {
|
||||
return false;
|
||||
}
|
||||
if (!pa || !ca) {
|
||||
return true;
|
||||
}
|
||||
if (pa.enabled !== ca.enabled) {
|
||||
return true;
|
||||
}
|
||||
if (pa.mode !== ca.mode) {
|
||||
return true;
|
||||
}
|
||||
// simple params
|
||||
const ps = pa.simple;
|
||||
const cs = ca.simple;
|
||||
if (
|
||||
ps.brightness !== cs.brightness ||
|
||||
ps.contrast !== cs.contrast ||
|
||||
ps.saturation !== cs.saturation ||
|
||||
ps.temperature !== cs.temperature ||
|
||||
ps.tint !== cs.tint ||
|
||||
ps.sharpness !== cs.sharpness
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
// curves reference (UI not implemented yet) - if arrays differ by ref, consider changed
|
||||
const pc = pa.curves;
|
||||
const cc = ca.curves;
|
||||
if (pc !== cc) {
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
};
|
||||
}
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -3,6 +3,10 @@
|
||||
* https://konvajs.org/docs/filters/Custom_Filter.html
|
||||
*/
|
||||
|
||||
import { clamp } from 'es-toolkit/compat';
|
||||
import { zCurvesAdjustmentsLUTs, zSimpleAdjustmentsConfig } from 'features/controlLayers/store/types';
|
||||
import type Konva from 'konva';
|
||||
|
||||
/**
|
||||
* Calculates the lightness (HSL) of a given pixel and sets the alpha channel to that value.
|
||||
* This is useful for edge maps and other masks, to make the black areas transparent.
|
||||
@@ -20,3 +24,177 @@ export const LightnessToAlphaFilter = (imageData: ImageData): void => {
|
||||
imageData.data[i * 4 + 3] = Math.min(a, (cMin + cMax) / 2);
|
||||
}
|
||||
};
|
||||
|
||||
/**
|
||||
* Per-layer simple adjustments filter (brightness, contrast, saturation, temp, tint, sharpness).
|
||||
*
|
||||
* Parameters are read from the Konva node attr `adjustmentsSimple` set by the adapter.
|
||||
*/
|
||||
export const AdjustmentsSimpleFilter = function (this: Konva.Node, imageData: ImageData): void {
|
||||
const paramsRaw = this.getAttr('adjustmentsSimple');
|
||||
const parseResult = zSimpleAdjustmentsConfig.safeParse(paramsRaw);
|
||||
if (!parseResult.success) {
|
||||
return;
|
||||
}
|
||||
const params = parseResult.data;
|
||||
|
||||
const { brightness, contrast, saturation, temperature, tint, sharpness } = params;
|
||||
|
||||
const data = imageData.data;
|
||||
const len = data.length / 4;
|
||||
const width = imageData.width;
|
||||
const height = imageData.height;
|
||||
|
||||
// Precompute factors
|
||||
const brightnessShift = brightness * 255; // additive shift
|
||||
const contrastFactor = 1 + contrast; // scale around 128
|
||||
|
||||
// Temperature/Tint multipliers
|
||||
const tempK = 0.5;
|
||||
const tintK = 0.5;
|
||||
const rTempMul = 1 + temperature * tempK;
|
||||
const bTempMul = 1 - temperature * tempK;
|
||||
// Tint: green <-> magenta. Positive = magenta (R/B up, G down). Negative = green (G up, R/B down).
|
||||
const t = clamp(tint, -1, 1) * tintK;
|
||||
const mag = Math.abs(t);
|
||||
const rTintMul = t >= 0 ? 1 + mag : 1 - mag;
|
||||
const gTintMul = t >= 0 ? 1 - mag : 1 + mag;
|
||||
const bTintMul = t >= 0 ? 1 + mag : 1 - mag;
|
||||
|
||||
// Saturation matrix (HSL-based approximation via luma coefficients)
|
||||
const lumaR = 0.2126;
|
||||
const lumaG = 0.7152;
|
||||
const lumaB = 0.0722;
|
||||
const S = 1 + saturation; // 0..2
|
||||
const m00 = lumaR * (1 - S) + S;
|
||||
const m01 = lumaG * (1 - S);
|
||||
const m02 = lumaB * (1 - S);
|
||||
const m10 = lumaR * (1 - S);
|
||||
const m11 = lumaG * (1 - S) + S;
|
||||
const m12 = lumaB * (1 - S);
|
||||
const m20 = lumaR * (1 - S);
|
||||
const m21 = lumaG * (1 - S);
|
||||
const m22 = lumaB * (1 - S) + S;
|
||||
|
||||
// First pass: apply per-pixel color adjustments (excluding sharpness)
|
||||
for (let i = 0; i < len; i++) {
|
||||
const idx = i * 4;
|
||||
let r = data[idx + 0] as number;
|
||||
let g = data[idx + 1] as number;
|
||||
let b = data[idx + 2] as number;
|
||||
const a = data[idx + 3] as number;
|
||||
|
||||
// Brightness (additive)
|
||||
r = r + brightnessShift;
|
||||
g = g + brightnessShift;
|
||||
b = b + brightnessShift;
|
||||
|
||||
// Contrast around mid-point 128
|
||||
r = (r - 128) * contrastFactor + 128;
|
||||
g = (g - 128) * contrastFactor + 128;
|
||||
b = (b - 128) * contrastFactor + 128;
|
||||
|
||||
// Temperature (R/B axis) and Tint (G vs Magenta)
|
||||
r = r * rTempMul * rTintMul;
|
||||
g = g * gTintMul;
|
||||
b = b * bTempMul * bTintMul;
|
||||
|
||||
// Saturation via matrix
|
||||
const r2 = r * m00 + g * m01 + b * m02;
|
||||
const g2 = r * m10 + g * m11 + b * m12;
|
||||
const b2 = r * m20 + g * m21 + b * m22;
|
||||
|
||||
data[idx + 0] = clamp(r2, 0, 255);
|
||||
data[idx + 1] = clamp(g2, 0, 255);
|
||||
data[idx + 2] = clamp(b2, 0, 255);
|
||||
data[idx + 3] = a;
|
||||
}
|
||||
|
||||
// Optional sharpen (simple unsharp mask with 3x3 kernel)
|
||||
if (Math.abs(sharpness) > 1e-3 && width > 2 && height > 2) {
|
||||
const src = new Uint8ClampedArray(data); // copy of modified data
|
||||
const a = Math.max(-1, Math.min(1, sharpness)) * 0.5; // amount
|
||||
const center = 1 + 4 * a;
|
||||
const neighbor = -a;
|
||||
for (let y = 1; y < height - 1; y++) {
|
||||
for (let x = 1; x < width - 1; x++) {
|
||||
const idx = (y * width + x) * 4;
|
||||
for (let c = 0; c < 3; c++) {
|
||||
const centerPx = src[idx + c] ?? 0;
|
||||
const leftPx = src[idx - 4 + c] ?? 0;
|
||||
const rightPx = src[idx + 4 + c] ?? 0;
|
||||
const topPx = src[idx - width * 4 + c] ?? 0;
|
||||
const bottomPx = src[idx + width * 4 + c] ?? 0;
|
||||
const v = centerPx * center + leftPx * neighbor + rightPx * neighbor + topPx * neighbor + bottomPx * neighbor;
|
||||
data[idx + c] = clamp(v, 0, 255);
|
||||
}
|
||||
// preserve alpha
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
// Build a 256-length LUT from 0..255 control points (linear interpolation for v1)
|
||||
export const buildCurveLUT = (points: Array<[number, number]>): number[] => {
|
||||
if (!points || points.length === 0) {
|
||||
return Array.from({ length: 256 }, (_, i) => i);
|
||||
}
|
||||
const pts = points
|
||||
.map(([x, y]) => [clamp(Math.round(x), 0, 255), clamp(Math.round(y), 0, 255)] as [number, number])
|
||||
.sort((a, b) => a[0] - b[0]);
|
||||
if ((pts[0]?.[0] ?? 0) !== 0) {
|
||||
pts.unshift([0, pts[0]?.[1] ?? 0]);
|
||||
}
|
||||
const last = pts[pts.length - 1];
|
||||
if ((last?.[0] ?? 255) !== 255) {
|
||||
pts.push([255, last?.[1] ?? 255]);
|
||||
}
|
||||
const lut = new Array<number>(256);
|
||||
let j = 0;
|
||||
for (let x = 0; x <= 255; x++) {
|
||||
while (j < pts.length - 2 && x > (pts[j + 1]?.[0] ?? 255)) {
|
||||
j++;
|
||||
}
|
||||
const p0 = pts[j] ?? [0, 0];
|
||||
const p1 = pts[j + 1] ?? [255, 255];
|
||||
const [x0, y0] = p0;
|
||||
const [x1, y1] = p1;
|
||||
const t = x1 === x0 ? 0 : (x - x0) / (x1 - x0);
|
||||
const y = y0 + (y1 - y0) * t;
|
||||
lut[x] = clamp(Math.round(y), 0, 255);
|
||||
}
|
||||
return lut;
|
||||
};
|
||||
|
||||
/**
|
||||
* Per-layer curves adjustments filter (master, r, g, b)
|
||||
*
|
||||
* Parameters are read from the Konva node attr `adjustmentsCurves` set by the adapter.
|
||||
*/
|
||||
export const AdjustmentsCurvesFilter = function (this: Konva.Node, imageData: ImageData): void {
|
||||
const paramsRaw = this.getAttr('adjustmentsCurves');
|
||||
const parseResult = zCurvesAdjustmentsLUTs.safeParse(paramsRaw);
|
||||
if (!parseResult.success) {
|
||||
return;
|
||||
}
|
||||
const params = parseResult.data;
|
||||
|
||||
const { master, r, g, b } = params;
|
||||
if (!master || !r || !g || !b) {
|
||||
return;
|
||||
}
|
||||
const data = imageData.data;
|
||||
const len = data.length / 4;
|
||||
for (let i = 0; i < len; i++) {
|
||||
const idx = i * 4;
|
||||
const r0 = data[idx + 0] as number;
|
||||
const g0 = data[idx + 1] as number;
|
||||
const b0 = data[idx + 2] as number;
|
||||
const rm = master[r0] ?? r0;
|
||||
const gm = master[g0] ?? g0;
|
||||
const bm = master[b0] ?? b0;
|
||||
data[idx + 0] = clamp(r[rm] ?? rm, 0, 255);
|
||||
data[idx + 1] = clamp(g[gm] ?? gm, 0, 255);
|
||||
data[idx + 2] = clamp(b[bm] ?? bm, 0, 255);
|
||||
}
|
||||
};
|
||||
|
||||
@@ -19,12 +19,16 @@ import type {
|
||||
CanvasEntityType,
|
||||
CanvasInpaintMaskState,
|
||||
CanvasMetadata,
|
||||
ChannelName,
|
||||
ChannelPoints,
|
||||
ControlLoRAConfig,
|
||||
EntityMovedByPayload,
|
||||
FillStyle,
|
||||
FLUXReduxImageInfluence,
|
||||
RasterLayerAdjustments,
|
||||
RegionalGuidanceRefImageState,
|
||||
RgbColor,
|
||||
SimpleAdjustmentsConfig,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import {
|
||||
calculateNewSize,
|
||||
@@ -78,10 +82,10 @@ import {
|
||||
IMAGEN_ASPECT_RATIOS,
|
||||
isChatGPT4oAspectRatioID,
|
||||
isFluxKontextAspectRatioID,
|
||||
isFLUXReduxConfig,
|
||||
isGemini2_5AspectRatioID,
|
||||
isImagenAspectRatioID,
|
||||
isIPAdapterConfig,
|
||||
isRegionalGuidanceFLUXReduxConfig,
|
||||
isRegionalGuidanceIPAdapterConfig,
|
||||
zCanvasState,
|
||||
} from './types';
|
||||
import {
|
||||
@@ -95,7 +99,9 @@ import {
|
||||
initialControlNet,
|
||||
initialFLUXRedux,
|
||||
initialIPAdapter,
|
||||
initialRegionalGuidanceIPAdapter,
|
||||
initialT2IAdapter,
|
||||
makeDefaultRasterLayerAdjustments,
|
||||
} from './util';
|
||||
|
||||
const slice = createSlice({
|
||||
@@ -104,6 +110,96 @@ const slice = createSlice({
|
||||
reducers: {
|
||||
// undoable canvas state
|
||||
//#region Raster layers
|
||||
rasterLayerAdjustmentsSet: (
|
||||
state,
|
||||
action: PayloadAction<EntityIdentifierPayload<{ adjustments: RasterLayerAdjustments | null }, 'raster_layer'>>
|
||||
) => {
|
||||
const { entityIdentifier, adjustments } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer) {
|
||||
return;
|
||||
}
|
||||
if (adjustments === null) {
|
||||
delete layer.adjustments;
|
||||
return;
|
||||
}
|
||||
if (!layer.adjustments) {
|
||||
layer.adjustments = makeDefaultRasterLayerAdjustments(adjustments.mode ?? 'simple');
|
||||
}
|
||||
layer.adjustments = merge(layer.adjustments, adjustments);
|
||||
},
|
||||
rasterLayerAdjustmentsReset: (state, action: PayloadAction<EntityIdentifierPayload<void, 'raster_layer'>>) => {
|
||||
const { entityIdentifier } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer?.adjustments) {
|
||||
return;
|
||||
}
|
||||
layer.adjustments.simple = makeDefaultRasterLayerAdjustments('simple').simple;
|
||||
layer.adjustments.curves = makeDefaultRasterLayerAdjustments('curves').curves;
|
||||
},
|
||||
rasterLayerAdjustmentsCancel: (state, action: PayloadAction<EntityIdentifierPayload<void, 'raster_layer'>>) => {
|
||||
const { entityIdentifier } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer) {
|
||||
return;
|
||||
}
|
||||
delete layer.adjustments;
|
||||
},
|
||||
rasterLayerAdjustmentsModeChanged: (
|
||||
state,
|
||||
action: PayloadAction<EntityIdentifierPayload<{ mode: 'simple' | 'curves' }, 'raster_layer'>>
|
||||
) => {
|
||||
const { entityIdentifier, mode } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer?.adjustments) {
|
||||
return;
|
||||
}
|
||||
layer.adjustments.mode = mode;
|
||||
},
|
||||
rasterLayerAdjustmentsSimpleUpdated: (
|
||||
state,
|
||||
action: PayloadAction<EntityIdentifierPayload<{ simple: Partial<SimpleAdjustmentsConfig> }, 'raster_layer'>>
|
||||
) => {
|
||||
const { entityIdentifier, simple } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer?.adjustments) {
|
||||
return;
|
||||
}
|
||||
layer.adjustments.simple = merge(layer.adjustments.simple, simple);
|
||||
},
|
||||
rasterLayerAdjustmentsCurvesUpdated: (
|
||||
state,
|
||||
action: PayloadAction<EntityIdentifierPayload<{ channel: ChannelName; points: ChannelPoints }, 'raster_layer'>>
|
||||
) => {
|
||||
const { entityIdentifier, channel, points } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer?.adjustments) {
|
||||
return;
|
||||
}
|
||||
layer.adjustments.curves[channel] = points;
|
||||
},
|
||||
rasterLayerAdjustmentsEnabledToggled: (
|
||||
state,
|
||||
action: PayloadAction<EntityIdentifierPayload<void, 'raster_layer'>>
|
||||
) => {
|
||||
const { entityIdentifier } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer?.adjustments) {
|
||||
return;
|
||||
}
|
||||
layer.adjustments.enabled = !layer.adjustments.enabled;
|
||||
},
|
||||
rasterLayerAdjustmentsCollapsedToggled: (
|
||||
state,
|
||||
action: PayloadAction<EntityIdentifierPayload<void, 'raster_layer'>>
|
||||
) => {
|
||||
const { entityIdentifier } = action.payload;
|
||||
const layer = selectEntity(state, entityIdentifier);
|
||||
if (!layer?.adjustments) {
|
||||
return;
|
||||
}
|
||||
layer.adjustments.collapsed = !layer.adjustments.collapsed;
|
||||
},
|
||||
rasterLayerAdded: {
|
||||
reducer: (
|
||||
state,
|
||||
@@ -709,7 +805,7 @@ const slice = createSlice({
|
||||
if (!entity) {
|
||||
return;
|
||||
}
|
||||
const config = { id: referenceImageId, config: deepClone(initialIPAdapter) };
|
||||
const config = { id: referenceImageId, config: deepClone(initialRegionalGuidanceIPAdapter) };
|
||||
merge(config, overrides);
|
||||
entity.referenceImages.push(config);
|
||||
},
|
||||
@@ -752,7 +848,7 @@ const slice = createSlice({
|
||||
if (!referenceImage) {
|
||||
return;
|
||||
}
|
||||
if (!isIPAdapterConfig(referenceImage.config)) {
|
||||
if (!isRegionalGuidanceIPAdapterConfig(referenceImage.config)) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -769,7 +865,7 @@ const slice = createSlice({
|
||||
if (!referenceImage) {
|
||||
return;
|
||||
}
|
||||
if (!isIPAdapterConfig(referenceImage.config)) {
|
||||
if (!isRegionalGuidanceIPAdapterConfig(referenceImage.config)) {
|
||||
return;
|
||||
}
|
||||
referenceImage.config.beginEndStepPct = beginEndStepPct;
|
||||
@@ -785,7 +881,7 @@ const slice = createSlice({
|
||||
if (!referenceImage) {
|
||||
return;
|
||||
}
|
||||
if (!isIPAdapterConfig(referenceImage.config)) {
|
||||
if (!isRegionalGuidanceIPAdapterConfig(referenceImage.config)) {
|
||||
return;
|
||||
}
|
||||
referenceImage.config.method = method;
|
||||
@@ -804,7 +900,7 @@ const slice = createSlice({
|
||||
if (!referenceImage) {
|
||||
return;
|
||||
}
|
||||
if (!isFLUXReduxConfig(referenceImage.config)) {
|
||||
if (!isRegionalGuidanceFLUXReduxConfig(referenceImage.config)) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -833,7 +929,7 @@ const slice = createSlice({
|
||||
return;
|
||||
}
|
||||
|
||||
if (isIPAdapterConfig(referenceImage.config) && isFluxReduxModelConfig(modelConfig)) {
|
||||
if (isRegionalGuidanceIPAdapterConfig(referenceImage.config) && isFluxReduxModelConfig(modelConfig)) {
|
||||
// Switching from ip_adapter to flux_redux
|
||||
referenceImage.config = {
|
||||
...initialFLUXRedux,
|
||||
@@ -843,7 +939,7 @@ const slice = createSlice({
|
||||
return;
|
||||
}
|
||||
|
||||
if (isFLUXReduxConfig(referenceImage.config) && isIPAdapterModelConfig(modelConfig)) {
|
||||
if (isRegionalGuidanceFLUXReduxConfig(referenceImage.config) && isIPAdapterModelConfig(modelConfig)) {
|
||||
// Switching from flux_redux to ip_adapter
|
||||
referenceImage.config = {
|
||||
...initialIPAdapter,
|
||||
@@ -853,7 +949,7 @@ const slice = createSlice({
|
||||
return;
|
||||
}
|
||||
|
||||
if (isIPAdapterConfig(referenceImage.config)) {
|
||||
if (isRegionalGuidanceIPAdapterConfig(referenceImage.config)) {
|
||||
referenceImage.config.model = zModelIdentifierField.parse(modelConfig);
|
||||
|
||||
// Ensure that the IP Adapter model is compatible with the CLIP Vision model
|
||||
@@ -876,7 +972,7 @@ const slice = createSlice({
|
||||
if (!referenceImage) {
|
||||
return;
|
||||
}
|
||||
if (!isIPAdapterConfig(referenceImage.config)) {
|
||||
if (!isRegionalGuidanceIPAdapterConfig(referenceImage.config)) {
|
||||
return;
|
||||
}
|
||||
referenceImage.config.clipVisionModel = clipVisionModel;
|
||||
@@ -1658,6 +1754,15 @@ export const {
|
||||
entityBrushLineAdded,
|
||||
entityEraserLineAdded,
|
||||
entityRectAdded,
|
||||
// Raster layer adjustments
|
||||
rasterLayerAdjustmentsSet,
|
||||
rasterLayerAdjustmentsCancel,
|
||||
rasterLayerAdjustmentsReset,
|
||||
rasterLayerAdjustmentsModeChanged,
|
||||
rasterLayerAdjustmentsEnabledToggled,
|
||||
rasterLayerAdjustmentsCollapsedToggled,
|
||||
rasterLayerAdjustmentsSimpleUpdated,
|
||||
rasterLayerAdjustmentsCurvesUpdated,
|
||||
entityDeleted,
|
||||
entityArrangedForwardOne,
|
||||
entityArrangedToFront,
|
||||
|
||||
@@ -1,14 +1,24 @@
|
||||
import { createSelector, createSlice, type PayloadAction } from '@reduxjs/toolkit';
|
||||
import type { RootState } from 'app/store/store';
|
||||
import type { SliceConfig } from 'app/store/types';
|
||||
import type { NumericalParameterConfig } from 'app/types/invokeai';
|
||||
import { paramsReset } from 'features/controlLayers/store/paramsSlice';
|
||||
import { type LoRA, zLoRA } from 'features/controlLayers/store/types';
|
||||
import { zModelIdentifierField } from 'features/nodes/types/common';
|
||||
import { DEFAULT_LORA_WEIGHT_CONFIG } from 'features/system/store/configSlice';
|
||||
import type { LoRAModelConfig } from 'services/api/types';
|
||||
import { v4 as uuidv4 } from 'uuid';
|
||||
import z from 'zod';
|
||||
|
||||
export const DEFAULT_LORA_WEIGHT_CONFIG: NumericalParameterConfig = {
|
||||
initial: 0.75,
|
||||
sliderMin: -1,
|
||||
sliderMax: 2,
|
||||
numberInputMin: -10,
|
||||
numberInputMax: 10,
|
||||
fineStep: 0.01,
|
||||
coarseStep: 0.05,
|
||||
};
|
||||
|
||||
const zLoRAsState = z.object({
|
||||
loras: z.array(zLoRA),
|
||||
});
|
||||
|
||||
@@ -4,6 +4,7 @@ import type { RootState } from 'app/store/store';
|
||||
import type { SliceConfig } from 'app/store/types';
|
||||
import { deepClone } from 'common/util/deepClone';
|
||||
import { roundDownToMultiple, roundToMultiple } from 'common/util/roundDownToMultiple';
|
||||
import { isPlainObject } from 'es-toolkit';
|
||||
import { clamp } from 'es-toolkit/compat';
|
||||
import type { AspectRatioID, ParamsState, RgbaColor } from 'features/controlLayers/store/types';
|
||||
import {
|
||||
@@ -18,6 +19,7 @@ import {
|
||||
isFluxKontextAspectRatioID,
|
||||
isGemini2_5AspectRatioID,
|
||||
isImagenAspectRatioID,
|
||||
MAX_POSITIVE_PROMPT_HISTORY,
|
||||
zParamsState,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import { calculateNewSize } from 'features/controlLayers/util/getScaledBoundingBoxDimensions';
|
||||
@@ -52,6 +54,7 @@ import type {
|
||||
import { getGridSize, getIsSizeOptimal, getOptimalDimension } from 'features/parameters/util/optimalDimension';
|
||||
import { modelConfigsAdapterSelectors, selectModelConfigsQuery } from 'services/api/endpoints/models';
|
||||
import { isNonRefinerMainModelConfig } from 'services/api/types';
|
||||
import { assert } from 'tsafe';
|
||||
|
||||
const slice = createSlice({
|
||||
name: 'params',
|
||||
@@ -122,8 +125,8 @@ const slice = createSlice({
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
state.dimensions.aspectRatio.value = 1;
|
||||
state.dimensions.aspectRatio.id = '1:1';
|
||||
state.dimensions.rect.width = 1024;
|
||||
state.dimensions.rect.height = 1024;
|
||||
state.dimensions.width = 1024;
|
||||
state.dimensions.height = 1024;
|
||||
}
|
||||
|
||||
applyClipSkip(state, model, state.clipSkip);
|
||||
@@ -190,6 +193,24 @@ const slice = createSlice({
|
||||
positivePromptChanged: (state, action: PayloadAction<ParameterPositivePrompt>) => {
|
||||
state.positivePrompt = action.payload;
|
||||
},
|
||||
positivePromptAddedToHistory: (state, action: PayloadAction<ParameterPositivePrompt>) => {
|
||||
const prompt = action.payload.trim();
|
||||
if (prompt.length === 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
state.positivePromptHistory = [prompt, ...state.positivePromptHistory.filter((p) => p !== prompt)];
|
||||
|
||||
if (state.positivePromptHistory.length > MAX_POSITIVE_PROMPT_HISTORY) {
|
||||
state.positivePromptHistory = state.positivePromptHistory.slice(0, MAX_POSITIVE_PROMPT_HISTORY);
|
||||
}
|
||||
},
|
||||
promptRemovedFromHistory: (state, action: PayloadAction<string>) => {
|
||||
state.positivePromptHistory = state.positivePromptHistory.filter((p) => p !== action.payload);
|
||||
},
|
||||
promptHistoryCleared: (state) => {
|
||||
state.positivePromptHistory = [];
|
||||
},
|
||||
negativePromptChanged: (state, action: PayloadAction<ParameterNegativePrompt>) => {
|
||||
state.negativePrompt = action.payload;
|
||||
},
|
||||
@@ -247,26 +268,26 @@ const slice = createSlice({
|
||||
sizeRecalled: (state, action: PayloadAction<{ width: number; height: number }>) => {
|
||||
const { width, height } = action.payload;
|
||||
const gridSize = getGridSize(state.model?.base);
|
||||
state.dimensions.rect.width = Math.max(roundDownToMultiple(width, gridSize), 64);
|
||||
state.dimensions.rect.height = Math.max(roundDownToMultiple(height, gridSize), 64);
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.width = Math.max(roundDownToMultiple(width, gridSize), 64);
|
||||
state.dimensions.height = Math.max(roundDownToMultiple(height, gridSize), 64);
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.id = 'Free';
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
},
|
||||
widthChanged: (state, action: PayloadAction<{ width: number; updateAspectRatio?: boolean; clamp?: boolean }>) => {
|
||||
const { width, updateAspectRatio, clamp } = action.payload;
|
||||
const gridSize = getGridSize(state.model?.base);
|
||||
state.dimensions.rect.width = clamp ? Math.max(roundDownToMultiple(width, gridSize), 64) : width;
|
||||
state.dimensions.width = clamp ? Math.max(roundDownToMultiple(width, gridSize), 64) : width;
|
||||
|
||||
if (state.dimensions.aspectRatio.isLocked) {
|
||||
state.dimensions.rect.height = roundToMultiple(
|
||||
state.dimensions.rect.width / state.dimensions.aspectRatio.value,
|
||||
state.dimensions.height = roundToMultiple(
|
||||
state.dimensions.width / state.dimensions.aspectRatio.value,
|
||||
gridSize
|
||||
);
|
||||
}
|
||||
|
||||
if (updateAspectRatio || !state.dimensions.aspectRatio.isLocked) {
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.id = 'Free';
|
||||
state.dimensions.aspectRatio.isLocked = false;
|
||||
}
|
||||
@@ -274,17 +295,17 @@ const slice = createSlice({
|
||||
heightChanged: (state, action: PayloadAction<{ height: number; updateAspectRatio?: boolean; clamp?: boolean }>) => {
|
||||
const { height, updateAspectRatio, clamp } = action.payload;
|
||||
const gridSize = getGridSize(state.model?.base);
|
||||
state.dimensions.rect.height = clamp ? Math.max(roundDownToMultiple(height, gridSize), 64) : height;
|
||||
state.dimensions.height = clamp ? Math.max(roundDownToMultiple(height, gridSize), 64) : height;
|
||||
|
||||
if (state.dimensions.aspectRatio.isLocked) {
|
||||
state.dimensions.rect.width = roundToMultiple(
|
||||
state.dimensions.rect.height * state.dimensions.aspectRatio.value,
|
||||
state.dimensions.width = roundToMultiple(
|
||||
state.dimensions.height * state.dimensions.aspectRatio.value,
|
||||
gridSize
|
||||
);
|
||||
}
|
||||
|
||||
if (updateAspectRatio || !state.dimensions.aspectRatio.isLocked) {
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.id = 'Free';
|
||||
state.dimensions.aspectRatio.isLocked = false;
|
||||
}
|
||||
@@ -299,55 +320,55 @@ const slice = createSlice({
|
||||
state.dimensions.aspectRatio.isLocked = false;
|
||||
} else if ((state.model?.base === 'imagen3' || state.model?.base === 'imagen4') && isImagenAspectRatioID(id)) {
|
||||
const { width, height } = IMAGEN_ASPECT_RATIOS[id];
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
} else if (state.model?.base === 'chatgpt-4o' && isChatGPT4oAspectRatioID(id)) {
|
||||
const { width, height } = CHATGPT_ASPECT_RATIOS[id];
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
} else if (state.model?.base === 'gemini-2.5' && isGemini2_5AspectRatioID(id)) {
|
||||
const { width, height } = GEMINI_2_5_ASPECT_RATIOS[id];
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
} else if (state.model?.base === 'flux-kontext' && isFluxKontextAspectRatioID(id)) {
|
||||
const { width, height } = FLUX_KONTEXT_ASPECT_RATIOS[id];
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.rect.width / state.dimensions.rect.height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
state.dimensions.aspectRatio.value = state.dimensions.width / state.dimensions.height;
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
} else {
|
||||
state.dimensions.aspectRatio.isLocked = true;
|
||||
state.dimensions.aspectRatio.value = ASPECT_RATIO_MAP[id].ratio;
|
||||
const { width, height } = calculateNewSize(
|
||||
state.dimensions.aspectRatio.value,
|
||||
state.dimensions.rect.width * state.dimensions.rect.height,
|
||||
state.dimensions.width * state.dimensions.height,
|
||||
state.model?.base
|
||||
);
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
}
|
||||
},
|
||||
dimensionsSwapped: (state) => {
|
||||
state.dimensions.aspectRatio.value = 1 / state.dimensions.aspectRatio.value;
|
||||
if (state.dimensions.aspectRatio.id === 'Free') {
|
||||
const newWidth = state.dimensions.rect.height;
|
||||
const newHeight = state.dimensions.rect.width;
|
||||
state.dimensions.rect.width = newWidth;
|
||||
state.dimensions.rect.height = newHeight;
|
||||
const newWidth = state.dimensions.height;
|
||||
const newHeight = state.dimensions.width;
|
||||
state.dimensions.width = newWidth;
|
||||
state.dimensions.height = newHeight;
|
||||
} else {
|
||||
const { width, height } = calculateNewSize(
|
||||
state.dimensions.aspectRatio.value,
|
||||
state.dimensions.rect.width * state.dimensions.rect.height,
|
||||
state.dimensions.width * state.dimensions.height,
|
||||
state.model?.base
|
||||
);
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
state.dimensions.aspectRatio.id = ASPECT_RATIO_MAP[state.dimensions.aspectRatio.id].inverseID;
|
||||
}
|
||||
},
|
||||
@@ -359,25 +380,25 @@ const slice = createSlice({
|
||||
optimalDimension * optimalDimension,
|
||||
state.model?.base
|
||||
);
|
||||
state.dimensions.rect.width = width;
|
||||
state.dimensions.rect.height = height;
|
||||
state.dimensions.width = width;
|
||||
state.dimensions.height = height;
|
||||
} else {
|
||||
state.dimensions.aspectRatio = deepClone(DEFAULT_ASPECT_RATIO_CONFIG);
|
||||
state.dimensions.rect.width = optimalDimension;
|
||||
state.dimensions.rect.height = optimalDimension;
|
||||
state.dimensions.width = optimalDimension;
|
||||
state.dimensions.height = optimalDimension;
|
||||
}
|
||||
},
|
||||
syncedToOptimalDimension: (state) => {
|
||||
const optimalDimension = getOptimalDimension(state.model?.base);
|
||||
|
||||
if (!getIsSizeOptimal(state.dimensions.rect.width, state.dimensions.rect.height, state.model?.base)) {
|
||||
if (!getIsSizeOptimal(state.dimensions.width, state.dimensions.height, state.model?.base)) {
|
||||
const bboxDims = calculateNewSize(
|
||||
state.dimensions.aspectRatio.value,
|
||||
optimalDimension * optimalDimension,
|
||||
state.model?.base
|
||||
);
|
||||
state.dimensions.rect.width = bboxDims.width;
|
||||
state.dimensions.rect.height = bboxDims.height;
|
||||
state.dimensions.width = bboxDims.width;
|
||||
state.dimensions.height = bboxDims.height;
|
||||
}
|
||||
},
|
||||
paramsReset: (state) => resetState(state),
|
||||
@@ -460,6 +481,9 @@ export const {
|
||||
setClipSkip,
|
||||
shouldUseCpuNoiseChanged,
|
||||
positivePromptChanged,
|
||||
positivePromptAddedToHistory,
|
||||
promptRemovedFromHistory,
|
||||
promptHistoryCleared,
|
||||
negativePromptChanged,
|
||||
refinerModelChanged,
|
||||
setRefinerSteps,
|
||||
@@ -488,7 +512,24 @@ export const paramsSliceConfig: SliceConfig<typeof slice> = {
|
||||
schema: zParamsState,
|
||||
getInitialState: getInitialParamsState,
|
||||
persistConfig: {
|
||||
migrate: (state) => zParamsState.parse(state),
|
||||
migrate: (state) => {
|
||||
assert(isPlainObject(state));
|
||||
|
||||
if (!('_version' in state)) {
|
||||
// v0 -> v1, add _version and remove x/y from dimensions, lifting width/height to top level
|
||||
state._version = 1;
|
||||
state.dimensions.width = state.dimensions.rect.width;
|
||||
state.dimensions.height = state.dimensions.rect.height;
|
||||
}
|
||||
|
||||
if (state._version === 1) {
|
||||
// v1 -> v2, add positive prompt history
|
||||
state._version = 2;
|
||||
state.positivePromptHistory = [];
|
||||
}
|
||||
|
||||
return zParamsState.parse(state);
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
@@ -587,6 +628,7 @@ export const selectShouldUseCPUNoise = createParamsSelector((params) => params.s
|
||||
export const selectUpscaleScheduler = createParamsSelector((params) => params.upscaleScheduler);
|
||||
export const selectUpscaleCfgScale = createParamsSelector((params) => params.upscaleCfgScale);
|
||||
|
||||
export const selectPositivePromptHistory = createParamsSelector((params) => params.positivePromptHistory);
|
||||
export const selectRefinerCFGScale = createParamsSelector((params) => params.refinerCFGScale);
|
||||
export const selectRefinerModel = createParamsSelector((params) => params.refinerModel);
|
||||
export const selectIsRefinerModelSelected = createParamsSelector((params) => Boolean(params.refinerModel));
|
||||
@@ -600,8 +642,8 @@ export const selectRefinerScheduler = createParamsSelector((params) => params.re
|
||||
export const selectRefinerStart = createParamsSelector((params) => params.refinerStart);
|
||||
export const selectRefinerSteps = createParamsSelector((params) => params.refinerSteps);
|
||||
|
||||
export const selectWidth = createParamsSelector((params) => params.dimensions.rect.width);
|
||||
export const selectHeight = createParamsSelector((params) => params.dimensions.rect.height);
|
||||
export const selectWidth = createParamsSelector((params) => params.dimensions.width);
|
||||
export const selectHeight = createParamsSelector((params) => params.dimensions.height);
|
||||
export const selectAspectRatioID = createParamsSelector((params) => params.dimensions.aspectRatio.id);
|
||||
export const selectAspectRatioValue = createParamsSelector((params) => params.dimensions.aspectRatio.value);
|
||||
export const selectAspectRatioIsLocked = createParamsSelector((params) => params.dimensions.aspectRatio.isLocked);
|
||||
|
||||
@@ -6,13 +6,16 @@ import type { RootState } from 'app/store/store';
|
||||
import type { SliceConfig } from 'app/store/types';
|
||||
import { clamp } from 'es-toolkit/compat';
|
||||
import { getPrefixedId } from 'features/controlLayers/konva/util';
|
||||
import type { FLUXReduxImageInfluence, RefImagesState } from 'features/controlLayers/store/types';
|
||||
import type {
|
||||
CroppableImageWithDims,
|
||||
FLUXReduxImageInfluence,
|
||||
RefImagesState,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import { zModelIdentifierField } from 'features/nodes/types/common';
|
||||
import type {
|
||||
ChatGPT4oModelConfig,
|
||||
FLUXKontextModelConfig,
|
||||
FLUXReduxModelConfig,
|
||||
ImageDTO,
|
||||
IPAdapterModelConfig,
|
||||
} from 'services/api/types';
|
||||
import { assert } from 'tsafe';
|
||||
@@ -22,7 +25,6 @@ import type { CLIPVisionModelV2, IPMethodV2, RefImageState } from './types';
|
||||
import { getInitialRefImagesState, isFLUXReduxConfig, isIPAdapterConfig, zRefImagesState } from './types';
|
||||
import {
|
||||
getReferenceImageState,
|
||||
imageDTOToImageWithDims,
|
||||
initialChatGPT4oReferenceImage,
|
||||
initialFluxKontextReferenceImage,
|
||||
initialFLUXRedux,
|
||||
@@ -65,13 +67,13 @@ const slice = createSlice({
|
||||
state.entities.push(...entities);
|
||||
}
|
||||
},
|
||||
refImageImageChanged: (state, action: PayloadActionWithId<{ imageDTO: ImageDTO | null }>) => {
|
||||
const { id, imageDTO } = action.payload;
|
||||
refImageImageChanged: (state, action: PayloadActionWithId<{ croppableImage: CroppableImageWithDims | null }>) => {
|
||||
const { id, croppableImage } = action.payload;
|
||||
const entity = selectRefImageEntity(state, id);
|
||||
if (!entity) {
|
||||
return;
|
||||
}
|
||||
entity.config.image = imageDTO ? imageDTOToImageWithDims(imageDTO) : null;
|
||||
entity.config.image = croppableImage;
|
||||
},
|
||||
refImageIPAdapterMethodChanged: (state, action: PayloadActionWithId<{ method: IPMethodV2 }>) => {
|
||||
const { id, method } = action.payload;
|
||||
|
||||
@@ -37,6 +37,45 @@ export const zImageWithDims = z.object({
|
||||
});
|
||||
export type ImageWithDims = z.infer<typeof zImageWithDims>;
|
||||
|
||||
const zCropBox = z.object({
|
||||
x: z.number().min(0),
|
||||
y: z.number().min(0),
|
||||
width: z.number().positive(),
|
||||
height: z.number().positive(),
|
||||
});
|
||||
// This new schema is an extension of zImageWithDims, with an optional crop field.
|
||||
//
|
||||
// When we added cropping support to certain entities (e.g. Ref Images, video Starting Frame Image), we changed
|
||||
// their schemas from using zImageWithDims to this new schema. To support loading pre-existing entities that
|
||||
// were created before cropping was supported, we can use zod's preprocess to transform old data into the new format.
|
||||
// Its essentially a data migration step.
|
||||
//
|
||||
// This parsing happens currently in two places:
|
||||
// - Recalling metadata.
|
||||
// - Loading/rehydrating persisted client state from storage.
|
||||
export const zCroppableImageWithDims = z.preprocess(
|
||||
(val) => {
|
||||
try {
|
||||
const imageWithDims = zImageWithDims.parse(val);
|
||||
const migrated = { original: { image: deepClone(imageWithDims) } };
|
||||
return migrated;
|
||||
} catch {
|
||||
return val;
|
||||
}
|
||||
},
|
||||
z.object({
|
||||
original: z.object({ image: zImageWithDims }),
|
||||
crop: z
|
||||
.object({
|
||||
box: zCropBox,
|
||||
ratio: z.number().gt(0).nullable(),
|
||||
image: zImageWithDims,
|
||||
})
|
||||
.optional(),
|
||||
})
|
||||
);
|
||||
export type CroppableImageWithDims = z.infer<typeof zCroppableImageWithDims>;
|
||||
|
||||
const zImageWithDimsDataURL = z.object({
|
||||
dataURL: z.string(),
|
||||
width: z.number().int().positive(),
|
||||
@@ -116,6 +155,9 @@ export type SAMPointLabel = z.infer<typeof zSAMPointLabel>;
|
||||
export const zSAMPointLabelString = z.enum(['background', 'neutral', 'foreground']);
|
||||
export type SAMPointLabelString = z.infer<typeof zSAMPointLabelString>;
|
||||
|
||||
export const zSAMModel = z.enum(['SAM1', 'SAM2']);
|
||||
export type SAMModel = z.infer<typeof zSAMModel>;
|
||||
|
||||
/**
|
||||
* A mapping of SAM point labels (as numbers) to their string representations.
|
||||
*/
|
||||
@@ -232,7 +274,7 @@ export type CanvasObjectState = z.infer<typeof zCanvasObjectState>;
|
||||
|
||||
const zIPAdapterConfig = z.object({
|
||||
type: z.literal('ip_adapter'),
|
||||
image: zImageWithDims.nullable(),
|
||||
image: zCroppableImageWithDims.nullable(),
|
||||
model: zModelIdentifierField.nullable(),
|
||||
weight: z.number().gte(-1).lte(2),
|
||||
beginEndStepPct: zBeginEndStepPct,
|
||||
@@ -241,21 +283,39 @@ const zIPAdapterConfig = z.object({
|
||||
});
|
||||
export type IPAdapterConfig = z.infer<typeof zIPAdapterConfig>;
|
||||
|
||||
const zRegionalGuidanceIPAdapterConfig = z.object({
|
||||
type: z.literal('ip_adapter'),
|
||||
image: zImageWithDims.nullable(),
|
||||
model: zModelIdentifierField.nullable(),
|
||||
weight: z.number().gte(-1).lte(2),
|
||||
beginEndStepPct: zBeginEndStepPct,
|
||||
method: zIPMethodV2,
|
||||
clipVisionModel: zCLIPVisionModelV2,
|
||||
});
|
||||
export type RegionalGuidanceIPAdapterConfig = z.infer<typeof zRegionalGuidanceIPAdapterConfig>;
|
||||
|
||||
const zFLUXReduxImageInfluence = z.enum(['lowest', 'low', 'medium', 'high', 'highest']);
|
||||
export const isFLUXReduxImageInfluence = (v: unknown): v is FLUXReduxImageInfluence =>
|
||||
zFLUXReduxImageInfluence.safeParse(v).success;
|
||||
export type FLUXReduxImageInfluence = z.infer<typeof zFLUXReduxImageInfluence>;
|
||||
const zFLUXReduxConfig = z.object({
|
||||
type: z.literal('flux_redux'),
|
||||
image: zImageWithDims.nullable(),
|
||||
image: zCroppableImageWithDims.nullable(),
|
||||
model: zModelIdentifierField.nullable(),
|
||||
imageInfluence: zFLUXReduxImageInfluence.default('highest'),
|
||||
});
|
||||
export type FLUXReduxConfig = z.infer<typeof zFLUXReduxConfig>;
|
||||
const zRegionalGuidanceFLUXReduxConfig = z.object({
|
||||
type: z.literal('flux_redux'),
|
||||
image: zImageWithDims.nullable(),
|
||||
model: zModelIdentifierField.nullable(),
|
||||
imageInfluence: zFLUXReduxImageInfluence.default('highest'),
|
||||
});
|
||||
type RegionalGuidanceFLUXReduxConfig = z.infer<typeof zRegionalGuidanceFLUXReduxConfig>;
|
||||
|
||||
const zChatGPT4oReferenceImageConfig = z.object({
|
||||
type: z.literal('chatgpt_4o_reference_image'),
|
||||
image: zImageWithDims.nullable(),
|
||||
image: zCroppableImageWithDims.nullable(),
|
||||
/**
|
||||
* TODO(psyche): Technically there is no model for ChatGPT 4o reference images - it's just a field in the API call.
|
||||
* But we use a model drop down to switch between different ref image types, so there needs to be a model here else
|
||||
@@ -267,14 +327,14 @@ export type ChatGPT4oReferenceImageConfig = z.infer<typeof zChatGPT4oReferenceIm
|
||||
|
||||
const zGemini2_5ReferenceImageConfig = z.object({
|
||||
type: z.literal('gemini_2_5_reference_image'),
|
||||
image: zImageWithDims.nullable(),
|
||||
image: zCroppableImageWithDims.nullable(),
|
||||
model: zModelIdentifierField.nullable(),
|
||||
});
|
||||
export type Gemini2_5ReferenceImageConfig = z.infer<typeof zGemini2_5ReferenceImageConfig>;
|
||||
|
||||
const zFluxKontextReferenceImageConfig = z.object({
|
||||
type: z.literal('flux_kontext_reference_image'),
|
||||
image: zImageWithDims.nullable(),
|
||||
image: zCroppableImageWithDims.nullable(),
|
||||
model: zModelIdentifierField.nullable(),
|
||||
});
|
||||
export type FluxKontextReferenceImageConfig = z.infer<typeof zFluxKontextReferenceImageConfig>;
|
||||
@@ -304,6 +364,7 @@ export const isIPAdapterConfig = (config: RefImageState['config']): config is IP
|
||||
|
||||
export const isFLUXReduxConfig = (config: RefImageState['config']): config is FLUXReduxConfig =>
|
||||
config.type === 'flux_redux';
|
||||
|
||||
export const isChatGPT4oReferenceImageConfig = (
|
||||
config: RefImageState['config']
|
||||
): config is ChatGPT4oReferenceImageConfig => config.type === 'chatgpt_4o_reference_image';
|
||||
@@ -323,10 +384,18 @@ const zFill = z.object({ style: zFillStyle, color: zRgbColor });
|
||||
|
||||
const zRegionalGuidanceRefImageState = z.object({
|
||||
id: zId,
|
||||
config: z.discriminatedUnion('type', [zIPAdapterConfig, zFLUXReduxConfig]),
|
||||
config: z.discriminatedUnion('type', [zRegionalGuidanceIPAdapterConfig, zRegionalGuidanceFLUXReduxConfig]),
|
||||
});
|
||||
export type RegionalGuidanceRefImageState = z.infer<typeof zRegionalGuidanceRefImageState>;
|
||||
|
||||
export const isRegionalGuidanceIPAdapterConfig = (
|
||||
config: RegionalGuidanceRefImageState['config']
|
||||
): config is RegionalGuidanceIPAdapterConfig => config.type === 'ip_adapter';
|
||||
|
||||
export const isRegionalGuidanceFLUXReduxConfig = (
|
||||
config: RegionalGuidanceRefImageState['config']
|
||||
): config is RegionalGuidanceFLUXReduxConfig => config.type === 'flux_redux';
|
||||
|
||||
const zCanvasRegionalGuidanceState = zCanvasEntityBase.extend({
|
||||
type: z.literal('regional_guidance'),
|
||||
position: zCoordinate,
|
||||
@@ -375,11 +444,57 @@ const zControlLoRAConfig = z.object({
|
||||
});
|
||||
export type ControlLoRAConfig = z.infer<typeof zControlLoRAConfig>;
|
||||
|
||||
/**
|
||||
* All simple params normalized to `[-1, 1]` except sharpness `[0, 1]`.
|
||||
*
|
||||
* - Brightness: -1 (darken) to 1 (brighten)
|
||||
* - Contrast: -1 (decrease contrast) to 1 (increase contrast)
|
||||
* - Saturation: -1 (desaturate) to 1 (saturate)
|
||||
* - Temperature: -1 (cooler/blue) to 1 (warmer/yellow)
|
||||
* - Tint: -1 (greener) to 1 (more magenta)
|
||||
* - Sharpness: 0 (no sharpening) to 1 (maximum sharpening)
|
||||
*/
|
||||
export const zSimpleAdjustmentsConfig = z.object({
|
||||
brightness: z.number().gte(-1).lte(1),
|
||||
contrast: z.number().gte(-1).lte(1),
|
||||
saturation: z.number().gte(-1).lte(1),
|
||||
temperature: z.number().gte(-1).lte(1),
|
||||
tint: z.number().gte(-1).lte(1),
|
||||
sharpness: z.number().gte(0).lte(1),
|
||||
});
|
||||
export type SimpleAdjustmentsConfig = z.infer<typeof zSimpleAdjustmentsConfig>;
|
||||
|
||||
const zUint8 = z.number().int().min(0).max(255);
|
||||
const zChannelPoints = z.array(z.tuple([zUint8, zUint8])).min(2);
|
||||
const zChannelName = z.enum(['master', 'r', 'g', 'b']);
|
||||
const zCurvesAdjustmentsConfig = z.record(zChannelName, zChannelPoints);
|
||||
export type ChannelName = z.infer<typeof zChannelName>;
|
||||
export type ChannelPoints = z.infer<typeof zChannelPoints>;
|
||||
export type CurvesAdjustmentsConfig = z.infer<typeof zCurvesAdjustmentsConfig>;
|
||||
|
||||
/**
|
||||
* The curves adjustments are stored as LUTs in the Konva node attributes. Konva will use these values when applying
|
||||
* the filter.
|
||||
*/
|
||||
export const zCurvesAdjustmentsLUTs = z.record(zChannelName, z.array(zUint8));
|
||||
|
||||
const zRasterLayerAdjustments = z.object({
|
||||
version: z.literal(1),
|
||||
enabled: z.boolean(),
|
||||
collapsed: z.boolean(),
|
||||
mode: z.enum(['simple', 'curves']),
|
||||
simple: zSimpleAdjustmentsConfig,
|
||||
curves: zCurvesAdjustmentsConfig,
|
||||
});
|
||||
export type RasterLayerAdjustments = z.infer<typeof zRasterLayerAdjustments>;
|
||||
|
||||
const zCanvasRasterLayerState = zCanvasEntityBase.extend({
|
||||
type: z.literal('raster_layer'),
|
||||
position: zCoordinate,
|
||||
opacity: zOpacity,
|
||||
objects: z.array(zCanvasObjectState),
|
||||
// Optional per-layer color adjustments (simple + curves). When undefined, no adjustments are applied.
|
||||
adjustments: zRasterLayerAdjustments.optional(),
|
||||
});
|
||||
export type CanvasRasterLayerState = z.infer<typeof zCanvasRasterLayerState>;
|
||||
|
||||
@@ -421,7 +536,7 @@ export const zLoRA = z.object({
|
||||
id: z.string(),
|
||||
isEnabled: z.boolean(),
|
||||
model: zModelIdentifierField,
|
||||
weight: z.number().gte(-1).lte(2),
|
||||
weight: z.number().gte(-10).lte(10),
|
||||
});
|
||||
export type LoRA = z.infer<typeof zLoRA>;
|
||||
|
||||
@@ -558,18 +673,18 @@ const zBboxState = z.object({
|
||||
});
|
||||
|
||||
const zDimensionsState = z.object({
|
||||
// TODO(psyche): There is no concept of x/y coords for the dimensions state here... It's just width and height.
|
||||
// Remove the extraneous data.
|
||||
rect: z.object({
|
||||
x: z.number().int(),
|
||||
y: z.number().int(),
|
||||
width: zParameterImageDimension,
|
||||
height: zParameterImageDimension,
|
||||
}),
|
||||
width: zParameterImageDimension,
|
||||
height: zParameterImageDimension,
|
||||
aspectRatio: zAspectRatioConfig,
|
||||
});
|
||||
|
||||
export const MAX_POSITIVE_PROMPT_HISTORY = 100;
|
||||
const zPositivePromptHistory = z
|
||||
.array(zParameterPositivePrompt)
|
||||
.transform((arr) => arr.slice(0, MAX_POSITIVE_PROMPT_HISTORY));
|
||||
|
||||
export const zParamsState = z.object({
|
||||
_version: z.literal(2),
|
||||
maskBlur: z.number(),
|
||||
maskBlurMethod: zParameterMaskBlurMethod,
|
||||
canvasCoherenceMode: zParameterCanvasCoherenceMode,
|
||||
@@ -600,6 +715,7 @@ export const zParamsState = z.object({
|
||||
clipSkip: z.number(),
|
||||
shouldUseCpuNoise: z.boolean(),
|
||||
positivePrompt: zParameterPositivePrompt,
|
||||
positivePromptHistory: zPositivePromptHistory,
|
||||
negativePrompt: zParameterNegativePrompt,
|
||||
refinerModel: zParameterSDXLRefinerModel.nullable(),
|
||||
refinerSteps: z.number(),
|
||||
@@ -617,6 +733,7 @@ export const zParamsState = z.object({
|
||||
});
|
||||
export type ParamsState = z.infer<typeof zParamsState>;
|
||||
export const getInitialParamsState = (): ParamsState => ({
|
||||
_version: 2,
|
||||
maskBlur: 16,
|
||||
maskBlurMethod: 'box',
|
||||
canvasCoherenceMode: 'Gaussian Blur',
|
||||
@@ -647,6 +764,7 @@ export const getInitialParamsState = (): ParamsState => ({
|
||||
clipSkip: 0,
|
||||
shouldUseCpuNoise: true,
|
||||
positivePrompt: '',
|
||||
positivePromptHistory: [],
|
||||
negativePrompt: null,
|
||||
refinerModel: null,
|
||||
refinerSteps: 20,
|
||||
@@ -661,7 +779,8 @@ export const getInitialParamsState = (): ParamsState => ({
|
||||
clipGEmbedModel: null,
|
||||
controlLora: null,
|
||||
dimensions: {
|
||||
rect: { x: 0, y: 0, width: 512, height: 512 },
|
||||
width: 512,
|
||||
height: 512,
|
||||
aspectRatio: deepClone(DEFAULT_ASPECT_RATIO_CONFIG),
|
||||
},
|
||||
});
|
||||
|
||||
@@ -10,12 +10,15 @@ import type {
|
||||
ChatGPT4oReferenceImageConfig,
|
||||
ControlLoRAConfig,
|
||||
ControlNetConfig,
|
||||
CroppableImageWithDims,
|
||||
FluxKontextReferenceImageConfig,
|
||||
FLUXReduxConfig,
|
||||
Gemini2_5ReferenceImageConfig,
|
||||
ImageWithDims,
|
||||
IPAdapterConfig,
|
||||
RasterLayerAdjustments,
|
||||
RefImageState,
|
||||
RegionalGuidanceIPAdapterConfig,
|
||||
RgbColor,
|
||||
T2IAdapterConfig,
|
||||
} from 'features/controlLayers/store/types';
|
||||
@@ -44,6 +47,21 @@ export const imageDTOToImageWithDims = ({ image_name, width, height }: ImageDTO)
|
||||
height,
|
||||
});
|
||||
|
||||
export const imageDTOToCroppableImage = (
|
||||
originalImageDTO: ImageDTO,
|
||||
crop?: CroppableImageWithDims['crop']
|
||||
): CroppableImageWithDims => {
|
||||
const { image_name, width, height } = originalImageDTO;
|
||||
const val: CroppableImageWithDims = {
|
||||
original: { image: { image_name, width, height } },
|
||||
};
|
||||
if (crop) {
|
||||
val.crop = deepClone(crop);
|
||||
}
|
||||
|
||||
return val;
|
||||
};
|
||||
|
||||
export const imageDTOToImageField = ({ image_name }: ImageDTO): ImageField => ({ image_name });
|
||||
|
||||
const DEFAULT_RG_MASK_FILL_COLORS: RgbColor[] = [
|
||||
@@ -78,6 +96,15 @@ export const initialIPAdapter: IPAdapterConfig = {
|
||||
clipVisionModel: 'ViT-H',
|
||||
weight: 1,
|
||||
};
|
||||
export const initialRegionalGuidanceIPAdapter: RegionalGuidanceIPAdapterConfig = {
|
||||
type: 'ip_adapter',
|
||||
image: null,
|
||||
model: null,
|
||||
beginEndStepPct: [0, 1],
|
||||
method: 'full',
|
||||
clipVisionModel: 'ViT-H',
|
||||
weight: 1,
|
||||
};
|
||||
export const initialFLUXRedux: FLUXReduxConfig = {
|
||||
type: 'flux_redux',
|
||||
image: null,
|
||||
@@ -118,6 +145,32 @@ export const initialControlLoRA: ControlLoRAConfig = {
|
||||
weight: 0.75,
|
||||
};
|
||||
|
||||
export const makeDefaultRasterLayerAdjustments = (mode: 'simple' | 'curves' = 'simple'): RasterLayerAdjustments => ({
|
||||
version: 1,
|
||||
enabled: true,
|
||||
collapsed: false,
|
||||
mode,
|
||||
simple: { brightness: 0, contrast: 0, saturation: 0, temperature: 0, tint: 0, sharpness: 0 },
|
||||
curves: {
|
||||
master: [
|
||||
[0, 0],
|
||||
[255, 255],
|
||||
],
|
||||
r: [
|
||||
[0, 0],
|
||||
[255, 255],
|
||||
],
|
||||
g: [
|
||||
[0, 0],
|
||||
[255, 255],
|
||||
],
|
||||
b: [
|
||||
[0, 0],
|
||||
[255, 255],
|
||||
],
|
||||
},
|
||||
});
|
||||
|
||||
export const getReferenceImageState = (id: string, overrides?: PartialDeep<RefImageState>): RefImageState => {
|
||||
const entityState: RefImageState = {
|
||||
id,
|
||||
@@ -187,6 +240,7 @@ export const getRasterLayerState = (
|
||||
objects: [],
|
||||
opacity: 1,
|
||||
position: { x: 0, y: 0 },
|
||||
adjustments: undefined,
|
||||
};
|
||||
merge(entityState, overrides);
|
||||
return entityState;
|
||||
|
||||
@@ -0,0 +1,215 @@
|
||||
import {
|
||||
Button,
|
||||
ButtonGroup,
|
||||
Divider,
|
||||
Flex,
|
||||
FormControl,
|
||||
FormLabel,
|
||||
Select,
|
||||
Spacer,
|
||||
Text,
|
||||
} from '@invoke-ai/ui-library';
|
||||
import { useAppSelector } from 'app/store/storeHooks';
|
||||
import type { AspectRatioID } from 'features/controlLayers/store/types';
|
||||
import { ASPECT_RATIO_MAP, isAspectRatioID } from 'features/controlLayers/store/types';
|
||||
import type { CropBox } from 'features/cropper/lib/editor';
|
||||
import { cropImageModalApi, type CropImageModalState } from 'features/cropper/store';
|
||||
import { selectAutoAddBoardId } from 'features/gallery/store/gallerySelectors';
|
||||
import React, { memo, useCallback, useEffect, useRef, useState } from 'react';
|
||||
import { useUploadImageMutation } from 'services/api/endpoints/images';
|
||||
import { objectEntries } from 'tsafe';
|
||||
|
||||
type Props = {
|
||||
editor: CropImageModalState['editor'];
|
||||
onApplyCrop: CropImageModalState['onApplyCrop'];
|
||||
onReady: CropImageModalState['onReady'];
|
||||
};
|
||||
|
||||
const getAspectRatioString = (ratio: number | null): AspectRatioID => {
|
||||
if (!ratio) {
|
||||
return 'Free';
|
||||
}
|
||||
const entries = objectEntries(ASPECT_RATIO_MAP);
|
||||
for (const [key, value] of entries) {
|
||||
if (value.ratio === ratio) {
|
||||
return key;
|
||||
}
|
||||
}
|
||||
return 'Free';
|
||||
};
|
||||
|
||||
export const CropImageEditor = memo(({ editor, onApplyCrop, onReady }: Props) => {
|
||||
const containerRef = useRef<HTMLDivElement>(null);
|
||||
const [zoom, setZoom] = useState(100);
|
||||
const [cropBox, setCropBox] = useState<CropBox | null>(null);
|
||||
const [aspectRatio, setAspectRatio] = useState<string>('free');
|
||||
const autoAddBoardId = useAppSelector(selectAutoAddBoardId);
|
||||
|
||||
const [uploadImage] = useUploadImageMutation({ fixedCacheKey: 'editorContainer' });
|
||||
|
||||
const setup = useCallback(
|
||||
async (container: HTMLDivElement) => {
|
||||
editor.init(container);
|
||||
editor.onZoomChange((zoom) => {
|
||||
setZoom(zoom);
|
||||
});
|
||||
editor.onCropBoxChange((crop) => {
|
||||
setCropBox(crop);
|
||||
});
|
||||
editor.onAspectRatioChange((ratio) => {
|
||||
setAspectRatio(getAspectRatioString(ratio));
|
||||
});
|
||||
await onReady();
|
||||
editor.fitToContainer();
|
||||
},
|
||||
[editor, onReady]
|
||||
);
|
||||
|
||||
useEffect(() => {
|
||||
const container = containerRef.current;
|
||||
if (!container) {
|
||||
return;
|
||||
}
|
||||
setup(container);
|
||||
const handleResize = () => {
|
||||
editor.resize(container.clientWidth, container.clientHeight);
|
||||
};
|
||||
|
||||
const resizeObserver = new ResizeObserver(handleResize);
|
||||
resizeObserver.observe(container);
|
||||
return () => {
|
||||
resizeObserver.disconnect();
|
||||
};
|
||||
}, [editor, setup]);
|
||||
|
||||
const handleAspectRatioChange = useCallback(
|
||||
(e: React.ChangeEvent<HTMLSelectElement>) => {
|
||||
const newRatio = e.target.value;
|
||||
if (!isAspectRatioID(newRatio)) {
|
||||
return;
|
||||
}
|
||||
setAspectRatio(newRatio);
|
||||
|
||||
if (newRatio === 'Free') {
|
||||
editor.setCropAspectRatio(null);
|
||||
} else {
|
||||
editor.setCropAspectRatio(ASPECT_RATIO_MAP[newRatio]?.ratio ?? null);
|
||||
}
|
||||
},
|
||||
[editor]
|
||||
);
|
||||
|
||||
const handleResetCrop = useCallback(() => {
|
||||
editor.resetCrop();
|
||||
}, [editor]);
|
||||
|
||||
const handleApplyCrop = useCallback(async () => {
|
||||
await onApplyCrop();
|
||||
cropImageModalApi.close();
|
||||
}, [onApplyCrop]);
|
||||
|
||||
const handleCancelCrop = useCallback(() => {
|
||||
cropImageModalApi.close();
|
||||
}, []);
|
||||
|
||||
const handleExport = useCallback(async () => {
|
||||
try {
|
||||
const blob = await editor.exportImage('blob');
|
||||
const file = new File([blob], 'image.png', { type: 'image/png' });
|
||||
|
||||
await uploadImage({
|
||||
file,
|
||||
is_intermediate: false,
|
||||
image_category: 'user',
|
||||
board_id: autoAddBoardId === 'none' ? undefined : autoAddBoardId,
|
||||
}).unwrap();
|
||||
} catch (err) {
|
||||
if (err instanceof Error && err.message.includes('tainted')) {
|
||||
alert(
|
||||
'Cannot export image: The image is from a different domain (CORS issue). To fix this:\n\n1. Load images from the same domain\n2. Use images from CORS-enabled sources\n3. Upload a local image file instead'
|
||||
);
|
||||
} else {
|
||||
alert(`Export failed: ${err instanceof Error ? err.message : String(err)}`);
|
||||
}
|
||||
}
|
||||
}, [autoAddBoardId, editor, uploadImage]);
|
||||
|
||||
const zoomIn = useCallback(() => {
|
||||
editor.zoomIn();
|
||||
}, [editor]);
|
||||
|
||||
const zoomOut = useCallback(() => {
|
||||
editor.zoomOut();
|
||||
}, [editor]);
|
||||
|
||||
const fitToContainer = useCallback(() => {
|
||||
editor.fitToContainer();
|
||||
}, [editor]);
|
||||
|
||||
const resetView = useCallback(() => {
|
||||
editor.resetView();
|
||||
}, [editor]);
|
||||
|
||||
return (
|
||||
<Flex w="full" h="full" flexDir="column" gap={4}>
|
||||
<Flex gap={2} alignItems="center">
|
||||
<FormControl flex={1}>
|
||||
<FormLabel>Aspect Ratio:</FormLabel>
|
||||
<Select size="sm" value={aspectRatio} onChange={handleAspectRatioChange} w={32}>
|
||||
<option value="Free">Free</option>
|
||||
<option value="16:9">16:9</option>
|
||||
<option value="3:2">3:2</option>
|
||||
<option value="4:3">4:3</option>
|
||||
<option value="1:1">1:1</option>
|
||||
<option value="3:4">3:4</option>
|
||||
<option value="2:3">2:3</option>
|
||||
<option value="9:16">9:16</option>
|
||||
</Select>
|
||||
</FormControl>
|
||||
|
||||
<Spacer />
|
||||
|
||||
<ButtonGroup size="sm" isAttached={false}>
|
||||
<Button onClick={fitToContainer}>Fit View</Button>
|
||||
<Button onClick={resetView}>Reset View</Button>
|
||||
<Button onClick={zoomIn}>Zoom In</Button>
|
||||
<Button onClick={zoomOut}>Zoom Out</Button>
|
||||
</ButtonGroup>
|
||||
|
||||
<Spacer />
|
||||
|
||||
<ButtonGroup size="sm" isAttached={false}>
|
||||
<Button onClick={handleApplyCrop}>Apply</Button>
|
||||
<Button onClick={handleResetCrop}>Reset</Button>
|
||||
<Button onClick={handleCancelCrop}>Cancel</Button>
|
||||
<Button onClick={handleExport}>Save to Assets</Button>
|
||||
</ButtonGroup>
|
||||
</Flex>
|
||||
|
||||
<Flex position="relative" w="full" h="full" bg="base.900">
|
||||
<Flex position="absolute" inset={0} ref={containerRef} />
|
||||
</Flex>
|
||||
|
||||
<Flex gap={2} color="base.300">
|
||||
<Text>Mouse wheel: Zoom</Text>
|
||||
<Divider orientation="vertical" />
|
||||
<Text>Space + Drag: Pan</Text>
|
||||
<Divider orientation="vertical" />
|
||||
<Text>Drag crop box or handles to adjust</Text>
|
||||
{cropBox && (
|
||||
<>
|
||||
<Divider orientation="vertical" />
|
||||
<Text>
|
||||
X: {Math.round(cropBox.x)}, Y: {Math.round(cropBox.y)}, Width: {Math.round(cropBox.width)}, Height:{' '}
|
||||
{Math.round(cropBox.height)}
|
||||
</Text>
|
||||
</>
|
||||
)}
|
||||
<Spacer key="help-spacer" />
|
||||
<Text key="help-zoom">Zoom: {Math.round(zoom * 100)}%</Text>
|
||||
</Flex>
|
||||
</Flex>
|
||||
);
|
||||
});
|
||||
|
||||
CropImageEditor.displayName = 'CropImageEditor';
|
||||
@@ -0,0 +1,29 @@
|
||||
import { Modal, ModalBody, ModalContent, ModalHeader, ModalOverlay } from '@invoke-ai/ui-library';
|
||||
import { useStore } from '@nanostores/react';
|
||||
import { cropImageModalApi } from 'features/cropper/store';
|
||||
import { memo } from 'react';
|
||||
|
||||
import { CropImageEditor } from './CropImageEditor';
|
||||
|
||||
export const CropImageModal = memo(() => {
|
||||
const state = useStore(cropImageModalApi.$state);
|
||||
|
||||
if (!state) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return (
|
||||
// This modal is always open when this component is rendered
|
||||
<Modal isOpen={true} onClose={cropImageModalApi.close} isCentered useInert={false} size="full">
|
||||
<ModalOverlay />
|
||||
<ModalContent minH="unset" minW="unset" maxH="90vh" maxW="90vw" w="full" h="full" borderRadius="base">
|
||||
<ModalHeader>Crop Image</ModalHeader>
|
||||
<ModalBody px={4} pb={4} pt={0}>
|
||||
<CropImageEditor editor={state.editor} onApplyCrop={state.onApplyCrop} onReady={state.onReady} />
|
||||
</ModalBody>
|
||||
</ModalContent>
|
||||
</Modal>
|
||||
);
|
||||
});
|
||||
|
||||
CropImageModal.displayName = 'CropImageModal';
|
||||
1557
invokeai/frontend/web/src/features/cropper/lib/editor.ts
Normal file
1557
invokeai/frontend/web/src/features/cropper/lib/editor.ts
Normal file
File diff suppressed because it is too large
Load Diff
26
invokeai/frontend/web/src/features/cropper/store/index.ts
Normal file
26
invokeai/frontend/web/src/features/cropper/store/index.ts
Normal file
@@ -0,0 +1,26 @@
|
||||
import type { Editor } from 'features/cropper/lib/editor';
|
||||
import { atom } from 'nanostores';
|
||||
|
||||
export type CropImageModalState = {
|
||||
editor: Editor;
|
||||
onApplyCrop: () => Promise<void> | void;
|
||||
onReady: () => Promise<void> | void;
|
||||
};
|
||||
|
||||
const $state = atom<CropImageModalState | null>(null);
|
||||
|
||||
const open = (state: CropImageModalState) => {
|
||||
$state.set(state);
|
||||
};
|
||||
|
||||
const close = () => {
|
||||
const state = $state.get();
|
||||
state?.editor.destroy();
|
||||
$state.set(null);
|
||||
};
|
||||
|
||||
export const cropImageModalApi = {
|
||||
$state,
|
||||
open,
|
||||
close,
|
||||
};
|
||||
@@ -236,8 +236,11 @@ const deleteControlLayerImages = (state: RootState, dispatch: AppDispatch, image
|
||||
|
||||
const deleteReferenceImages = (state: RootState, dispatch: AppDispatch, image_name: string) => {
|
||||
selectReferenceImageEntities(state).forEach((entity) => {
|
||||
if (entity.config.image?.image_name === image_name) {
|
||||
dispatch(refImageImageChanged({ id: entity.id, imageDTO: null }));
|
||||
if (
|
||||
entity.config.image?.original.image.image_name === image_name ||
|
||||
entity.config.image?.crop?.image.image_name === image_name
|
||||
) {
|
||||
dispatch(refImageImageChanged({ id: entity.id, croppableImage: null }));
|
||||
}
|
||||
});
|
||||
};
|
||||
@@ -284,7 +287,10 @@ export const getImageUsage = (
|
||||
|
||||
const isUpscaleImage = upscale.upscaleInitialImage?.image_name === image_name;
|
||||
|
||||
const isReferenceImage = refImages.entities.some(({ config }) => config.image?.image_name === image_name);
|
||||
const isReferenceImage = refImages.entities.some(
|
||||
({ config }) =>
|
||||
config.image?.original.image.image_name === image_name || config.image?.crop?.image.image_name === image_name
|
||||
);
|
||||
|
||||
const isRasterLayerImage = canvas.rasterLayers.entities.some(({ objects }) =>
|
||||
objects.some((obj) => obj.type === 'image' && 'image_name' in obj.image && obj.image.image_name === image_name)
|
||||
|
||||
@@ -3,7 +3,7 @@ import { IconButton } from '@invoke-ai/ui-library';
|
||||
import type { MouseEvent } from 'react';
|
||||
import { memo } from 'react';
|
||||
|
||||
const sx: SystemStyleObject = {
|
||||
export const imageButtonSx: SystemStyleObject = {
|
||||
minW: 0,
|
||||
svg: {
|
||||
transitionProperty: 'common',
|
||||
@@ -31,7 +31,7 @@ export const DndImageIcon = memo((props: Props) => {
|
||||
aria-label={tooltip}
|
||||
icon={icon}
|
||||
variant="link"
|
||||
sx={sx}
|
||||
sx={imageButtonSx}
|
||||
data-testid={tooltip}
|
||||
{...rest}
|
||||
/>
|
||||
|
||||
@@ -4,7 +4,7 @@ import { getDefaultRefImageConfig } from 'features/controlLayers/hooks/addLayerH
|
||||
import { getPrefixedId } from 'features/controlLayers/konva/util';
|
||||
import { refImageAdded } from 'features/controlLayers/store/refImagesSlice';
|
||||
import type { CanvasEntityIdentifier, CanvasEntityType } from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageWithDims } from 'features/controlLayers/store/util';
|
||||
import { imageDTOToCroppableImage } from 'features/controlLayers/store/util';
|
||||
import { selectComparisonImages } from 'features/gallery/components/ImageViewer/common';
|
||||
import type { BoardId } from 'features/gallery/store/types';
|
||||
import {
|
||||
@@ -211,7 +211,7 @@ export const addGlobalReferenceImageDndTarget: DndTarget<
|
||||
handler: ({ sourceData, dispatch, getState }) => {
|
||||
const { imageDTO } = sourceData.payload;
|
||||
const config = getDefaultRefImageConfig(getState);
|
||||
config.image = imageDTOToImageWithDims(imageDTO);
|
||||
config.image = imageDTOToCroppableImage(imageDTO);
|
||||
dispatch(refImageAdded({ overrides: { config } }));
|
||||
},
|
||||
};
|
||||
@@ -641,7 +641,7 @@ export const videoFrameFromImageDndTarget: DndTarget<VideoFrameFromImageDndTarge
|
||||
},
|
||||
handler: ({ sourceData, dispatch }) => {
|
||||
const { imageDTO } = sourceData.payload;
|
||||
dispatch(startingFrameImageChanged(imageDTOToImageWithDims(imageDTO)));
|
||||
dispatch(startingFrameImageChanged(imageDTOToCroppableImage(imageDTO)));
|
||||
},
|
||||
};
|
||||
//#endregion
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
import { MenuItem } from '@invoke-ai/ui-library';
|
||||
import { imageDTOToCroppableImage } from 'features/controlLayers/store/util';
|
||||
import { useItemDTOContextImageOnly } from 'features/gallery/contexts/ItemDTOContext';
|
||||
import { startingFrameImageChanged } from 'features/parameters/store/videoSlice';
|
||||
import { navigationApi } from 'features/ui/layouts/navigation-api';
|
||||
@@ -13,7 +14,7 @@ export const ContextMenuItemSendToVideo = memo(() => {
|
||||
const dispatch = useDispatch();
|
||||
|
||||
const onClick = useCallback(() => {
|
||||
dispatch(startingFrameImageChanged(imageDTO));
|
||||
dispatch(startingFrameImageChanged(imageDTOToCroppableImage(imageDTO)));
|
||||
navigationApi.switchToTab('video');
|
||||
}, [imageDTO, dispatch]);
|
||||
|
||||
|
||||
@@ -2,7 +2,7 @@ import { MenuItem } from '@invoke-ai/ui-library';
|
||||
import { useAppStore } from 'app/store/storeHooks';
|
||||
import { getDefaultRefImageConfig } from 'features/controlLayers/hooks/addLayerHooks';
|
||||
import { refImageAdded } from 'features/controlLayers/store/refImagesSlice';
|
||||
import { imageDTOToImageWithDims } from 'features/controlLayers/store/util';
|
||||
import { imageDTOToCroppableImage } from 'features/controlLayers/store/util';
|
||||
import { useItemDTOContextImageOnly } from 'features/gallery/contexts/ItemDTOContext';
|
||||
import { toast } from 'features/toast/toast';
|
||||
import { memo, useCallback } from 'react';
|
||||
@@ -17,7 +17,7 @@ export const ContextMenuItemUseAsRefImage = memo(() => {
|
||||
const onClickNewGlobalReferenceImageFromImage = useCallback(() => {
|
||||
const { dispatch, getState } = store;
|
||||
const config = getDefaultRefImageConfig(getState);
|
||||
config.image = imageDTOToImageWithDims(imageDTO);
|
||||
config.image = imageDTOToCroppableImage(imageDTO);
|
||||
dispatch(refImageAdded({ overrides: { config } }));
|
||||
toast({
|
||||
id: 'SENT_TO_CANVAS',
|
||||
|
||||
@@ -26,7 +26,12 @@ import type {
|
||||
CanvasRasterLayerState,
|
||||
CanvasRegionalGuidanceState,
|
||||
} from 'features/controlLayers/store/types';
|
||||
import { imageDTOToImageObject, imageDTOToImageWithDims, initialControlNet } from 'features/controlLayers/store/util';
|
||||
import {
|
||||
imageDTOToCroppableImage,
|
||||
imageDTOToImageObject,
|
||||
imageDTOToImageWithDims,
|
||||
initialControlNet,
|
||||
} from 'features/controlLayers/store/util';
|
||||
import { calculateNewSize } from 'features/controlLayers/util/getScaledBoundingBoxDimensions';
|
||||
import { imageToCompareChanged, selectionChanged } from 'features/gallery/store/gallerySlice';
|
||||
import type { BoardId } from 'features/gallery/store/types';
|
||||
@@ -44,7 +49,7 @@ import { assert } from 'tsafe';
|
||||
|
||||
export const setGlobalReferenceImage = (arg: { imageDTO: ImageDTO; id: string; dispatch: AppDispatch }) => {
|
||||
const { imageDTO, id, dispatch } = arg;
|
||||
dispatch(refImageImageChanged({ id, imageDTO }));
|
||||
dispatch(refImageImageChanged({ id, croppableImage: imageDTOToCroppableImage(imageDTO) }));
|
||||
};
|
||||
|
||||
export const setRegionalGuidanceReferenceImage = (arg: {
|
||||
|
||||
@@ -13,16 +13,18 @@ import { useAppDispatch, useAppSelector } from 'app/store/storeHooks';
|
||||
import { InformationalPopover } from 'common/components/InformationalPopover/InformationalPopover';
|
||||
import {
|
||||
buildSelectLoRA,
|
||||
DEFAULT_LORA_WEIGHT_CONFIG,
|
||||
loraDeleted,
|
||||
loraIsEnabledChanged,
|
||||
loraWeightChanged,
|
||||
} from 'features/controlLayers/store/lorasSlice';
|
||||
import type { LoRA } from 'features/controlLayers/store/types';
|
||||
import { DEFAULT_LORA_WEIGHT_CONFIG } from 'features/system/store/configSlice';
|
||||
import { memo, useCallback, useMemo } from 'react';
|
||||
import { PiTrashSimpleBold } from 'react-icons/pi';
|
||||
import { useGetModelConfigQuery } from 'services/api/endpoints/models';
|
||||
|
||||
const MARKS = [-1, 0, 1, 2];
|
||||
|
||||
export const LoRACard = memo((props: { id: string }) => {
|
||||
const selectLoRA = useMemo(() => buildSelectLoRA(props.id), [props.id]);
|
||||
const lora = useAppSelector(selectLoRA);
|
||||
@@ -81,7 +83,8 @@ const LoRAContent = memo(({ lora }: { lora: LoRA }) => {
|
||||
min={DEFAULT_LORA_WEIGHT_CONFIG.sliderMin}
|
||||
max={DEFAULT_LORA_WEIGHT_CONFIG.sliderMax}
|
||||
step={DEFAULT_LORA_WEIGHT_CONFIG.coarseStep}
|
||||
marks={DEFAULT_LORA_WEIGHT_CONFIG.marks.slice()}
|
||||
fineStep={DEFAULT_LORA_WEIGHT_CONFIG.fineStep}
|
||||
marks={MARKS}
|
||||
defaultValue={DEFAULT_LORA_WEIGHT_CONFIG.initial}
|
||||
isDisabled={!lora.isEnabled}
|
||||
/>
|
||||
@@ -91,6 +94,7 @@ const LoRAContent = memo(({ lora }: { lora: LoRA }) => {
|
||||
min={DEFAULT_LORA_WEIGHT_CONFIG.numberInputMin}
|
||||
max={DEFAULT_LORA_WEIGHT_CONFIG.numberInputMax}
|
||||
step={DEFAULT_LORA_WEIGHT_CONFIG.coarseStep}
|
||||
fineStep={DEFAULT_LORA_WEIGHT_CONFIG.fineStep}
|
||||
w={20}
|
||||
flexShrink={0}
|
||||
defaultValue={DEFAULT_LORA_WEIGHT_CONFIG.initial}
|
||||
|
||||
@@ -975,7 +975,7 @@ const RefImages: CollectionMetadataHandler<RefImageState[]> = {
|
||||
|
||||
for (const refImage of parsed) {
|
||||
if (refImage.config.image) {
|
||||
await throwIfImageDoesNotExist(refImage.config.image.image_name, store);
|
||||
await throwIfImageDoesNotExist(refImage.config.image.original.image.image_name, store);
|
||||
}
|
||||
if (refImage.config.model) {
|
||||
await throwIfModelDoesNotExist(refImage.config.model.key, store);
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
import { isNil } from 'es-toolkit/compat';
|
||||
import { DEFAULT_LORA_WEIGHT_CONFIG } from 'features/system/store/configSlice';
|
||||
import { DEFAULT_LORA_WEIGHT_CONFIG } from 'features/controlLayers/store/lorasSlice';
|
||||
import { useMemo } from 'react';
|
||||
import type { LoRAModelConfig } from 'services/api/types';
|
||||
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user